Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 1 | #!/usr/bin/perl -w |
Dave Jones | dbf004d | 2010-01-12 16:59:52 -0500 | [diff] [blame] | 2 | # (c) 2001, Dave Jones. (the file handling bit) |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3 | # (c) 2005, Joel Schopp <jschopp@austin.ibm.com> (the ugly bit) |
Andy Whitcroft | 2a5a2c2 | 2009-01-06 14:41:23 -0800 | [diff] [blame] | 4 | # (c) 2007,2008, Andy Whitcroft <apw@uk.ibm.com> (new conditions, test suite) |
Andy Whitcroft | 015830be | 2010-10-26 14:23:17 -0700 | [diff] [blame] | 5 | # (c) 2008-2010 Andy Whitcroft <apw@canonical.com> |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 6 | # Licensed under the terms of the GNU GPL License version 2 |
| 7 | |
| 8 | use strict; |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 9 | use POSIX; |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 10 | use File::Basename; |
| 11 | use Cwd 'abs_path'; |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 12 | use Term::ANSIColor qw(:constants); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 13 | |
| 14 | my $P = $0; |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 15 | my $D = dirname(abs_path($P)); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 16 | |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 17 | my $V = '0.32'; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 18 | |
| 19 | use Getopt::Long qw(:config no_auto_abbrev); |
| 20 | |
| 21 | my $quiet = 0; |
| 22 | my $tree = 1; |
| 23 | my $chk_signoff = 1; |
| 24 | my $chk_patch = 1; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 25 | my $tst_only; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 26 | my $emacs = 0; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 27 | my $terse = 0; |
Joe Perches | 34d8815 | 2015-06-25 15:03:05 -0700 | [diff] [blame] | 28 | my $showfile = 0; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 29 | my $file = 0; |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 30 | my $git = 0; |
Joe Perches | 0dea9f1e | 2016-05-20 17:04:19 -0700 | [diff] [blame] | 31 | my %git_commits = (); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 32 | my $check = 0; |
Joe Perches | 2ac73b4 | 2014-06-04 16:12:05 -0700 | [diff] [blame] | 33 | my $check_orig = 0; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 34 | my $summary = 1; |
| 35 | my $mailback = 0; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 36 | my $summary_file = 0; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 37 | my $show_types = 0; |
Joe Perches | 3beb42e | 2016-05-20 17:04:14 -0700 | [diff] [blame] | 38 | my $list_types = 0; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 39 | my $fix = 0; |
Joe Perches | 9624b8d | 2014-01-23 15:54:44 -0800 | [diff] [blame] | 40 | my $fix_inplace = 0; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 41 | my $root; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 42 | my %debug; |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 43 | my %camelcase = (); |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 44 | my %use_type = (); |
| 45 | my @use = (); |
| 46 | my %ignore_type = (); |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 47 | my @ignore = (); |
Hannes Eder | 77f5b10 | 2009-09-21 17:04:37 -0700 | [diff] [blame] | 48 | my $help = 0; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 49 | my $configuration_file = ".checkpatch.conf"; |
Joe Perches | 6cd7f38 | 2012-12-17 16:01:54 -0800 | [diff] [blame] | 50 | my $max_line_length = 80; |
Dave Hansen | d62a201 | 2013-09-11 14:23:56 -0700 | [diff] [blame] | 51 | my $ignore_perl_version = 0; |
| 52 | my $minimum_perl_version = 5.10.0; |
Vadim Bendebury | 5619327 | 2014-10-13 15:51:48 -0700 | [diff] [blame] | 53 | my $min_conf_desc_length = 4; |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 54 | my $spelling_file = "$D/spelling.txt"; |
Joe Perches | ebfd7d6 | 2015-04-16 12:44:14 -0700 | [diff] [blame] | 55 | my $codespell = 0; |
Maxim Uvarov | f1a6367 | 2015-06-25 15:03:08 -0700 | [diff] [blame] | 56 | my $codespellfile = "/usr/share/codespell/dictionary.txt"; |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 57 | my $color = 1; |
Joe Perches | dadf680 | 2016-08-02 14:04:33 -0700 | [diff] [blame] | 58 | my $allow_c99_comments = 1; |
Hannes Eder | 77f5b10 | 2009-09-21 17:04:37 -0700 | [diff] [blame] | 59 | |
| 60 | sub help { |
| 61 | my ($exitcode) = @_; |
| 62 | |
| 63 | print << "EOM"; |
| 64 | Usage: $P [OPTION]... [FILE]... |
| 65 | Version: $V |
| 66 | |
| 67 | Options: |
| 68 | -q, --quiet quiet |
| 69 | --no-tree run without a kernel tree |
| 70 | --no-signoff do not check for 'Signed-off-by' line |
| 71 | --patch treat FILE as patchfile (default) |
| 72 | --emacs emacs compile window format |
| 73 | --terse one line per report |
Joe Perches | 34d8815 | 2015-06-25 15:03:05 -0700 | [diff] [blame] | 74 | --showfile emit diffed file position, not input file position |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 75 | -g, --git treat FILE as a single commit or git revision range |
| 76 | single git commit with: |
| 77 | <rev> |
| 78 | <rev>^ |
| 79 | <rev>~n |
| 80 | multiple git commits with: |
| 81 | <rev1>..<rev2> |
| 82 | <rev1>...<rev2> |
| 83 | <rev>-<count> |
| 84 | git merges are ignored |
Hannes Eder | 77f5b10 | 2009-09-21 17:04:37 -0700 | [diff] [blame] | 85 | -f, --file treat FILE as regular source file |
| 86 | --subjective, --strict enable more subjective tests |
Joe Perches | 3beb42e | 2016-05-20 17:04:14 -0700 | [diff] [blame] | 87 | --list-types list the possible message types |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 88 | --types TYPE(,TYPE2...) show only these comma separated message types |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 89 | --ignore TYPE(,TYPE2...) ignore various comma separated message types |
Joe Perches | 3beb42e | 2016-05-20 17:04:14 -0700 | [diff] [blame] | 90 | --show-types show the specific message type in the output |
Joe Perches | 6cd7f38 | 2012-12-17 16:01:54 -0800 | [diff] [blame] | 91 | --max-line-length=n set the maximum line length, if exceeded, warn |
Vadim Bendebury | 5619327 | 2014-10-13 15:51:48 -0700 | [diff] [blame] | 92 | --min-conf-desc-length=n set the min description length, if shorter, warn |
Hannes Eder | 77f5b10 | 2009-09-21 17:04:37 -0700 | [diff] [blame] | 93 | --root=PATH PATH to the kernel tree root |
| 94 | --no-summary suppress the per-file summary |
| 95 | --mailback only produce a report in case of warnings/errors |
| 96 | --summary-file include the filename in summary |
| 97 | --debug KEY=[0|1] turn on/off debugging of KEY, where KEY is one of |
| 98 | 'values', 'possible', 'type', and 'attr' (default |
| 99 | is all off) |
| 100 | --test-only=WORD report only warnings/errors containing WORD |
| 101 | literally |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 102 | --fix EXPERIMENTAL - may create horrible results |
| 103 | If correctable single-line errors exist, create |
| 104 | "<inputfile>.EXPERIMENTAL-checkpatch-fixes" |
| 105 | with potential errors corrected to the preferred |
| 106 | checkpatch style |
Joe Perches | 9624b8d | 2014-01-23 15:54:44 -0800 | [diff] [blame] | 107 | --fix-inplace EXPERIMENTAL - may create horrible results |
| 108 | Is the same as --fix, but overwrites the input |
| 109 | file. It's your fault if there's no backup or git |
Dave Hansen | d62a201 | 2013-09-11 14:23:56 -0700 | [diff] [blame] | 110 | --ignore-perl-version override checking of perl version. expect |
| 111 | runtime errors. |
Joe Perches | ebfd7d6 | 2015-04-16 12:44:14 -0700 | [diff] [blame] | 112 | --codespell Use the codespell dictionary for spelling/typos |
Maxim Uvarov | f1a6367 | 2015-06-25 15:03:08 -0700 | [diff] [blame] | 113 | (default:/usr/share/codespell/dictionary.txt) |
Joe Perches | ebfd7d6 | 2015-04-16 12:44:14 -0700 | [diff] [blame] | 114 | --codespellfile Use this codespell dictionary |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 115 | --color Use colors when output is STDOUT (default: on) |
Hannes Eder | 77f5b10 | 2009-09-21 17:04:37 -0700 | [diff] [blame] | 116 | -h, --help, --version display this help and exit |
| 117 | |
| 118 | When FILE is - read standard input. |
| 119 | EOM |
| 120 | |
| 121 | exit($exitcode); |
| 122 | } |
| 123 | |
Joe Perches | 3beb42e | 2016-05-20 17:04:14 -0700 | [diff] [blame] | 124 | sub uniq { |
| 125 | my %seen; |
| 126 | return grep { !$seen{$_}++ } @_; |
| 127 | } |
| 128 | |
| 129 | sub list_types { |
| 130 | my ($exitcode) = @_; |
| 131 | |
| 132 | my $count = 0; |
| 133 | |
| 134 | local $/ = undef; |
| 135 | |
| 136 | open(my $script, '<', abs_path($P)) or |
| 137 | die "$P: Can't read '$P' $!\n"; |
| 138 | |
| 139 | my $text = <$script>; |
| 140 | close($script); |
| 141 | |
| 142 | my @types = (); |
| 143 | for ($text =~ /\b(?:(?:CHK|WARN|ERROR)\s*\(\s*"([^"]+)")/g) { |
| 144 | push (@types, $_); |
| 145 | } |
| 146 | @types = sort(uniq(@types)); |
| 147 | print("#\tMessage type\n\n"); |
| 148 | foreach my $type (@types) { |
| 149 | print(++$count . "\t" . $type . "\n"); |
| 150 | } |
| 151 | |
| 152 | exit($exitcode); |
| 153 | } |
| 154 | |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 155 | my $conf = which_conf($configuration_file); |
| 156 | if (-f $conf) { |
| 157 | my @conf_args; |
| 158 | open(my $conffile, '<', "$conf") |
| 159 | or warn "$P: Can't find a readable $configuration_file file $!\n"; |
| 160 | |
| 161 | while (<$conffile>) { |
| 162 | my $line = $_; |
| 163 | |
| 164 | $line =~ s/\s*\n?$//g; |
| 165 | $line =~ s/^\s*//g; |
| 166 | $line =~ s/\s+/ /g; |
| 167 | |
| 168 | next if ($line =~ m/^\s*#/); |
| 169 | next if ($line =~ m/^\s*$/); |
| 170 | |
| 171 | my @words = split(" ", $line); |
| 172 | foreach my $word (@words) { |
| 173 | last if ($word =~ m/^#/); |
| 174 | push (@conf_args, $word); |
| 175 | } |
| 176 | } |
| 177 | close($conffile); |
| 178 | unshift(@ARGV, @conf_args) if @conf_args; |
| 179 | } |
| 180 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 181 | GetOptions( |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 182 | 'q|quiet+' => \$quiet, |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 183 | 'tree!' => \$tree, |
| 184 | 'signoff!' => \$chk_signoff, |
| 185 | 'patch!' => \$chk_patch, |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 186 | 'emacs!' => \$emacs, |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 187 | 'terse!' => \$terse, |
Joe Perches | 34d8815 | 2015-06-25 15:03:05 -0700 | [diff] [blame] | 188 | 'showfile!' => \$showfile, |
Hannes Eder | 77f5b10 | 2009-09-21 17:04:37 -0700 | [diff] [blame] | 189 | 'f|file!' => \$file, |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 190 | 'g|git!' => \$git, |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 191 | 'subjective!' => \$check, |
| 192 | 'strict!' => \$check, |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 193 | 'ignore=s' => \@ignore, |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 194 | 'types=s' => \@use, |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 195 | 'show-types!' => \$show_types, |
Joe Perches | 3beb42e | 2016-05-20 17:04:14 -0700 | [diff] [blame] | 196 | 'list-types!' => \$list_types, |
Joe Perches | 6cd7f38 | 2012-12-17 16:01:54 -0800 | [diff] [blame] | 197 | 'max-line-length=i' => \$max_line_length, |
Vadim Bendebury | 5619327 | 2014-10-13 15:51:48 -0700 | [diff] [blame] | 198 | 'min-conf-desc-length=i' => \$min_conf_desc_length, |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 199 | 'root=s' => \$root, |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 200 | 'summary!' => \$summary, |
| 201 | 'mailback!' => \$mailback, |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 202 | 'summary-file!' => \$summary_file, |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 203 | 'fix!' => \$fix, |
Joe Perches | 9624b8d | 2014-01-23 15:54:44 -0800 | [diff] [blame] | 204 | 'fix-inplace!' => \$fix_inplace, |
Dave Hansen | d62a201 | 2013-09-11 14:23:56 -0700 | [diff] [blame] | 205 | 'ignore-perl-version!' => \$ignore_perl_version, |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 206 | 'debug=s' => \%debug, |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 207 | 'test-only=s' => \$tst_only, |
Joe Perches | ebfd7d6 | 2015-04-16 12:44:14 -0700 | [diff] [blame] | 208 | 'codespell!' => \$codespell, |
| 209 | 'codespellfile=s' => \$codespellfile, |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 210 | 'color!' => \$color, |
Hannes Eder | 77f5b10 | 2009-09-21 17:04:37 -0700 | [diff] [blame] | 211 | 'h|help' => \$help, |
| 212 | 'version' => \$help |
| 213 | ) or help(1); |
| 214 | |
| 215 | help(0) if ($help); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 216 | |
Joe Perches | 3beb42e | 2016-05-20 17:04:14 -0700 | [diff] [blame] | 217 | list_types(0) if ($list_types); |
| 218 | |
Joe Perches | 9624b8d | 2014-01-23 15:54:44 -0800 | [diff] [blame] | 219 | $fix = 1 if ($fix_inplace); |
Joe Perches | 2ac73b4 | 2014-06-04 16:12:05 -0700 | [diff] [blame] | 220 | $check_orig = $check; |
Joe Perches | 9624b8d | 2014-01-23 15:54:44 -0800 | [diff] [blame] | 221 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 222 | my $exit = 0; |
| 223 | |
Dave Hansen | d62a201 | 2013-09-11 14:23:56 -0700 | [diff] [blame] | 224 | if ($^V && $^V lt $minimum_perl_version) { |
| 225 | printf "$P: requires at least perl version %vd\n", $minimum_perl_version; |
| 226 | if (!$ignore_perl_version) { |
| 227 | exit(1); |
| 228 | } |
| 229 | } |
| 230 | |
Allen Hubbe | 45107ff | 2016-08-02 14:04:47 -0700 | [diff] [blame] | 231 | #if no filenames are given, push '-' to read patch from stdin |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 232 | if ($#ARGV < 0) { |
Allen Hubbe | 45107ff | 2016-08-02 14:04:47 -0700 | [diff] [blame] | 233 | push(@ARGV, '-'); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 234 | } |
| 235 | |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 236 | sub hash_save_array_words { |
| 237 | my ($hashRef, $arrayRef) = @_; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 238 | |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 239 | my @array = split(/,/, join(',', @$arrayRef)); |
| 240 | foreach my $word (@array) { |
| 241 | $word =~ s/\s*\n?$//g; |
| 242 | $word =~ s/^\s*//g; |
| 243 | $word =~ s/\s+/ /g; |
| 244 | $word =~ tr/[a-z]/[A-Z]/; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 245 | |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 246 | next if ($word =~ m/^\s*#/); |
| 247 | next if ($word =~ m/^\s*$/); |
| 248 | |
| 249 | $hashRef->{$word}++; |
| 250 | } |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 251 | } |
| 252 | |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 253 | sub hash_show_words { |
| 254 | my ($hashRef, $prefix) = @_; |
| 255 | |
Joe Perches | 3c816e4 | 2015-06-25 15:03:29 -0700 | [diff] [blame] | 256 | if (keys %$hashRef) { |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 257 | print "\nNOTE: $prefix message types:"; |
Joe Perches | 58cb3cf | 2013-09-11 14:24:04 -0700 | [diff] [blame] | 258 | foreach my $word (sort keys %$hashRef) { |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 259 | print " $word"; |
| 260 | } |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 261 | print "\n"; |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 262 | } |
| 263 | } |
| 264 | |
| 265 | hash_save_array_words(\%ignore_type, \@ignore); |
| 266 | hash_save_array_words(\%use_type, \@use); |
| 267 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 268 | my $dbg_values = 0; |
| 269 | my $dbg_possible = 0; |
Andy Whitcroft | 7429c69 | 2008-07-23 21:29:06 -0700 | [diff] [blame] | 270 | my $dbg_type = 0; |
Andy Whitcroft | a1ef277 | 2008-10-15 22:02:17 -0700 | [diff] [blame] | 271 | my $dbg_attr = 0; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 272 | for my $key (keys %debug) { |
Andy Whitcroft | 21caa13 | 2009-01-06 14:41:30 -0800 | [diff] [blame] | 273 | ## no critic |
| 274 | eval "\${dbg_$key} = '$debug{$key}';"; |
| 275 | die "$@" if ($@); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 276 | } |
| 277 | |
Andy Whitcroft | d2c0a23 | 2010-10-26 14:23:12 -0700 | [diff] [blame] | 278 | my $rpt_cleaners = 0; |
| 279 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 280 | if ($terse) { |
| 281 | $emacs = 1; |
| 282 | $quiet++; |
| 283 | } |
| 284 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 285 | if ($tree) { |
| 286 | if (defined $root) { |
| 287 | if (!top_of_kernel_tree($root)) { |
| 288 | die "$P: $root: --root does not point at a valid tree\n"; |
| 289 | } |
| 290 | } else { |
| 291 | if (top_of_kernel_tree('.')) { |
| 292 | $root = '.'; |
| 293 | } elsif ($0 =~ m@(.*)/scripts/[^/]*$@ && |
| 294 | top_of_kernel_tree($1)) { |
| 295 | $root = $1; |
| 296 | } |
| 297 | } |
| 298 | |
| 299 | if (!defined $root) { |
| 300 | print "Must be run from the top-level dir. of a kernel tree\n"; |
| 301 | exit(2); |
| 302 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 303 | } |
| 304 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 305 | my $emitted_corrupt = 0; |
| 306 | |
Andy Whitcroft | 2ceb532 | 2009-10-26 16:50:14 -0700 | [diff] [blame] | 307 | our $Ident = qr{ |
| 308 | [A-Za-z_][A-Za-z\d_]* |
| 309 | (?:\s*\#\#\s*[A-Za-z_][A-Za-z\d_]*)* |
| 310 | }x; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 311 | our $Storage = qr{extern|static|asmlinkage}; |
| 312 | our $Sparse = qr{ |
| 313 | __user| |
| 314 | __kernel| |
| 315 | __force| |
| 316 | __iomem| |
| 317 | __must_check| |
| 318 | __init_refok| |
Andy Whitcroft | 417495e | 2009-02-27 14:03:08 -0800 | [diff] [blame] | 319 | __kprobes| |
Sven Eckelmann | 165e72a | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 320 | __ref| |
Boqun Feng | ad31545 | 2015-12-29 12:18:46 +0800 | [diff] [blame] | 321 | __rcu| |
| 322 | __private |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 323 | }x; |
Joe Perches | e970b884 | 2013-11-12 15:10:10 -0800 | [diff] [blame] | 324 | our $InitAttributePrefix = qr{__(?:mem|cpu|dev|net_|)}; |
| 325 | our $InitAttributeData = qr{$InitAttributePrefix(?:initdata\b)}; |
| 326 | our $InitAttributeConst = qr{$InitAttributePrefix(?:initconst\b)}; |
| 327 | our $InitAttributeInit = qr{$InitAttributePrefix(?:init\b)}; |
| 328 | our $InitAttribute = qr{$InitAttributeData|$InitAttributeConst|$InitAttributeInit}; |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 329 | |
Wolfram Sang | 5213129 | 2010-03-05 13:43:51 -0800 | [diff] [blame] | 330 | # Notes to $Attribute: |
| 331 | # We need \b after 'init' otherwise 'initconst' will cause a false positive in a check |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 332 | our $Attribute = qr{ |
| 333 | const| |
Joe Perches | 03f1df7 | 2010-10-26 14:23:16 -0700 | [diff] [blame] | 334 | __percpu| |
| 335 | __nocast| |
| 336 | __safe| |
| 337 | __bitwise__| |
| 338 | __packed__| |
| 339 | __packed2__| |
| 340 | __naked| |
| 341 | __maybe_unused| |
| 342 | __always_unused| |
| 343 | __noreturn| |
| 344 | __used| |
| 345 | __cold| |
Joe Perches | e23ef1f | 2015-02-13 14:38:24 -0800 | [diff] [blame] | 346 | __pure| |
Joe Perches | 03f1df7 | 2010-10-26 14:23:16 -0700 | [diff] [blame] | 347 | __noclone| |
| 348 | __deprecated| |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 349 | __read_mostly| |
| 350 | __kprobes| |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 351 | $InitAttribute| |
Andy Whitcroft | 24e1d81 | 2008-10-15 22:02:18 -0700 | [diff] [blame] | 352 | ____cacheline_aligned| |
| 353 | ____cacheline_aligned_in_smp| |
Andy Whitcroft | 5fe3af1 | 2009-01-06 14:41:18 -0800 | [diff] [blame] | 354 | ____cacheline_internodealigned_in_smp| |
| 355 | __weak |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 356 | }x; |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 357 | our $Modifier; |
Joe Perches | 91cb519 | 2014-04-03 14:49:32 -0700 | [diff] [blame] | 358 | our $Inline = qr{inline|__always_inline|noinline|__inline|__inline__}; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 359 | our $Member = qr{->$Ident|\.$Ident|\[[^]]*\]}; |
| 360 | our $Lval = qr{$Ident(?:$Member)*}; |
| 361 | |
Joe Perches | 95e2c60 | 2013-07-03 15:05:20 -0700 | [diff] [blame] | 362 | our $Int_type = qr{(?i)llu|ull|ll|lu|ul|l|u}; |
| 363 | our $Binary = qr{(?i)0b[01]+$Int_type?}; |
| 364 | our $Hex = qr{(?i)0x[0-9a-f]+$Int_type?}; |
| 365 | our $Int = qr{[0-9]+$Int_type?}; |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 366 | our $Octal = qr{0[0-7]+$Int_type?}; |
Joe Perches | c0a5c89 | 2015-02-13 14:38:21 -0800 | [diff] [blame] | 367 | our $String = qr{"[X\t]*"}; |
Joe Perches | 326b1ff | 2013-02-04 14:28:51 -0800 | [diff] [blame] | 368 | our $Float_hex = qr{(?i)0x[0-9a-f]+p-?[0-9]+[fl]?}; |
| 369 | our $Float_dec = qr{(?i)(?:[0-9]+\.[0-9]*|[0-9]*\.[0-9]+)(?:e-?[0-9]+)?[fl]?}; |
| 370 | our $Float_int = qr{(?i)[0-9]+e-?[0-9]+[fl]?}; |
Joe Perches | 74349bc | 2012-12-17 16:02:05 -0800 | [diff] [blame] | 371 | our $Float = qr{$Float_hex|$Float_dec|$Float_int}; |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 372 | our $Constant = qr{$Float|$Binary|$Octal|$Hex|$Int}; |
Joe Perches | 326b1ff | 2013-02-04 14:28:51 -0800 | [diff] [blame] | 373 | our $Assignment = qr{\*\=|/=|%=|\+=|-=|<<=|>>=|&=|\^=|\|=|=}; |
Joe Perches | 447432f | 2014-04-03 14:49:17 -0700 | [diff] [blame] | 374 | our $Compare = qr{<=|>=|==|!=|<|(?<!-)>}; |
Joe Perches | 23f780c | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 375 | our $Arithmetic = qr{\+|-|\*|\/|%}; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 376 | our $Operators = qr{ |
| 377 | <=|>=|==|!=| |
| 378 | =>|->|<<|>>|<|>|!|~| |
Joe Perches | 23f780c | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 379 | &&|\|\||,|\^|\+\+|--|&|\||$Arithmetic |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 380 | }x; |
| 381 | |
Joe Perches | 91cb519 | 2014-04-03 14:49:32 -0700 | [diff] [blame] | 382 | our $c90_Keywords = qr{do|for|while|if|else|return|goto|continue|switch|default|case|break}x; |
| 383 | |
Joe Perches | ab7e23f | 2015-04-16 12:44:22 -0700 | [diff] [blame] | 384 | our $BasicType; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 385 | our $NonptrType; |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 386 | our $NonptrTypeMisordered; |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 387 | our $NonptrTypeWithAttr; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 388 | our $Type; |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 389 | our $TypeMisordered; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 390 | our $Declare; |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 391 | our $DeclareMisordered; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 392 | |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 393 | our $NON_ASCII_UTF8 = qr{ |
| 394 | [\xC2-\xDF][\x80-\xBF] # non-overlong 2-byte |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 395 | | \xE0[\xA0-\xBF][\x80-\xBF] # excluding overlongs |
| 396 | | [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2} # straight 3-byte |
| 397 | | \xED[\x80-\x9F][\x80-\xBF] # excluding surrogates |
| 398 | | \xF0[\x90-\xBF][\x80-\xBF]{2} # planes 1-3 |
| 399 | | [\xF1-\xF3][\x80-\xBF]{3} # planes 4-15 |
| 400 | | \xF4[\x80-\x8F][\x80-\xBF]{2} # plane 16 |
| 401 | }x; |
| 402 | |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 403 | our $UTF8 = qr{ |
| 404 | [\x09\x0A\x0D\x20-\x7E] # ASCII |
| 405 | | $NON_ASCII_UTF8 |
| 406 | }x; |
| 407 | |
Joe Perches | e6176fa | 2015-06-25 15:02:49 -0700 | [diff] [blame] | 408 | our $typeC99Typedefs = qr{(?:__)?(?:[us]_?)?int_?(?:8|16|32|64)_t}; |
Joe Perches | 021158b | 2015-02-13 14:38:43 -0800 | [diff] [blame] | 409 | our $typeOtherOSTypedefs = qr{(?x: |
| 410 | u_(?:char|short|int|long) | # bsd |
| 411 | u(?:nchar|short|int|long) # sysv |
| 412 | )}; |
Joe Perches | e6176fa | 2015-06-25 15:02:49 -0700 | [diff] [blame] | 413 | our $typeKernelTypedefs = qr{(?x: |
Andy Whitcroft | fb9e909 | 2009-09-21 17:04:38 -0700 | [diff] [blame] | 414 | (?:__)?(?:u|s|be|le)(?:8|16|32|64)| |
Andy Whitcroft | 8ed22ca | 2008-10-15 22:02:32 -0700 | [diff] [blame] | 415 | atomic_t |
| 416 | )}; |
Joe Perches | e6176fa | 2015-06-25 15:02:49 -0700 | [diff] [blame] | 417 | our $typeTypedefs = qr{(?x: |
| 418 | $typeC99Typedefs\b| |
| 419 | $typeOtherOSTypedefs\b| |
| 420 | $typeKernelTypedefs\b |
| 421 | )}; |
Andy Whitcroft | 8ed22ca | 2008-10-15 22:02:32 -0700 | [diff] [blame] | 422 | |
Joe Perches | 6d32f7a | 2015-11-06 16:31:37 -0800 | [diff] [blame] | 423 | our $zero_initializer = qr{(?:(?:0[xX])?0+$Int_type?|NULL|false)\b}; |
| 424 | |
Joe Perches | 691e669 | 2010-03-05 13:43:51 -0800 | [diff] [blame] | 425 | our $logFunctions = qr{(?x: |
Joe Perches | 6e60c02 | 2011-07-25 17:13:27 -0700 | [diff] [blame] | 426 | printk(?:_ratelimited|_once|)| |
Jacob Keller | 7d0b659 | 2013-07-03 15:05:35 -0700 | [diff] [blame] | 427 | (?:[a-z0-9]+_){1,2}(?:printk|emerg|alert|crit|err|warning|warn|notice|info|debug|dbg|vdbg|devel|cont|WARN)(?:_ratelimited|_once|)| |
Joe Perches | 6e60c02 | 2011-07-25 17:13:27 -0700 | [diff] [blame] | 428 | WARN(?:_RATELIMIT|_ONCE|)| |
Joe Perches | b053172 | 2011-05-24 17:13:40 -0700 | [diff] [blame] | 429 | panic| |
Joe Perches | 0666872 | 2013-11-12 15:10:07 -0800 | [diff] [blame] | 430 | MODULE_[A-Z_]+| |
| 431 | seq_vprintf|seq_printf|seq_puts |
Joe Perches | 691e669 | 2010-03-05 13:43:51 -0800 | [diff] [blame] | 432 | )}; |
| 433 | |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 434 | our $signature_tags = qr{(?xi: |
| 435 | Signed-off-by:| |
| 436 | Acked-by:| |
| 437 | Tested-by:| |
| 438 | Reviewed-by:| |
| 439 | Reported-by:| |
Mugunthan V N | 8543ae1 | 2013-04-29 16:18:17 -0700 | [diff] [blame] | 440 | Suggested-by:| |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 441 | To:| |
| 442 | Cc: |
| 443 | )}; |
| 444 | |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 445 | our @typeListMisordered = ( |
| 446 | qr{char\s+(?:un)?signed}, |
| 447 | qr{int\s+(?:(?:un)?signed\s+)?short\s}, |
| 448 | qr{int\s+short(?:\s+(?:un)?signed)}, |
| 449 | qr{short\s+int(?:\s+(?:un)?signed)}, |
| 450 | qr{(?:un)?signed\s+int\s+short}, |
| 451 | qr{short\s+(?:un)?signed}, |
| 452 | qr{long\s+int\s+(?:un)?signed}, |
| 453 | qr{int\s+long\s+(?:un)?signed}, |
| 454 | qr{long\s+(?:un)?signed\s+int}, |
| 455 | qr{int\s+(?:un)?signed\s+long}, |
| 456 | qr{int\s+(?:un)?signed}, |
| 457 | qr{int\s+long\s+long\s+(?:un)?signed}, |
| 458 | qr{long\s+long\s+int\s+(?:un)?signed}, |
| 459 | qr{long\s+long\s+(?:un)?signed\s+int}, |
| 460 | qr{long\s+long\s+(?:un)?signed}, |
| 461 | qr{long\s+(?:un)?signed}, |
| 462 | ); |
| 463 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 464 | our @typeList = ( |
| 465 | qr{void}, |
Joe Perches | 0c773d9 | 2014-08-06 16:11:20 -0700 | [diff] [blame] | 466 | qr{(?:(?:un)?signed\s+)?char}, |
| 467 | qr{(?:(?:un)?signed\s+)?short\s+int}, |
| 468 | qr{(?:(?:un)?signed\s+)?short}, |
| 469 | qr{(?:(?:un)?signed\s+)?int}, |
| 470 | qr{(?:(?:un)?signed\s+)?long\s+int}, |
| 471 | qr{(?:(?:un)?signed\s+)?long\s+long\s+int}, |
| 472 | qr{(?:(?:un)?signed\s+)?long\s+long}, |
| 473 | qr{(?:(?:un)?signed\s+)?long}, |
| 474 | qr{(?:un)?signed}, |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 475 | qr{float}, |
| 476 | qr{double}, |
| 477 | qr{bool}, |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 478 | qr{struct\s+$Ident}, |
| 479 | qr{union\s+$Ident}, |
| 480 | qr{enum\s+$Ident}, |
| 481 | qr{${Ident}_t}, |
| 482 | qr{${Ident}_handler}, |
| 483 | qr{${Ident}_handler_fn}, |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 484 | @typeListMisordered, |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 485 | ); |
Joe Perches | 938224b | 2016-01-20 14:59:15 -0800 | [diff] [blame] | 486 | |
| 487 | our $C90_int_types = qr{(?x: |
| 488 | long\s+long\s+int\s+(?:un)?signed| |
| 489 | long\s+long\s+(?:un)?signed\s+int| |
| 490 | long\s+long\s+(?:un)?signed| |
| 491 | (?:(?:un)?signed\s+)?long\s+long\s+int| |
| 492 | (?:(?:un)?signed\s+)?long\s+long| |
| 493 | int\s+long\s+long\s+(?:un)?signed| |
| 494 | int\s+(?:(?:un)?signed\s+)?long\s+long| |
| 495 | |
| 496 | long\s+int\s+(?:un)?signed| |
| 497 | long\s+(?:un)?signed\s+int| |
| 498 | long\s+(?:un)?signed| |
| 499 | (?:(?:un)?signed\s+)?long\s+int| |
| 500 | (?:(?:un)?signed\s+)?long| |
| 501 | int\s+long\s+(?:un)?signed| |
| 502 | int\s+(?:(?:un)?signed\s+)?long| |
| 503 | |
| 504 | int\s+(?:un)?signed| |
| 505 | (?:(?:un)?signed\s+)?int |
| 506 | )}; |
| 507 | |
Alex Dowad | 485ff23 | 2015-06-25 15:02:52 -0700 | [diff] [blame] | 508 | our @typeListFile = (); |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 509 | our @typeListWithAttr = ( |
| 510 | @typeList, |
| 511 | qr{struct\s+$InitAttribute\s+$Ident}, |
| 512 | qr{union\s+$InitAttribute\s+$Ident}, |
| 513 | ); |
| 514 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 515 | our @modifierList = ( |
| 516 | qr{fastcall}, |
| 517 | ); |
Alex Dowad | 485ff23 | 2015-06-25 15:02:52 -0700 | [diff] [blame] | 518 | our @modifierListFile = (); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 519 | |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 520 | our @mode_permission_funcs = ( |
| 521 | ["module_param", 3], |
| 522 | ["module_param_(?:array|named|string)", 4], |
| 523 | ["module_param_array_named", 5], |
| 524 | ["debugfs_create_(?:file|u8|u16|u32|u64|x8|x16|x32|x64|size_t|atomic_t|bool|blob|regset32|u32_array)", 2], |
| 525 | ["proc_create(?:_data|)", 2], |
| 526 | ["(?:CLASS|DEVICE|SENSOR)_ATTR", 2], |
| 527 | ); |
| 528 | |
Joe Perches | 515a235 | 2014-04-03 14:49:24 -0700 | [diff] [blame] | 529 | #Create a search pattern for all these functions to speed up a loop below |
| 530 | our $mode_perms_search = ""; |
| 531 | foreach my $entry (@mode_permission_funcs) { |
| 532 | $mode_perms_search .= '|' if ($mode_perms_search ne ""); |
| 533 | $mode_perms_search .= $entry->[0]; |
| 534 | } |
| 535 | |
Joe Perches | b392c64 | 2015-04-16 12:44:16 -0700 | [diff] [blame] | 536 | our $mode_perms_world_writable = qr{ |
| 537 | S_IWUGO | |
| 538 | S_IWOTH | |
| 539 | S_IRWXUGO | |
| 540 | S_IALLUGO | |
| 541 | 0[0-7][0-7][2367] |
| 542 | }x; |
| 543 | |
Wolfram Sang | 7840a94 | 2010-08-09 17:20:57 -0700 | [diff] [blame] | 544 | our $allowed_asm_includes = qr{(?x: |
| 545 | irq| |
Sergey Ryazanov | cdcee68 | 2014-10-13 15:51:44 -0700 | [diff] [blame] | 546 | memory| |
| 547 | time| |
| 548 | reboot |
Wolfram Sang | 7840a94 | 2010-08-09 17:20:57 -0700 | [diff] [blame] | 549 | )}; |
| 550 | # memory.h: ARM has a custom one |
| 551 | |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 552 | # Load common spelling mistakes and build regular expression list. |
| 553 | my $misspellings; |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 554 | my %spelling_fix; |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 555 | |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 556 | if (open(my $spelling, '<', $spelling_file)) { |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 557 | while (<$spelling>) { |
| 558 | my $line = $_; |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 559 | |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 560 | $line =~ s/\s*\n?$//g; |
| 561 | $line =~ s/^\s*//g; |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 562 | |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 563 | next if ($line =~ m/^\s*#/); |
| 564 | next if ($line =~ m/^\s*$/); |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 565 | |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 566 | my ($suspect, $fix) = split(/\|\|/, $line); |
| 567 | |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 568 | $spelling_fix{$suspect} = $fix; |
| 569 | } |
| 570 | close($spelling); |
Joe Perches | 36061e3 | 2014-12-10 15:51:43 -0800 | [diff] [blame] | 571 | } else { |
| 572 | warn "No typos will be found - file '$spelling_file': $!\n"; |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 573 | } |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 574 | |
Joe Perches | ebfd7d6 | 2015-04-16 12:44:14 -0700 | [diff] [blame] | 575 | if ($codespell) { |
| 576 | if (open(my $spelling, '<', $codespellfile)) { |
| 577 | while (<$spelling>) { |
| 578 | my $line = $_; |
| 579 | |
| 580 | $line =~ s/\s*\n?$//g; |
| 581 | $line =~ s/^\s*//g; |
| 582 | |
| 583 | next if ($line =~ m/^\s*#/); |
| 584 | next if ($line =~ m/^\s*$/); |
| 585 | next if ($line =~ m/, disabled/i); |
| 586 | |
| 587 | $line =~ s/,.*$//; |
| 588 | |
| 589 | my ($suspect, $fix) = split(/->/, $line); |
| 590 | |
| 591 | $spelling_fix{$suspect} = $fix; |
| 592 | } |
| 593 | close($spelling); |
| 594 | } else { |
| 595 | warn "No codespell typos will be found - file '$codespellfile': $!\n"; |
| 596 | } |
| 597 | } |
| 598 | |
| 599 | $misspellings = join("|", sort keys %spelling_fix) if keys %spelling_fix; |
| 600 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 601 | sub build_types { |
Alex Dowad | 485ff23 | 2015-06-25 15:02:52 -0700 | [diff] [blame] | 602 | my $mods = "(?x: \n" . join("|\n ", (@modifierList, @modifierListFile)) . "\n)"; |
| 603 | my $all = "(?x: \n" . join("|\n ", (@typeList, @typeListFile)) . "\n)"; |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 604 | my $Misordered = "(?x: \n" . join("|\n ", @typeListMisordered) . "\n)"; |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 605 | my $allWithAttr = "(?x: \n" . join("|\n ", @typeListWithAttr) . "\n)"; |
Andy Whitcroft | c8cb2ca | 2008-07-23 21:28:57 -0700 | [diff] [blame] | 606 | $Modifier = qr{(?:$Attribute|$Sparse|$mods)}; |
Joe Perches | ab7e23f | 2015-04-16 12:44:22 -0700 | [diff] [blame] | 607 | $BasicType = qr{ |
Joe Perches | ab7e23f | 2015-04-16 12:44:22 -0700 | [diff] [blame] | 608 | (?:$typeTypedefs\b)| |
| 609 | (?:${all}\b) |
| 610 | }x; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 611 | $NonptrType = qr{ |
Andy Whitcroft | d2172eb | 2008-07-23 21:29:07 -0700 | [diff] [blame] | 612 | (?:$Modifier\s+|const\s+)* |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 613 | (?: |
Andy Whitcroft | 6b48db2 | 2012-01-10 15:10:13 -0800 | [diff] [blame] | 614 | (?:typeof|__typeof__)\s*\([^\)]*\)| |
Andy Whitcroft | 8ed22ca | 2008-10-15 22:02:32 -0700 | [diff] [blame] | 615 | (?:$typeTypedefs\b)| |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 616 | (?:${all}\b) |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 617 | ) |
Andy Whitcroft | c8cb2ca | 2008-07-23 21:28:57 -0700 | [diff] [blame] | 618 | (?:\s+$Modifier|\s+const)* |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 619 | }x; |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 620 | $NonptrTypeMisordered = qr{ |
| 621 | (?:$Modifier\s+|const\s+)* |
| 622 | (?: |
| 623 | (?:${Misordered}\b) |
| 624 | ) |
| 625 | (?:\s+$Modifier|\s+const)* |
| 626 | }x; |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 627 | $NonptrTypeWithAttr = qr{ |
| 628 | (?:$Modifier\s+|const\s+)* |
| 629 | (?: |
| 630 | (?:typeof|__typeof__)\s*\([^\)]*\)| |
| 631 | (?:$typeTypedefs\b)| |
| 632 | (?:${allWithAttr}\b) |
| 633 | ) |
| 634 | (?:\s+$Modifier|\s+const)* |
| 635 | }x; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 636 | $Type = qr{ |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 637 | $NonptrType |
Joe Perches | 1574a29 | 2014-08-06 16:10:50 -0700 | [diff] [blame] | 638 | (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? |
Andy Whitcroft | c8cb2ca | 2008-07-23 21:28:57 -0700 | [diff] [blame] | 639 | (?:\s+$Inline|\s+$Modifier)* |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 640 | }x; |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 641 | $TypeMisordered = qr{ |
| 642 | $NonptrTypeMisordered |
| 643 | (?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)? |
| 644 | (?:\s+$Inline|\s+$Modifier)* |
| 645 | }x; |
Joe Perches | 91cb519 | 2014-04-03 14:49:32 -0700 | [diff] [blame] | 646 | $Declare = qr{(?:$Storage\s+(?:$Inline\s+)?)?$Type}; |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 647 | $DeclareMisordered = qr{(?:$Storage\s+(?:$Inline\s+)?)?$TypeMisordered}; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 648 | } |
| 649 | build_types(); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 650 | |
Joe Perches | 7d2367a | 2011-07-25 17:13:22 -0700 | [diff] [blame] | 651 | our $Typecast = qr{\s*(\(\s*$NonptrType\s*\)){0,1}\s*}; |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 652 | |
| 653 | # Using $balanced_parens, $LvalOrFunc, or $FuncArg |
| 654 | # requires at least perl version v5.10.0 |
| 655 | # Any use must be runtime checked with $^V |
| 656 | |
| 657 | our $balanced_parens = qr/(\((?:[^\(\)]++|(?-1))*\))/; |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 658 | our $LvalOrFunc = qr{((?:[\&\*]\s*)?$Lval)\s*($balanced_parens{0,1})\s*}; |
Joe Perches | c0a5c89 | 2015-02-13 14:38:21 -0800 | [diff] [blame] | 659 | our $FuncArg = qr{$Typecast{0,1}($LvalOrFunc|$Constant|$String)}; |
Joe Perches | 7d2367a | 2011-07-25 17:13:22 -0700 | [diff] [blame] | 660 | |
Joe Perches | f842230 | 2014-08-06 16:11:31 -0700 | [diff] [blame] | 661 | our $declaration_macros = qr{(?x: |
Joe Perches | 3e838b6 | 2015-09-09 15:37:33 -0700 | [diff] [blame] | 662 | (?:$Storage\s+)?(?:[A-Z_][A-Z0-9]*_){0,2}(?:DEFINE|DECLARE)(?:_[A-Z0-9]+){1,6}\s*\(| |
Joe Perches | f842230 | 2014-08-06 16:11:31 -0700 | [diff] [blame] | 663 | (?:$Storage\s+)?LIST_HEAD\s*\(| |
| 664 | (?:$Storage\s+)?${Type}\s+uninitialized_var\s*\( |
| 665 | )}; |
| 666 | |
Joe Perches | 7d2367a | 2011-07-25 17:13:22 -0700 | [diff] [blame] | 667 | sub deparenthesize { |
| 668 | my ($string) = @_; |
| 669 | return "" if (!defined($string)); |
Joe Perches | 5b9553a | 2014-04-03 14:49:21 -0700 | [diff] [blame] | 670 | |
| 671 | while ($string =~ /^\s*\(.*\)\s*$/) { |
| 672 | $string =~ s@^\s*\(\s*@@; |
| 673 | $string =~ s@\s*\)\s*$@@; |
| 674 | } |
| 675 | |
Joe Perches | 7d2367a | 2011-07-25 17:13:22 -0700 | [diff] [blame] | 676 | $string =~ s@\s+@ @g; |
Joe Perches | 5b9553a | 2014-04-03 14:49:21 -0700 | [diff] [blame] | 677 | |
Joe Perches | 7d2367a | 2011-07-25 17:13:22 -0700 | [diff] [blame] | 678 | return $string; |
| 679 | } |
| 680 | |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 681 | sub seed_camelcase_file { |
| 682 | my ($file) = @_; |
| 683 | |
| 684 | return if (!(-f $file)); |
| 685 | |
| 686 | local $/; |
| 687 | |
| 688 | open(my $include_file, '<', "$file") |
| 689 | or warn "$P: Can't read '$file' $!\n"; |
| 690 | my $text = <$include_file>; |
| 691 | close($include_file); |
| 692 | |
| 693 | my @lines = split('\n', $text); |
| 694 | |
| 695 | foreach my $line (@lines) { |
| 696 | next if ($line !~ /(?:[A-Z][a-z]|[a-z][A-Z])/); |
| 697 | if ($line =~ /^[ \t]*(?:#[ \t]*define|typedef\s+$Type)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)/) { |
| 698 | $camelcase{$1} = 1; |
Joe Perches | 11ea516 | 2013-11-12 15:10:08 -0800 | [diff] [blame] | 699 | } elsif ($line =~ /^\s*$Declare\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[\(\[,;]/) { |
| 700 | $camelcase{$1} = 1; |
| 701 | } elsif ($line =~ /^\s*(?:union|struct|enum)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[;\{]/) { |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 702 | $camelcase{$1} = 1; |
| 703 | } |
| 704 | } |
| 705 | } |
| 706 | |
| 707 | my $camelcase_seeded = 0; |
| 708 | sub seed_camelcase_includes { |
| 709 | return if ($camelcase_seeded); |
| 710 | |
| 711 | my $files; |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 712 | my $camelcase_cache = ""; |
| 713 | my @include_files = (); |
| 714 | |
| 715 | $camelcase_seeded = 1; |
Joe Perches | 351b2a1 | 2013-07-03 15:05:36 -0700 | [diff] [blame] | 716 | |
Richard Genoud | 3645e32 | 2014-02-10 14:25:32 -0800 | [diff] [blame] | 717 | if (-e ".git") { |
Joe Perches | 351b2a1 | 2013-07-03 15:05:36 -0700 | [diff] [blame] | 718 | my $git_last_include_commit = `git log --no-merges --pretty=format:"%h%n" -1 -- include`; |
| 719 | chomp $git_last_include_commit; |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 720 | $camelcase_cache = ".checkpatch-camelcase.git.$git_last_include_commit"; |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 721 | } else { |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 722 | my $last_mod_date = 0; |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 723 | $files = `find $root/include -name "*.h"`; |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 724 | @include_files = split('\n', $files); |
| 725 | foreach my $file (@include_files) { |
| 726 | my $date = POSIX::strftime("%Y%m%d%H%M", |
| 727 | localtime((stat $file)[9])); |
| 728 | $last_mod_date = $date if ($last_mod_date < $date); |
| 729 | } |
| 730 | $camelcase_cache = ".checkpatch-camelcase.date.$last_mod_date"; |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 731 | } |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 732 | |
| 733 | if ($camelcase_cache ne "" && -f $camelcase_cache) { |
| 734 | open(my $camelcase_file, '<', "$camelcase_cache") |
| 735 | or warn "$P: Can't read '$camelcase_cache' $!\n"; |
| 736 | while (<$camelcase_file>) { |
| 737 | chomp; |
| 738 | $camelcase{$_} = 1; |
| 739 | } |
| 740 | close($camelcase_file); |
| 741 | |
| 742 | return; |
| 743 | } |
| 744 | |
Richard Genoud | 3645e32 | 2014-02-10 14:25:32 -0800 | [diff] [blame] | 745 | if (-e ".git") { |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 746 | $files = `git ls-files "include/*.h"`; |
| 747 | @include_files = split('\n', $files); |
| 748 | } |
| 749 | |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 750 | foreach my $file (@include_files) { |
| 751 | seed_camelcase_file($file); |
| 752 | } |
Joe Perches | 351b2a1 | 2013-07-03 15:05:36 -0700 | [diff] [blame] | 753 | |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 754 | if ($camelcase_cache ne "") { |
Joe Perches | 351b2a1 | 2013-07-03 15:05:36 -0700 | [diff] [blame] | 755 | unlink glob ".checkpatch-camelcase.*"; |
Joe Perches | c707a81 | 2013-07-08 16:00:43 -0700 | [diff] [blame] | 756 | open(my $camelcase_file, '>', "$camelcase_cache") |
| 757 | or warn "$P: Can't write '$camelcase_cache' $!\n"; |
Joe Perches | 351b2a1 | 2013-07-03 15:05:36 -0700 | [diff] [blame] | 758 | foreach (sort { lc($a) cmp lc($b) } keys(%camelcase)) { |
| 759 | print $camelcase_file ("$_\n"); |
| 760 | } |
| 761 | close($camelcase_file); |
| 762 | } |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 763 | } |
| 764 | |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 765 | sub git_commit_info { |
| 766 | my ($commit, $id, $desc) = @_; |
| 767 | |
| 768 | return ($id, $desc) if ((which("git") eq "") || !(-e ".git")); |
| 769 | |
| 770 | my $output = `git log --no-color --format='%H %s' -1 $commit 2>&1`; |
| 771 | $output =~ s/^\s*//gm; |
| 772 | my @lines = split("\n", $output); |
| 773 | |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 774 | return ($id, $desc) if ($#lines < 0); |
| 775 | |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 776 | if ($lines[0] =~ /^error: short SHA1 $commit is ambiguous\./) { |
| 777 | # Maybe one day convert this block of bash into something that returns |
| 778 | # all matching commit ids, but it's very slow... |
| 779 | # |
| 780 | # echo "checking commits $1..." |
| 781 | # git rev-list --remotes | grep -i "^$1" | |
| 782 | # while read line ; do |
| 783 | # git log --format='%H %s' -1 $line | |
| 784 | # echo "commit $(cut -c 1-12,41-)" |
| 785 | # done |
| 786 | } elsif ($lines[0] =~ /^fatal: ambiguous argument '$commit': unknown revision or path not in the working tree\./) { |
| 787 | } else { |
| 788 | $id = substr($lines[0], 0, 12); |
| 789 | $desc = substr($lines[0], 41); |
| 790 | } |
| 791 | |
| 792 | return ($id, $desc); |
| 793 | } |
| 794 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 795 | $chk_signoff = 0 if ($file); |
| 796 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 797 | my @rawlines = (); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 798 | my @lines = (); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 799 | my @fixed = (); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 800 | my @fixed_inserted = (); |
| 801 | my @fixed_deleted = (); |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 802 | my $fixlinenr = -1; |
| 803 | |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 804 | # If input is git commits, extract all commits from the commit expressions. |
| 805 | # For example, HEAD-3 means we need check 'HEAD, HEAD~1, HEAD~2'. |
| 806 | die "$P: No git repository found\n" if ($git && !-e ".git"); |
| 807 | |
| 808 | if ($git) { |
| 809 | my @commits = (); |
Joe Perches | 0dea9f1e | 2016-05-20 17:04:19 -0700 | [diff] [blame] | 810 | foreach my $commit_expr (@ARGV) { |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 811 | my $git_range; |
Joe Perches | 28898fd | 2016-05-20 17:04:22 -0700 | [diff] [blame] | 812 | if ($commit_expr =~ m/^(.*)-(\d+)$/) { |
| 813 | $git_range = "-$2 $1"; |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 814 | } elsif ($commit_expr =~ m/\.\./) { |
| 815 | $git_range = "$commit_expr"; |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 816 | } else { |
Joe Perches | 0dea9f1e | 2016-05-20 17:04:19 -0700 | [diff] [blame] | 817 | $git_range = "-1 $commit_expr"; |
| 818 | } |
| 819 | my $lines = `git log --no-color --no-merges --pretty=format:'%H %s' $git_range`; |
| 820 | foreach my $line (split(/\n/, $lines)) { |
Joe Perches | 28898fd | 2016-05-20 17:04:22 -0700 | [diff] [blame] | 821 | $line =~ /^([0-9a-fA-F]{40,40}) (.*)$/; |
| 822 | next if (!defined($1) || !defined($2)); |
Joe Perches | 0dea9f1e | 2016-05-20 17:04:19 -0700 | [diff] [blame] | 823 | my $sha1 = $1; |
| 824 | my $subject = $2; |
| 825 | unshift(@commits, $sha1); |
| 826 | $git_commits{$sha1} = $subject; |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 827 | } |
| 828 | } |
| 829 | die "$P: no git commits after extraction!\n" if (@commits == 0); |
| 830 | @ARGV = @commits; |
| 831 | } |
| 832 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 833 | my $vname; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 834 | for my $filename (@ARGV) { |
Andy Whitcroft | 21caa13 | 2009-01-06 14:41:30 -0800 | [diff] [blame] | 835 | my $FILE; |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 836 | if ($git) { |
| 837 | open($FILE, '-|', "git format-patch -M --stdout -1 $filename") || |
| 838 | die "$P: $filename: git format-patch failed - $!\n"; |
| 839 | } elsif ($file) { |
Andy Whitcroft | 21caa13 | 2009-01-06 14:41:30 -0800 | [diff] [blame] | 840 | open($FILE, '-|', "diff -u /dev/null $filename") || |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 841 | die "$P: $filename: diff failed - $!\n"; |
Andy Whitcroft | 21caa13 | 2009-01-06 14:41:30 -0800 | [diff] [blame] | 842 | } elsif ($filename eq '-') { |
| 843 | open($FILE, '<&STDIN'); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 844 | } else { |
Andy Whitcroft | 21caa13 | 2009-01-06 14:41:30 -0800 | [diff] [blame] | 845 | open($FILE, '<', "$filename") || |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 846 | die "$P: $filename: open failed - $!\n"; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 847 | } |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 848 | if ($filename eq '-') { |
| 849 | $vname = 'Your patch'; |
Du, Changbin | 4a593c3 | 2016-05-20 17:04:16 -0700 | [diff] [blame] | 850 | } elsif ($git) { |
Joe Perches | 0dea9f1e | 2016-05-20 17:04:19 -0700 | [diff] [blame] | 851 | $vname = "Commit " . substr($filename, 0, 12) . ' ("' . $git_commits{$filename} . '")'; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 852 | } else { |
| 853 | $vname = $filename; |
| 854 | } |
Andy Whitcroft | 21caa13 | 2009-01-06 14:41:30 -0800 | [diff] [blame] | 855 | while (<$FILE>) { |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 856 | chomp; |
| 857 | push(@rawlines, $_); |
| 858 | } |
Andy Whitcroft | 21caa13 | 2009-01-06 14:41:30 -0800 | [diff] [blame] | 859 | close($FILE); |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 860 | |
| 861 | if ($#ARGV > 0 && $quiet == 0) { |
| 862 | print '-' x length($vname) . "\n"; |
| 863 | print "$vname\n"; |
| 864 | print '-' x length($vname) . "\n"; |
| 865 | } |
| 866 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 867 | if (!process($filename)) { |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 868 | $exit = 1; |
| 869 | } |
| 870 | @rawlines = (); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 871 | @lines = (); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 872 | @fixed = (); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 873 | @fixed_inserted = (); |
| 874 | @fixed_deleted = (); |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 875 | $fixlinenr = -1; |
Alex Dowad | 485ff23 | 2015-06-25 15:02:52 -0700 | [diff] [blame] | 876 | @modifierListFile = (); |
| 877 | @typeListFile = (); |
| 878 | build_types(); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 879 | } |
| 880 | |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 881 | if (!$quiet) { |
Joe Perches | 3c816e4 | 2015-06-25 15:03:29 -0700 | [diff] [blame] | 882 | hash_show_words(\%use_type, "Used"); |
| 883 | hash_show_words(\%ignore_type, "Ignored"); |
| 884 | |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 885 | if ($^V lt 5.10.0) { |
| 886 | print << "EOM" |
| 887 | |
| 888 | NOTE: perl $^V is not modern enough to detect all possible issues. |
| 889 | An upgrade to at least perl v5.10.0 is suggested. |
| 890 | EOM |
| 891 | } |
| 892 | if ($exit) { |
| 893 | print << "EOM" |
| 894 | |
| 895 | NOTE: If any of the errors are false positives, please report |
| 896 | them to the maintainer, see CHECKPATCH in MAINTAINERS. |
| 897 | EOM |
| 898 | } |
| 899 | } |
| 900 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 901 | exit($exit); |
| 902 | |
| 903 | sub top_of_kernel_tree { |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 904 | my ($root) = @_; |
| 905 | |
| 906 | my @tree_check = ( |
| 907 | "COPYING", "CREDITS", "Kbuild", "MAINTAINERS", "Makefile", |
| 908 | "README", "Documentation", "arch", "include", "drivers", |
| 909 | "fs", "init", "ipc", "kernel", "lib", "scripts", |
| 910 | ); |
| 911 | |
| 912 | foreach my $check (@tree_check) { |
| 913 | if (! -e $root . '/' . $check) { |
| 914 | return 0; |
| 915 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 916 | } |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 917 | return 1; |
Joe Perches | 8f26b83 | 2012-10-04 17:13:32 -0700 | [diff] [blame] | 918 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 919 | |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 920 | sub parse_email { |
| 921 | my ($formatted_email) = @_; |
| 922 | |
| 923 | my $name = ""; |
| 924 | my $address = ""; |
| 925 | my $comment = ""; |
| 926 | |
| 927 | if ($formatted_email =~ /^(.*)<(\S+\@\S+)>(.*)$/) { |
| 928 | $name = $1; |
| 929 | $address = $2; |
| 930 | $comment = $3 if defined $3; |
| 931 | } elsif ($formatted_email =~ /^\s*<(\S+\@\S+)>(.*)$/) { |
| 932 | $address = $1; |
| 933 | $comment = $2 if defined $2; |
| 934 | } elsif ($formatted_email =~ /(\S+\@\S+)(.*)$/) { |
| 935 | $address = $1; |
| 936 | $comment = $2 if defined $2; |
| 937 | $formatted_email =~ s/$address.*$//; |
| 938 | $name = $formatted_email; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 939 | $name = trim($name); |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 940 | $name =~ s/^\"|\"$//g; |
| 941 | # If there's a name left after stripping spaces and |
| 942 | # leading quotes, and the address doesn't have both |
| 943 | # leading and trailing angle brackets, the address |
| 944 | # is invalid. ie: |
| 945 | # "joe smith joe@smith.com" bad |
| 946 | # "joe smith <joe@smith.com" bad |
| 947 | if ($name ne "" && $address !~ /^<[^>]+>$/) { |
| 948 | $name = ""; |
| 949 | $address = ""; |
| 950 | $comment = ""; |
| 951 | } |
| 952 | } |
| 953 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 954 | $name = trim($name); |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 955 | $name =~ s/^\"|\"$//g; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 956 | $address = trim($address); |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 957 | $address =~ s/^\<|\>$//g; |
| 958 | |
| 959 | if ($name =~ /[^\w \-]/i) { ##has "must quote" chars |
| 960 | $name =~ s/(?<!\\)"/\\"/g; ##escape quotes |
| 961 | $name = "\"$name\""; |
| 962 | } |
| 963 | |
| 964 | return ($name, $address, $comment); |
| 965 | } |
| 966 | |
| 967 | sub format_email { |
| 968 | my ($name, $address) = @_; |
| 969 | |
| 970 | my $formatted_email; |
| 971 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 972 | $name = trim($name); |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 973 | $name =~ s/^\"|\"$//g; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 974 | $address = trim($address); |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 975 | |
| 976 | if ($name =~ /[^\w \-]/i) { ##has "must quote" chars |
| 977 | $name =~ s/(?<!\\)"/\\"/g; ##escape quotes |
| 978 | $name = "\"$name\""; |
| 979 | } |
| 980 | |
| 981 | if ("$name" eq "") { |
| 982 | $formatted_email = "$address"; |
| 983 | } else { |
| 984 | $formatted_email = "$name <$address>"; |
| 985 | } |
| 986 | |
| 987 | return $formatted_email; |
| 988 | } |
| 989 | |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 990 | sub which { |
Joe Perches | bd474ca | 2014-08-06 16:11:10 -0700 | [diff] [blame] | 991 | my ($bin) = @_; |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 992 | |
Joe Perches | bd474ca | 2014-08-06 16:11:10 -0700 | [diff] [blame] | 993 | foreach my $path (split(/:/, $ENV{PATH})) { |
| 994 | if (-e "$path/$bin") { |
| 995 | return "$path/$bin"; |
| 996 | } |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 997 | } |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 998 | |
Joe Perches | bd474ca | 2014-08-06 16:11:10 -0700 | [diff] [blame] | 999 | return ""; |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 1000 | } |
| 1001 | |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 1002 | sub which_conf { |
| 1003 | my ($conf) = @_; |
| 1004 | |
| 1005 | foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) { |
| 1006 | if (-e "$path/$conf") { |
| 1007 | return "$path/$conf"; |
| 1008 | } |
| 1009 | } |
| 1010 | |
| 1011 | return ""; |
| 1012 | } |
| 1013 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 1014 | sub expand_tabs { |
| 1015 | my ($str) = @_; |
| 1016 | |
| 1017 | my $res = ''; |
| 1018 | my $n = 0; |
| 1019 | for my $c (split(//, $str)) { |
| 1020 | if ($c eq "\t") { |
| 1021 | $res .= ' '; |
| 1022 | $n++; |
| 1023 | for (; ($n % 8) != 0; $n++) { |
| 1024 | $res .= ' '; |
| 1025 | } |
| 1026 | next; |
| 1027 | } |
| 1028 | $res .= $c; |
| 1029 | $n++; |
| 1030 | } |
| 1031 | |
| 1032 | return $res; |
| 1033 | } |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1034 | sub copy_spacing { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1035 | (my $res = shift) =~ tr/\t/ /c; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1036 | return $res; |
| 1037 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 1038 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1039 | sub line_stats { |
| 1040 | my ($line) = @_; |
| 1041 | |
| 1042 | # Drop the diff line leader and expand tabs |
| 1043 | $line =~ s/^.//; |
| 1044 | $line = expand_tabs($line); |
| 1045 | |
| 1046 | # Pick the indent from the front of the line. |
| 1047 | my ($white) = ($line =~ /^(\s*)/); |
| 1048 | |
| 1049 | return (length($line), length($white)); |
| 1050 | } |
| 1051 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1052 | my $sanitise_quote = ''; |
| 1053 | |
| 1054 | sub sanitise_line_reset { |
| 1055 | my ($in_comment) = @_; |
| 1056 | |
| 1057 | if ($in_comment) { |
| 1058 | $sanitise_quote = '*/'; |
| 1059 | } else { |
| 1060 | $sanitise_quote = ''; |
| 1061 | } |
| 1062 | } |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1063 | sub sanitise_line { |
| 1064 | my ($line) = @_; |
| 1065 | |
| 1066 | my $res = ''; |
| 1067 | my $l = ''; |
| 1068 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1069 | my $qlen = 0; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1070 | my $off = 0; |
| 1071 | my $c; |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1072 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1073 | # Always copy over the diff marker. |
| 1074 | $res = substr($line, 0, 1); |
| 1075 | |
| 1076 | for ($off = 1; $off < length($line); $off++) { |
| 1077 | $c = substr($line, $off, 1); |
| 1078 | |
| 1079 | # Comments we are wacking completly including the begin |
| 1080 | # and end, all to $;. |
| 1081 | if ($sanitise_quote eq '' && substr($line, $off, 2) eq '/*') { |
| 1082 | $sanitise_quote = '*/'; |
| 1083 | |
| 1084 | substr($res, $off, 2, "$;$;"); |
| 1085 | $off++; |
| 1086 | next; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1087 | } |
Andy Whitcroft | 81bc0e0 | 2008-10-15 22:02:26 -0700 | [diff] [blame] | 1088 | if ($sanitise_quote eq '*/' && substr($line, $off, 2) eq '*/') { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1089 | $sanitise_quote = ''; |
| 1090 | substr($res, $off, 2, "$;$;"); |
| 1091 | $off++; |
| 1092 | next; |
| 1093 | } |
Daniel Walker | 113f04a | 2009-09-21 17:04:35 -0700 | [diff] [blame] | 1094 | if ($sanitise_quote eq '' && substr($line, $off, 2) eq '//') { |
| 1095 | $sanitise_quote = '//'; |
| 1096 | |
| 1097 | substr($res, $off, 2, $sanitise_quote); |
| 1098 | $off++; |
| 1099 | next; |
| 1100 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1101 | |
| 1102 | # A \ in a string means ignore the next character. |
| 1103 | if (($sanitise_quote eq "'" || $sanitise_quote eq '"') && |
| 1104 | $c eq "\\") { |
| 1105 | substr($res, $off, 2, 'XX'); |
| 1106 | $off++; |
| 1107 | next; |
| 1108 | } |
| 1109 | # Regular quotes. |
| 1110 | if ($c eq "'" || $c eq '"') { |
| 1111 | if ($sanitise_quote eq '') { |
| 1112 | $sanitise_quote = $c; |
| 1113 | |
| 1114 | substr($res, $off, 1, $c); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1115 | next; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1116 | } elsif ($sanitise_quote eq $c) { |
| 1117 | $sanitise_quote = ''; |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1118 | } |
| 1119 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1120 | |
Andy Whitcroft | fae17da | 2009-01-06 14:41:20 -0800 | [diff] [blame] | 1121 | #print "c<$c> SQ<$sanitise_quote>\n"; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1122 | if ($off != 0 && $sanitise_quote eq '*/' && $c ne "\t") { |
| 1123 | substr($res, $off, 1, $;); |
Daniel Walker | 113f04a | 2009-09-21 17:04:35 -0700 | [diff] [blame] | 1124 | } elsif ($off != 0 && $sanitise_quote eq '//' && $c ne "\t") { |
| 1125 | substr($res, $off, 1, $;); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1126 | } elsif ($off != 0 && $sanitise_quote && $c ne "\t") { |
| 1127 | substr($res, $off, 1, 'X'); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1128 | } else { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1129 | substr($res, $off, 1, $c); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1130 | } |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1131 | } |
| 1132 | |
Daniel Walker | 113f04a | 2009-09-21 17:04:35 -0700 | [diff] [blame] | 1133 | if ($sanitise_quote eq '//') { |
| 1134 | $sanitise_quote = ''; |
| 1135 | } |
| 1136 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1137 | # The pathname on a #include may be surrounded by '<' and '>'. |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1138 | if ($res =~ /^.\s*\#\s*include\s+\<(.*)\>/) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1139 | my $clean = 'X' x length($1); |
| 1140 | $res =~ s@\<.*\>@<$clean>@; |
| 1141 | |
| 1142 | # The whole of a #error is a string. |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1143 | } elsif ($res =~ /^.\s*\#\s*(?:error|warning)\s+(.*)\b/) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1144 | my $clean = 'X' x length($1); |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1145 | $res =~ s@(\#\s*(?:error|warning)\s+).*@$1$clean@; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1146 | } |
| 1147 | |
Joe Perches | dadf680 | 2016-08-02 14:04:33 -0700 | [diff] [blame] | 1148 | if ($allow_c99_comments && $res =~ m@(//.*$)@) { |
| 1149 | my $match = $1; |
| 1150 | $res =~ s/\Q$match\E/"$;" x length($match)/e; |
| 1151 | } |
| 1152 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1153 | return $res; |
| 1154 | } |
| 1155 | |
Joe Perches | a6962d7 | 2013-04-29 16:18:13 -0700 | [diff] [blame] | 1156 | sub get_quoted_string { |
| 1157 | my ($line, $rawline) = @_; |
| 1158 | |
Joe Perches | 33acb54 | 2015-06-25 15:02:54 -0700 | [diff] [blame] | 1159 | return "" if ($line !~ m/($String)/g); |
Joe Perches | a6962d7 | 2013-04-29 16:18:13 -0700 | [diff] [blame] | 1160 | return substr($rawline, $-[0], $+[0] - $-[0]); |
| 1161 | } |
| 1162 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1163 | sub ctx_statement_block { |
| 1164 | my ($linenr, $remain, $off) = @_; |
| 1165 | my $line = $linenr - 1; |
| 1166 | my $blk = ''; |
| 1167 | my $soff = $off; |
| 1168 | my $coff = $off - 1; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1169 | my $coff_set = 0; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1170 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1171 | my $loff = 0; |
| 1172 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1173 | my $type = ''; |
| 1174 | my $level = 0; |
Andy Whitcroft | a275064 | 2009-01-15 13:51:04 -0800 | [diff] [blame] | 1175 | my @stack = (); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1176 | my $p; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1177 | my $c; |
| 1178 | my $len = 0; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1179 | |
| 1180 | my $remainder; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1181 | while (1) { |
Andy Whitcroft | a275064 | 2009-01-15 13:51:04 -0800 | [diff] [blame] | 1182 | @stack = (['', 0]) if ($#stack == -1); |
| 1183 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1184 | #warn "CSB: blk<$blk> remain<$remain>\n"; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1185 | # If we are about to drop off the end, pull in more |
| 1186 | # context. |
| 1187 | if ($off >= $len) { |
| 1188 | for (; $remain > 0; $line++) { |
Andy Whitcroft | dea3349 | 2008-10-15 22:02:25 -0700 | [diff] [blame] | 1189 | last if (!defined $lines[$line]); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1190 | next if ($lines[$line] =~ /^-/); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1191 | $remain--; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1192 | $loff = $len; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1193 | $blk .= $lines[$line] . "\n"; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1194 | $len = length($blk); |
| 1195 | $line++; |
| 1196 | last; |
| 1197 | } |
| 1198 | # Bail if there is no further context. |
| 1199 | #warn "CSB: blk<$blk> off<$off> len<$len>\n"; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1200 | if ($off >= $len) { |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1201 | last; |
| 1202 | } |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 1203 | if ($level == 0 && substr($blk, $off) =~ /^.\s*#\s*define/) { |
| 1204 | $level++; |
| 1205 | $type = '#'; |
| 1206 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1207 | } |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1208 | $p = $c; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1209 | $c = substr($blk, $off, 1); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1210 | $remainder = substr($blk, $off); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1211 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1212 | #warn "CSB: c<$c> type<$type> level<$level> remainder<$remainder> coff_set<$coff_set>\n"; |
Andy Whitcroft | 4635f4f | 2009-01-06 14:41:27 -0800 | [diff] [blame] | 1213 | |
| 1214 | # Handle nested #if/#else. |
| 1215 | if ($remainder =~ /^#\s*(?:ifndef|ifdef|if)\s/) { |
| 1216 | push(@stack, [ $type, $level ]); |
| 1217 | } elsif ($remainder =~ /^#\s*(?:else|elif)\b/) { |
| 1218 | ($type, $level) = @{$stack[$#stack - 1]}; |
| 1219 | } elsif ($remainder =~ /^#\s*endif\b/) { |
| 1220 | ($type, $level) = @{pop(@stack)}; |
| 1221 | } |
| 1222 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1223 | # Statement ends at the ';' or a close '}' at the |
| 1224 | # outermost level. |
| 1225 | if ($level == 0 && $c eq ';') { |
| 1226 | last; |
| 1227 | } |
| 1228 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1229 | # An else is really a conditional as long as its not else if |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1230 | if ($level == 0 && $coff_set == 0 && |
| 1231 | (!defined($p) || $p =~ /(?:\s|\}|\+)/) && |
| 1232 | $remainder =~ /^(else)(?:\s|{)/ && |
| 1233 | $remainder !~ /^else\s+if\b/) { |
| 1234 | $coff = $off + length($1) - 1; |
| 1235 | $coff_set = 1; |
| 1236 | #warn "CSB: mark coff<$coff> soff<$soff> 1<$1>\n"; |
| 1237 | #warn "[" . substr($blk, $soff, $coff - $soff + 1) . "]\n"; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1238 | } |
| 1239 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1240 | if (($type eq '' || $type eq '(') && $c eq '(') { |
| 1241 | $level++; |
| 1242 | $type = '('; |
| 1243 | } |
| 1244 | if ($type eq '(' && $c eq ')') { |
| 1245 | $level--; |
| 1246 | $type = ($level != 0)? '(' : ''; |
| 1247 | |
| 1248 | if ($level == 0 && $coff < $soff) { |
| 1249 | $coff = $off; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1250 | $coff_set = 1; |
| 1251 | #warn "CSB: mark coff<$coff>\n"; |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1252 | } |
| 1253 | } |
| 1254 | if (($type eq '' || $type eq '{') && $c eq '{') { |
| 1255 | $level++; |
| 1256 | $type = '{'; |
| 1257 | } |
| 1258 | if ($type eq '{' && $c eq '}') { |
| 1259 | $level--; |
| 1260 | $type = ($level != 0)? '{' : ''; |
| 1261 | |
| 1262 | if ($level == 0) { |
Patrick Pannuto | b998e00 | 2010-08-09 17:21:03 -0700 | [diff] [blame] | 1263 | if (substr($blk, $off + 1, 1) eq ';') { |
| 1264 | $off++; |
| 1265 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1266 | last; |
| 1267 | } |
| 1268 | } |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 1269 | # Preprocessor commands end at the newline unless escaped. |
| 1270 | if ($type eq '#' && $c eq "\n" && $p ne "\\") { |
| 1271 | $level--; |
| 1272 | $type = ''; |
| 1273 | $off++; |
| 1274 | last; |
| 1275 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1276 | $off++; |
| 1277 | } |
Andy Whitcroft | a3bb97a | 2008-07-23 21:29:00 -0700 | [diff] [blame] | 1278 | # We are truly at the end, so shuffle to the next line. |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1279 | if ($off == $len) { |
Andy Whitcroft | a3bb97a | 2008-07-23 21:29:00 -0700 | [diff] [blame] | 1280 | $loff = $len + 1; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1281 | $line++; |
| 1282 | $remain--; |
| 1283 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1284 | |
| 1285 | my $statement = substr($blk, $soff, $off - $soff + 1); |
| 1286 | my $condition = substr($blk, $soff, $coff - $soff + 1); |
| 1287 | |
| 1288 | #warn "STATEMENT<$statement>\n"; |
| 1289 | #warn "CONDITION<$condition>\n"; |
| 1290 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1291 | #print "coff<$coff> soff<$off> loff<$loff>\n"; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1292 | |
| 1293 | return ($statement, $condition, |
| 1294 | $line, $remain + 1, $off - $loff + 1, $level); |
| 1295 | } |
| 1296 | |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1297 | sub statement_lines { |
| 1298 | my ($stmt) = @_; |
| 1299 | |
| 1300 | # Strip the diff line prefixes and rip blank lines at start and end. |
| 1301 | $stmt =~ s/(^|\n)./$1/g; |
| 1302 | $stmt =~ s/^\s*//; |
| 1303 | $stmt =~ s/\s*$//; |
| 1304 | |
| 1305 | my @stmt_lines = ($stmt =~ /\n/g); |
| 1306 | |
| 1307 | return $#stmt_lines + 2; |
| 1308 | } |
| 1309 | |
| 1310 | sub statement_rawlines { |
| 1311 | my ($stmt) = @_; |
| 1312 | |
| 1313 | my @stmt_lines = ($stmt =~ /\n/g); |
| 1314 | |
| 1315 | return $#stmt_lines + 2; |
| 1316 | } |
| 1317 | |
| 1318 | sub statement_block_size { |
| 1319 | my ($stmt) = @_; |
| 1320 | |
| 1321 | $stmt =~ s/(^|\n)./$1/g; |
| 1322 | $stmt =~ s/^\s*{//; |
| 1323 | $stmt =~ s/}\s*$//; |
| 1324 | $stmt =~ s/^\s*//; |
| 1325 | $stmt =~ s/\s*$//; |
| 1326 | |
| 1327 | my @stmt_lines = ($stmt =~ /\n/g); |
| 1328 | my @stmt_statements = ($stmt =~ /;/g); |
| 1329 | |
| 1330 | my $stmt_lines = $#stmt_lines + 2; |
| 1331 | my $stmt_statements = $#stmt_statements + 1; |
| 1332 | |
| 1333 | if ($stmt_lines > $stmt_statements) { |
| 1334 | return $stmt_lines; |
| 1335 | } else { |
| 1336 | return $stmt_statements; |
| 1337 | } |
| 1338 | } |
| 1339 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1340 | sub ctx_statement_full { |
| 1341 | my ($linenr, $remain, $off) = @_; |
| 1342 | my ($statement, $condition, $level); |
| 1343 | |
| 1344 | my (@chunks); |
| 1345 | |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1346 | # Grab the first conditional/block pair. |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1347 | ($statement, $condition, $linenr, $remain, $off, $level) = |
| 1348 | ctx_statement_block($linenr, $remain, $off); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1349 | #print "F: c<$condition> s<$statement> remain<$remain>\n"; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1350 | push(@chunks, [ $condition, $statement ]); |
| 1351 | if (!($remain > 0 && $condition =~ /^\s*(?:\n[+-])?\s*(?:if|else|do)\b/s)) { |
| 1352 | return ($level, $linenr, @chunks); |
| 1353 | } |
| 1354 | |
| 1355 | # Pull in the following conditional/block pairs and see if they |
| 1356 | # could continue the statement. |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1357 | for (;;) { |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1358 | ($statement, $condition, $linenr, $remain, $off, $level) = |
| 1359 | ctx_statement_block($linenr, $remain, $off); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1360 | #print "C: c<$condition> s<$statement> remain<$remain>\n"; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1361 | last if (!($remain > 0 && $condition =~ /^(?:\s*\n[+-])*\s*(?:else|do)\b/s)); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1362 | #print "C: push\n"; |
| 1363 | push(@chunks, [ $condition, $statement ]); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1364 | } |
| 1365 | |
| 1366 | return ($level, $linenr, @chunks); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1367 | } |
| 1368 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1369 | sub ctx_block_get { |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1370 | my ($linenr, $remain, $outer, $open, $close, $off) = @_; |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1371 | my $line; |
| 1372 | my $start = $linenr - 1; |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1373 | my $blk = ''; |
| 1374 | my @o; |
| 1375 | my @c; |
| 1376 | my @res = (); |
| 1377 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1378 | my $level = 0; |
Andy Whitcroft | 4635f4f | 2009-01-06 14:41:27 -0800 | [diff] [blame] | 1379 | my @stack = ($level); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1380 | for ($line = $start; $remain > 0; $line++) { |
| 1381 | next if ($rawlines[$line] =~ /^-/); |
| 1382 | $remain--; |
| 1383 | |
| 1384 | $blk .= $rawlines[$line]; |
Andy Whitcroft | 4635f4f | 2009-01-06 14:41:27 -0800 | [diff] [blame] | 1385 | |
| 1386 | # Handle nested #if/#else. |
Andy Whitcroft | 01464f3 | 2010-10-26 14:23:19 -0700 | [diff] [blame] | 1387 | if ($lines[$line] =~ /^.\s*#\s*(?:ifndef|ifdef|if)\s/) { |
Andy Whitcroft | 4635f4f | 2009-01-06 14:41:27 -0800 | [diff] [blame] | 1388 | push(@stack, $level); |
Andy Whitcroft | 01464f3 | 2010-10-26 14:23:19 -0700 | [diff] [blame] | 1389 | } elsif ($lines[$line] =~ /^.\s*#\s*(?:else|elif)\b/) { |
Andy Whitcroft | 4635f4f | 2009-01-06 14:41:27 -0800 | [diff] [blame] | 1390 | $level = $stack[$#stack - 1]; |
Andy Whitcroft | 01464f3 | 2010-10-26 14:23:19 -0700 | [diff] [blame] | 1391 | } elsif ($lines[$line] =~ /^.\s*#\s*endif\b/) { |
Andy Whitcroft | 4635f4f | 2009-01-06 14:41:27 -0800 | [diff] [blame] | 1392 | $level = pop(@stack); |
| 1393 | } |
| 1394 | |
Andy Whitcroft | 01464f3 | 2010-10-26 14:23:19 -0700 | [diff] [blame] | 1395 | foreach my $c (split(//, $lines[$line])) { |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1396 | ##print "C<$c>L<$level><$open$close>O<$off>\n"; |
| 1397 | if ($off > 0) { |
| 1398 | $off--; |
| 1399 | next; |
| 1400 | } |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1401 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1402 | if ($c eq $close && $level > 0) { |
| 1403 | $level--; |
| 1404 | last if ($level == 0); |
| 1405 | } elsif ($c eq $open) { |
| 1406 | $level++; |
| 1407 | } |
| 1408 | } |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1409 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1410 | if (!$outer || $level <= 1) { |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1411 | push(@res, $rawlines[$line]); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1412 | } |
| 1413 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1414 | last if ($level == 0); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1415 | } |
| 1416 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1417 | return ($level, @res); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1418 | } |
| 1419 | sub ctx_block_outer { |
| 1420 | my ($linenr, $remain) = @_; |
| 1421 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1422 | my ($level, @r) = ctx_block_get($linenr, $remain, 1, '{', '}', 0); |
| 1423 | return @r; |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1424 | } |
| 1425 | sub ctx_block { |
| 1426 | my ($linenr, $remain) = @_; |
| 1427 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1428 | my ($level, @r) = ctx_block_get($linenr, $remain, 0, '{', '}', 0); |
| 1429 | return @r; |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 1430 | } |
| 1431 | sub ctx_statement { |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1432 | my ($linenr, $remain, $off) = @_; |
| 1433 | |
| 1434 | my ($level, @r) = ctx_block_get($linenr, $remain, 0, '(', ')', $off); |
| 1435 | return @r; |
| 1436 | } |
| 1437 | sub ctx_block_level { |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 1438 | my ($linenr, $remain) = @_; |
| 1439 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1440 | return ctx_block_get($linenr, $remain, 0, '{', '}', 0); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1441 | } |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 1442 | sub ctx_statement_level { |
| 1443 | my ($linenr, $remain, $off) = @_; |
| 1444 | |
| 1445 | return ctx_block_get($linenr, $remain, 0, '(', ')', $off); |
| 1446 | } |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1447 | |
| 1448 | sub ctx_locate_comment { |
| 1449 | my ($first_line, $end_line) = @_; |
| 1450 | |
| 1451 | # Catch a comment on the end of the line itself. |
Andy Whitcroft | beae633 | 2008-07-23 21:28:59 -0700 | [diff] [blame] | 1452 | my ($current_comment) = ($rawlines[$end_line - 1] =~ m@.*(/\*.*\*/)\s*(?:\\\s*)?$@); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1453 | return $current_comment if (defined $current_comment); |
| 1454 | |
| 1455 | # Look through the context and try and figure out if there is a |
| 1456 | # comment. |
| 1457 | my $in_comment = 0; |
| 1458 | $current_comment = ''; |
| 1459 | for (my $linenr = $first_line; $linenr < $end_line; $linenr++) { |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1460 | my $line = $rawlines[$linenr - 1]; |
| 1461 | #warn " $line\n"; |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1462 | if ($linenr == $first_line and $line =~ m@^.\s*\*@) { |
| 1463 | $in_comment = 1; |
| 1464 | } |
| 1465 | if ($line =~ m@/\*@) { |
| 1466 | $in_comment = 1; |
| 1467 | } |
| 1468 | if (!$in_comment && $current_comment ne '') { |
| 1469 | $current_comment = ''; |
| 1470 | } |
| 1471 | $current_comment .= $line . "\n" if ($in_comment); |
| 1472 | if ($line =~ m@\*/@) { |
| 1473 | $in_comment = 0; |
| 1474 | } |
| 1475 | } |
| 1476 | |
| 1477 | chomp($current_comment); |
| 1478 | return($current_comment); |
| 1479 | } |
| 1480 | sub ctx_has_comment { |
| 1481 | my ($first_line, $end_line) = @_; |
| 1482 | my $cmt = ctx_locate_comment($first_line, $end_line); |
| 1483 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 1484 | ##print "LINE: $rawlines[$end_line - 1 ]\n"; |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 1485 | ##print "CMMT: $cmt\n"; |
| 1486 | |
| 1487 | return ($cmt ne ''); |
| 1488 | } |
| 1489 | |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 1490 | sub raw_line { |
| 1491 | my ($linenr, $cnt) = @_; |
| 1492 | |
| 1493 | my $offset = $linenr - 1; |
| 1494 | $cnt++; |
| 1495 | |
| 1496 | my $line; |
| 1497 | while ($cnt) { |
| 1498 | $line = $rawlines[$offset++]; |
| 1499 | next if (defined($line) && $line =~ /^-/); |
| 1500 | $cnt--; |
| 1501 | } |
| 1502 | |
| 1503 | return $line; |
| 1504 | } |
| 1505 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 1506 | sub cat_vet { |
| 1507 | my ($vet) = @_; |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 1508 | my ($res, $coded); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 1509 | |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 1510 | $res = ''; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1511 | while ($vet =~ /([^[:cntrl:]]*)([[:cntrl:]]|$)/g) { |
| 1512 | $res .= $1; |
| 1513 | if ($2 ne '') { |
| 1514 | $coded = sprintf("^%c", unpack('C', $2) + 64); |
| 1515 | $res .= $coded; |
| 1516 | } |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 1517 | } |
| 1518 | $res =~ s/$/\$/; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 1519 | |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 1520 | return $res; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 1521 | } |
| 1522 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1523 | my $av_preprocessor = 0; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1524 | my $av_pending; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1525 | my @av_paren_type; |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1526 | my $av_pend_colon; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1527 | |
| 1528 | sub annotate_reset { |
| 1529 | $av_preprocessor = 0; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1530 | $av_pending = '_'; |
| 1531 | @av_paren_type = ('E'); |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1532 | $av_pend_colon = 'O'; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1533 | } |
| 1534 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1535 | sub annotate_values { |
| 1536 | my ($stream, $type) = @_; |
| 1537 | |
| 1538 | my $res; |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1539 | my $var = '_' x length($stream); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1540 | my $cur = $stream; |
| 1541 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1542 | print "$stream\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1543 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1544 | while (length($cur)) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1545 | @av_paren_type = ('E') if ($#av_paren_type < 0); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1546 | print " <" . join('', @av_paren_type) . |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1547 | "> <$type> <$av_pending>" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1548 | if ($cur =~ /^(\s+)/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1549 | print "WS($1)\n" if ($dbg_values > 1); |
| 1550 | if ($1 =~ /\n/ && $av_preprocessor) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1551 | $type = pop(@av_paren_type); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1552 | $av_preprocessor = 0; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1553 | } |
| 1554 | |
Florian Mickler | c023e473 | 2011-01-12 16:59:58 -0800 | [diff] [blame] | 1555 | } elsif ($cur =~ /^(\(\s*$Type\s*)\)/ && $av_pending eq '_') { |
Andy Whitcroft | 9446ef5 | 2010-10-26 14:23:13 -0700 | [diff] [blame] | 1556 | print "CAST($1)\n" if ($dbg_values > 1); |
| 1557 | push(@av_paren_type, $type); |
Andy Whitcroft | addcdce | 2012-01-10 15:10:11 -0800 | [diff] [blame] | 1558 | $type = 'c'; |
Andy Whitcroft | 9446ef5 | 2010-10-26 14:23:13 -0700 | [diff] [blame] | 1559 | |
Andy Whitcroft | e91b6e2 | 2010-10-26 14:23:11 -0700 | [diff] [blame] | 1560 | } elsif ($cur =~ /^($Type)\s*(?:$Ident|,|\)|\(|\s*$)/) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1561 | print "DECLARE($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1562 | $type = 'T'; |
| 1563 | |
Andy Whitcroft | 389a2fe | 2008-07-23 21:29:05 -0700 | [diff] [blame] | 1564 | } elsif ($cur =~ /^($Modifier)\s*/) { |
| 1565 | print "MODIFIER($1)\n" if ($dbg_values > 1); |
| 1566 | $type = 'T'; |
| 1567 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1568 | } elsif ($cur =~ /^(\#\s*define\s*$Ident)(\(?)/o) { |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1569 | print "DEFINE($1,$2)\n" if ($dbg_values > 1); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1570 | $av_preprocessor = 1; |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1571 | push(@av_paren_type, $type); |
| 1572 | if ($2 ne '') { |
| 1573 | $av_pending = 'N'; |
| 1574 | } |
| 1575 | $type = 'E'; |
| 1576 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1577 | } elsif ($cur =~ /^(\#\s*(?:undef\s*$Ident|include\b))/o) { |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1578 | print "UNDEF($1)\n" if ($dbg_values > 1); |
| 1579 | $av_preprocessor = 1; |
| 1580 | push(@av_paren_type, $type); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1581 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1582 | } elsif ($cur =~ /^(\#\s*(?:ifdef|ifndef|if))/o) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1583 | print "PRE_START($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1584 | $av_preprocessor = 1; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1585 | |
| 1586 | push(@av_paren_type, $type); |
| 1587 | push(@av_paren_type, $type); |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1588 | $type = 'E'; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1589 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1590 | } elsif ($cur =~ /^(\#\s*(?:else|elif))/o) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1591 | print "PRE_RESTART($1)\n" if ($dbg_values > 1); |
| 1592 | $av_preprocessor = 1; |
| 1593 | |
| 1594 | push(@av_paren_type, $av_paren_type[$#av_paren_type]); |
| 1595 | |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1596 | $type = 'E'; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1597 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1598 | } elsif ($cur =~ /^(\#\s*(?:endif))/o) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1599 | print "PRE_END($1)\n" if ($dbg_values > 1); |
| 1600 | |
| 1601 | $av_preprocessor = 1; |
| 1602 | |
| 1603 | # Assume all arms of the conditional end as this |
| 1604 | # one does, and continue as if the #endif was not here. |
| 1605 | pop(@av_paren_type); |
| 1606 | push(@av_paren_type, $type); |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1607 | $type = 'E'; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1608 | |
| 1609 | } elsif ($cur =~ /^(\\\n)/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1610 | print "PRECONT($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1611 | |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 1612 | } elsif ($cur =~ /^(__attribute__)\s*\(?/o) { |
| 1613 | print "ATTR($1)\n" if ($dbg_values > 1); |
| 1614 | $av_pending = $type; |
| 1615 | $type = 'N'; |
| 1616 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1617 | } elsif ($cur =~ /^(sizeof)\s*(\()?/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1618 | print "SIZEOF($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1619 | if (defined $2) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1620 | $av_pending = 'V'; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1621 | } |
| 1622 | $type = 'N'; |
| 1623 | |
Andy Whitcroft | 14b111c | 2008-10-15 22:02:16 -0700 | [diff] [blame] | 1624 | } elsif ($cur =~ /^(if|while|for)\b/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1625 | print "COND($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 14b111c | 2008-10-15 22:02:16 -0700 | [diff] [blame] | 1626 | $av_pending = 'E'; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1627 | $type = 'N'; |
| 1628 | |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1629 | } elsif ($cur =~/^(case)/o) { |
| 1630 | print "CASE($1)\n" if ($dbg_values > 1); |
| 1631 | $av_pend_colon = 'C'; |
| 1632 | $type = 'N'; |
| 1633 | |
Andy Whitcroft | 14b111c | 2008-10-15 22:02:16 -0700 | [diff] [blame] | 1634 | } elsif ($cur =~/^(return|else|goto|typeof|__typeof__)\b/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1635 | print "KEYWORD($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1636 | $type = 'N'; |
| 1637 | |
| 1638 | } elsif ($cur =~ /^(\()/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1639 | print "PAREN('$1')\n" if ($dbg_values > 1); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1640 | push(@av_paren_type, $av_pending); |
| 1641 | $av_pending = '_'; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1642 | $type = 'N'; |
| 1643 | |
| 1644 | } elsif ($cur =~ /^(\))/o) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1645 | my $new_type = pop(@av_paren_type); |
| 1646 | if ($new_type ne '_') { |
| 1647 | $type = $new_type; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1648 | print "PAREN('$1') -> $type\n" |
| 1649 | if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1650 | } else { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1651 | print "PAREN('$1')\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1652 | } |
| 1653 | |
Andy Whitcroft | c8cb2ca | 2008-07-23 21:28:57 -0700 | [diff] [blame] | 1654 | } elsif ($cur =~ /^($Ident)\s*\(/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1655 | print "FUNC($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | c8cb2ca | 2008-07-23 21:28:57 -0700 | [diff] [blame] | 1656 | $type = 'V'; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1657 | $av_pending = 'V'; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1658 | |
Andy Whitcroft | 8e761b0 | 2009-01-06 14:41:19 -0800 | [diff] [blame] | 1659 | } elsif ($cur =~ /^($Ident\s*):(?:\s*\d+\s*(,|=|;))?/) { |
| 1660 | if (defined $2 && $type eq 'C' || $type eq 'T') { |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1661 | $av_pend_colon = 'B'; |
Andy Whitcroft | 8e761b0 | 2009-01-06 14:41:19 -0800 | [diff] [blame] | 1662 | } elsif ($type eq 'E') { |
| 1663 | $av_pend_colon = 'L'; |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1664 | } |
| 1665 | print "IDENT_COLON($1,$type>$av_pend_colon)\n" if ($dbg_values > 1); |
| 1666 | $type = 'V'; |
| 1667 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1668 | } elsif ($cur =~ /^($Ident|$Constant)/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1669 | print "IDENT($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1670 | $type = 'V'; |
| 1671 | |
| 1672 | } elsif ($cur =~ /^($Assignment)/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1673 | print "ASSIGN($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1674 | $type = 'N'; |
| 1675 | |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 1676 | } elsif ($cur =~/^(;|{|})/) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1677 | print "END($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1678 | $type = 'E'; |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1679 | $av_pend_colon = 'O'; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1680 | |
Andy Whitcroft | 8e761b0 | 2009-01-06 14:41:19 -0800 | [diff] [blame] | 1681 | } elsif ($cur =~/^(,)/) { |
| 1682 | print "COMMA($1)\n" if ($dbg_values > 1); |
| 1683 | $type = 'C'; |
| 1684 | |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1685 | } elsif ($cur =~ /^(\?)/o) { |
| 1686 | print "QUESTION($1)\n" if ($dbg_values > 1); |
| 1687 | $type = 'N'; |
| 1688 | |
| 1689 | } elsif ($cur =~ /^(:)/o) { |
| 1690 | print "COLON($1,$av_pend_colon)\n" if ($dbg_values > 1); |
| 1691 | |
| 1692 | substr($var, length($res), 1, $av_pend_colon); |
| 1693 | if ($av_pend_colon eq 'C' || $av_pend_colon eq 'L') { |
| 1694 | $type = 'E'; |
| 1695 | } else { |
| 1696 | $type = 'N'; |
| 1697 | } |
| 1698 | $av_pend_colon = 'O'; |
| 1699 | |
Andy Whitcroft | 8e761b0 | 2009-01-06 14:41:19 -0800 | [diff] [blame] | 1700 | } elsif ($cur =~ /^(\[)/o) { |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1701 | print "CLOSE($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1702 | $type = 'N'; |
| 1703 | |
Andy Whitcroft | 0d41386 | 2008-10-15 22:02:16 -0700 | [diff] [blame] | 1704 | } elsif ($cur =~ /^(-(?![->])|\+(?!\+)|\*|\&\&|\&)/o) { |
Andy Whitcroft | 74048ed | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1705 | my $variant; |
| 1706 | |
| 1707 | print "OPV($1)\n" if ($dbg_values > 1); |
| 1708 | if ($type eq 'V') { |
| 1709 | $variant = 'B'; |
| 1710 | } else { |
| 1711 | $variant = 'U'; |
| 1712 | } |
| 1713 | |
| 1714 | substr($var, length($res), 1, $variant); |
| 1715 | $type = 'N'; |
| 1716 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1717 | } elsif ($cur =~ /^($Operators)/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1718 | print "OP($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1719 | if ($1 ne '++' && $1 ne '--') { |
| 1720 | $type = 'N'; |
| 1721 | } |
| 1722 | |
| 1723 | } elsif ($cur =~ /(^.)/o) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 1724 | print "C($1)\n" if ($dbg_values > 1); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1725 | } |
| 1726 | if (defined $1) { |
| 1727 | $cur = substr($cur, length($1)); |
| 1728 | $res .= $type x length($1); |
| 1729 | } |
| 1730 | } |
| 1731 | |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 1732 | return ($res, $var); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1733 | } |
| 1734 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1735 | sub possible { |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1736 | my ($possible, $line) = @_; |
Andy Whitcroft | 9a974fd | 2009-10-26 16:50:12 -0700 | [diff] [blame] | 1737 | my $notPermitted = qr{(?: |
Andy Whitcroft | 0776e59 | 2008-10-15 22:02:29 -0700 | [diff] [blame] | 1738 | ^(?: |
| 1739 | $Modifier| |
| 1740 | $Storage| |
| 1741 | $Type| |
Andy Whitcroft | 9a974fd | 2009-10-26 16:50:12 -0700 | [diff] [blame] | 1742 | DEFINE_\S+ |
| 1743 | )$| |
| 1744 | ^(?: |
Andy Whitcroft | 0776e59 | 2008-10-15 22:02:29 -0700 | [diff] [blame] | 1745 | goto| |
| 1746 | return| |
| 1747 | case| |
| 1748 | else| |
| 1749 | asm|__asm__| |
Andy Whitcroft | 89a8835 | 2012-01-10 15:10:00 -0800 | [diff] [blame] | 1750 | do| |
| 1751 | \#| |
| 1752 | \#\#| |
Andy Whitcroft | 9a974fd | 2009-10-26 16:50:12 -0700 | [diff] [blame] | 1753 | )(?:\s|$)| |
Andy Whitcroft | 0776e59 | 2008-10-15 22:02:29 -0700 | [diff] [blame] | 1754 | ^(?:typedef|struct|enum)\b |
Andy Whitcroft | 9a974fd | 2009-10-26 16:50:12 -0700 | [diff] [blame] | 1755 | )}x; |
| 1756 | warn "CHECK<$possible> ($line)\n" if ($dbg_possible > 2); |
| 1757 | if ($possible !~ $notPermitted) { |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1758 | # Check for modifiers. |
| 1759 | $possible =~ s/\s*$Storage\s*//g; |
| 1760 | $possible =~ s/\s*$Sparse\s*//g; |
| 1761 | if ($possible =~ /^\s*$/) { |
| 1762 | |
| 1763 | } elsif ($possible =~ /\s/) { |
| 1764 | $possible =~ s/\s*$Type\s*//g; |
Andy Whitcroft | d250658 | 2008-07-23 21:29:09 -0700 | [diff] [blame] | 1765 | for my $modifier (split(' ', $possible)) { |
Andy Whitcroft | 9a974fd | 2009-10-26 16:50:12 -0700 | [diff] [blame] | 1766 | if ($modifier !~ $notPermitted) { |
| 1767 | warn "MODIFIER: $modifier ($possible) ($line)\n" if ($dbg_possible); |
Alex Dowad | 485ff23 | 2015-06-25 15:02:52 -0700 | [diff] [blame] | 1768 | push(@modifierListFile, $modifier); |
Andy Whitcroft | 9a974fd | 2009-10-26 16:50:12 -0700 | [diff] [blame] | 1769 | } |
Andy Whitcroft | d250658 | 2008-07-23 21:29:09 -0700 | [diff] [blame] | 1770 | } |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1771 | |
| 1772 | } else { |
| 1773 | warn "POSSIBLE: $possible ($line)\n" if ($dbg_possible); |
Alex Dowad | 485ff23 | 2015-06-25 15:02:52 -0700 | [diff] [blame] | 1774 | push(@typeListFile, $possible); |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 1775 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1776 | build_types(); |
Andy Whitcroft | 0776e59 | 2008-10-15 22:02:29 -0700 | [diff] [blame] | 1777 | } else { |
| 1778 | warn "NOTPOSS: $possible ($line)\n" if ($dbg_possible > 1); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1779 | } |
| 1780 | } |
| 1781 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1782 | my $prefix = ''; |
| 1783 | |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 1784 | sub show_type { |
Joe Perches | cbec18a | 2014-04-03 14:49:19 -0700 | [diff] [blame] | 1785 | my ($type) = @_; |
Joe Perches | 91bfe48 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 1786 | |
Joe Perches | cbec18a | 2014-04-03 14:49:19 -0700 | [diff] [blame] | 1787 | return defined $use_type{$type} if (scalar keys %use_type > 0); |
| 1788 | |
| 1789 | return !defined $ignore_type{$type}; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 1790 | } |
| 1791 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1792 | sub report { |
Joe Perches | cbec18a | 2014-04-03 14:49:19 -0700 | [diff] [blame] | 1793 | my ($level, $type, $msg) = @_; |
| 1794 | |
| 1795 | if (!show_type($type) || |
| 1796 | (defined $tst_only && $msg !~ /\Q$tst_only\E/)) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1797 | return 0; |
| 1798 | } |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 1799 | my $output = ''; |
| 1800 | if (-t STDOUT && $color) { |
| 1801 | if ($level eq 'ERROR') { |
| 1802 | $output .= RED; |
| 1803 | } elsif ($level eq 'WARNING') { |
| 1804 | $output .= YELLOW; |
| 1805 | } else { |
| 1806 | $output .= GREEN; |
| 1807 | } |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 1808 | } |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 1809 | $output .= $prefix . $level . ':'; |
| 1810 | if ($show_types) { |
| 1811 | $output .= BLUE if (-t STDOUT && $color); |
| 1812 | $output .= "$type:"; |
| 1813 | } |
| 1814 | $output .= RESET if (-t STDOUT && $color); |
| 1815 | $output .= ' ' . $msg . "\n"; |
Joe Perches | 34d8815 | 2015-06-25 15:03:05 -0700 | [diff] [blame] | 1816 | |
| 1817 | if ($showfile) { |
| 1818 | my @lines = split("\n", $output, -1); |
| 1819 | splice(@lines, 1, 1); |
| 1820 | $output = join("\n", @lines); |
| 1821 | } |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 1822 | $output = (split('\n', $output))[0] . "\n" if ($terse); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 1823 | |
Joe Perches | 5723029 | 2015-06-25 15:03:03 -0700 | [diff] [blame] | 1824 | push(our @report, $output); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1825 | |
| 1826 | return 1; |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1827 | } |
Joe Perches | cbec18a | 2014-04-03 14:49:19 -0700 | [diff] [blame] | 1828 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1829 | sub report_dump { |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 1830 | our @report; |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 1831 | } |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 1832 | |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 1833 | sub fixup_current_range { |
| 1834 | my ($lineRef, $offset, $length) = @_; |
| 1835 | |
| 1836 | if ($$lineRef =~ /^\@\@ -\d+,\d+ \+(\d+),(\d+) \@\@/) { |
| 1837 | my $o = $1; |
| 1838 | my $l = $2; |
| 1839 | my $no = $o + $offset; |
| 1840 | my $nl = $l + $length; |
| 1841 | $$lineRef =~ s/\+$o,$l \@\@/\+$no,$nl \@\@/; |
| 1842 | } |
| 1843 | } |
| 1844 | |
| 1845 | sub fix_inserted_deleted_lines { |
| 1846 | my ($linesRef, $insertedRef, $deletedRef) = @_; |
| 1847 | |
| 1848 | my $range_last_linenr = 0; |
| 1849 | my $delta_offset = 0; |
| 1850 | |
| 1851 | my $old_linenr = 0; |
| 1852 | my $new_linenr = 0; |
| 1853 | |
| 1854 | my $next_insert = 0; |
| 1855 | my $next_delete = 0; |
| 1856 | |
| 1857 | my @lines = (); |
| 1858 | |
| 1859 | my $inserted = @{$insertedRef}[$next_insert++]; |
| 1860 | my $deleted = @{$deletedRef}[$next_delete++]; |
| 1861 | |
| 1862 | foreach my $old_line (@{$linesRef}) { |
| 1863 | my $save_line = 1; |
| 1864 | my $line = $old_line; #don't modify the array |
Joe Perches | 323b267 | 2015-04-16 12:44:50 -0700 | [diff] [blame] | 1865 | if ($line =~ /^(?:\+\+\+|\-\-\-)\s+\S+/) { #new filename |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 1866 | $delta_offset = 0; |
| 1867 | } elsif ($line =~ /^\@\@ -\d+,\d+ \+\d+,\d+ \@\@/) { #new hunk |
| 1868 | $range_last_linenr = $new_linenr; |
| 1869 | fixup_current_range(\$line, $delta_offset, 0); |
| 1870 | } |
| 1871 | |
| 1872 | while (defined($deleted) && ${$deleted}{'LINENR'} == $old_linenr) { |
| 1873 | $deleted = @{$deletedRef}[$next_delete++]; |
| 1874 | $save_line = 0; |
| 1875 | fixup_current_range(\$lines[$range_last_linenr], $delta_offset--, -1); |
| 1876 | } |
| 1877 | |
| 1878 | while (defined($inserted) && ${$inserted}{'LINENR'} == $old_linenr) { |
| 1879 | push(@lines, ${$inserted}{'LINE'}); |
| 1880 | $inserted = @{$insertedRef}[$next_insert++]; |
| 1881 | $new_linenr++; |
| 1882 | fixup_current_range(\$lines[$range_last_linenr], $delta_offset++, 1); |
| 1883 | } |
| 1884 | |
| 1885 | if ($save_line) { |
| 1886 | push(@lines, $line); |
| 1887 | $new_linenr++; |
| 1888 | } |
| 1889 | |
| 1890 | $old_linenr++; |
| 1891 | } |
| 1892 | |
| 1893 | return @lines; |
| 1894 | } |
| 1895 | |
Joe Perches | f2d7e4d | 2014-08-06 16:11:07 -0700 | [diff] [blame] | 1896 | sub fix_insert_line { |
| 1897 | my ($linenr, $line) = @_; |
| 1898 | |
| 1899 | my $inserted = { |
| 1900 | LINENR => $linenr, |
| 1901 | LINE => $line, |
| 1902 | }; |
| 1903 | push(@fixed_inserted, $inserted); |
| 1904 | } |
| 1905 | |
| 1906 | sub fix_delete_line { |
| 1907 | my ($linenr, $line) = @_; |
| 1908 | |
| 1909 | my $deleted = { |
| 1910 | LINENR => $linenr, |
| 1911 | LINE => $line, |
| 1912 | }; |
| 1913 | |
| 1914 | push(@fixed_deleted, $deleted); |
| 1915 | } |
| 1916 | |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 1917 | sub ERROR { |
Joe Perches | cbec18a | 2014-04-03 14:49:19 -0700 | [diff] [blame] | 1918 | my ($type, $msg) = @_; |
| 1919 | |
| 1920 | if (report("ERROR", $type, $msg)) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1921 | our $clean = 0; |
| 1922 | our $cnt_error++; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1923 | return 1; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1924 | } |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1925 | return 0; |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 1926 | } |
| 1927 | sub WARN { |
Joe Perches | cbec18a | 2014-04-03 14:49:19 -0700 | [diff] [blame] | 1928 | my ($type, $msg) = @_; |
| 1929 | |
| 1930 | if (report("WARNING", $type, $msg)) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1931 | our $clean = 0; |
| 1932 | our $cnt_warn++; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1933 | return 1; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 1934 | } |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1935 | return 0; |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 1936 | } |
| 1937 | sub CHK { |
Joe Perches | cbec18a | 2014-04-03 14:49:19 -0700 | [diff] [blame] | 1938 | my ($type, $msg) = @_; |
| 1939 | |
| 1940 | if ($check && report("CHECK", $type, $msg)) { |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1941 | our $clean = 0; |
| 1942 | our $cnt_chk++; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1943 | return 1; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 1944 | } |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1945 | return 0; |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 1946 | } |
| 1947 | |
Andy Whitcroft | 6ecd967 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 1948 | sub check_absolute_file { |
| 1949 | my ($absolute, $herecurr) = @_; |
| 1950 | my $file = $absolute; |
| 1951 | |
| 1952 | ##print "absolute<$absolute>\n"; |
| 1953 | |
| 1954 | # See if any suffix of this path is a path within the tree. |
| 1955 | while ($file =~ s@^[^/]*/@@) { |
| 1956 | if (-f "$root/$file") { |
| 1957 | ##print "file<$file>\n"; |
| 1958 | last; |
| 1959 | } |
| 1960 | } |
| 1961 | if (! -f _) { |
| 1962 | return 0; |
| 1963 | } |
| 1964 | |
| 1965 | # It is, so see if the prefix is acceptable. |
| 1966 | my $prefix = $absolute; |
| 1967 | substr($prefix, -length($file)) = ''; |
| 1968 | |
| 1969 | ##print "prefix<$prefix>\n"; |
| 1970 | if ($prefix ne ".../") { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 1971 | WARN("USE_RELATIVE_PATH", |
| 1972 | "use relative pathname instead of absolute in changelog text\n" . $herecurr); |
Andy Whitcroft | 6ecd967 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 1973 | } |
| 1974 | } |
| 1975 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1976 | sub trim { |
| 1977 | my ($string) = @_; |
| 1978 | |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 1979 | $string =~ s/^\s+|\s+$//g; |
| 1980 | |
| 1981 | return $string; |
| 1982 | } |
| 1983 | |
| 1984 | sub ltrim { |
| 1985 | my ($string) = @_; |
| 1986 | |
| 1987 | $string =~ s/^\s+//; |
| 1988 | |
| 1989 | return $string; |
| 1990 | } |
| 1991 | |
| 1992 | sub rtrim { |
| 1993 | my ($string) = @_; |
| 1994 | |
| 1995 | $string =~ s/\s+$//; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 1996 | |
| 1997 | return $string; |
| 1998 | } |
| 1999 | |
Joe Perches | 52ea850 | 2013-11-12 15:10:09 -0800 | [diff] [blame] | 2000 | sub string_find_replace { |
| 2001 | my ($string, $find, $replace) = @_; |
| 2002 | |
| 2003 | $string =~ s/$find/$replace/g; |
| 2004 | |
| 2005 | return $string; |
| 2006 | } |
| 2007 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2008 | sub tabify { |
| 2009 | my ($leading) = @_; |
| 2010 | |
| 2011 | my $source_indent = 8; |
| 2012 | my $max_spaces_before_tab = $source_indent - 1; |
| 2013 | my $spaces_to_tab = " " x $source_indent; |
| 2014 | |
| 2015 | #convert leading spaces to tabs |
| 2016 | 1 while $leading =~ s@^([\t]*)$spaces_to_tab@$1\t@g; |
| 2017 | #Remove spaces before a tab |
| 2018 | 1 while $leading =~ s@^([\t]*)( {1,$max_spaces_before_tab})\t@$1\t@g; |
| 2019 | |
| 2020 | return "$leading"; |
| 2021 | } |
| 2022 | |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 2023 | sub pos_last_openparen { |
| 2024 | my ($line) = @_; |
| 2025 | |
| 2026 | my $pos = 0; |
| 2027 | |
| 2028 | my $opens = $line =~ tr/\(/\(/; |
| 2029 | my $closes = $line =~ tr/\)/\)/; |
| 2030 | |
| 2031 | my $last_openparen = 0; |
| 2032 | |
| 2033 | if (($opens == 0) || ($closes >= $opens)) { |
| 2034 | return -1; |
| 2035 | } |
| 2036 | |
| 2037 | my $len = length($line); |
| 2038 | |
| 2039 | for ($pos = 0; $pos < $len; $pos++) { |
| 2040 | my $string = substr($line, $pos); |
| 2041 | if ($string =~ /^($FuncArg|$balanced_parens)/) { |
| 2042 | $pos += length($1) - 1; |
| 2043 | } elsif (substr($line, $pos, 1) eq '(') { |
| 2044 | $last_openparen = $pos; |
| 2045 | } elsif (index($string, '(') == -1) { |
| 2046 | last; |
| 2047 | } |
| 2048 | } |
| 2049 | |
Joe Perches | 91cb519 | 2014-04-03 14:49:32 -0700 | [diff] [blame] | 2050 | return length(expand_tabs(substr($line, 0, $last_openparen))) + 1; |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 2051 | } |
| 2052 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2053 | sub process { |
| 2054 | my $filename = shift; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2055 | |
| 2056 | my $linenr=0; |
| 2057 | my $prevline=""; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2058 | my $prevrawline=""; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2059 | my $stashline=""; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2060 | my $stashrawline=""; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2061 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 2062 | my $length; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2063 | my $indent; |
| 2064 | my $previndent=0; |
| 2065 | my $stashindent=0; |
| 2066 | |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2067 | our $clean = 1; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2068 | my $signoff = 0; |
| 2069 | my $is_patch = 0; |
Joe Perches | 29ee1b0 | 2014-08-06 16:10:35 -0700 | [diff] [blame] | 2070 | my $in_header_lines = $file ? 0 : 1; |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 2071 | my $in_commit_log = 0; #Scanning lines before patch |
Allen Hubbe | ed43c4e | 2016-08-02 14:04:45 -0700 | [diff] [blame] | 2072 | my $has_commit_log = 0; #Encountered lines before patch |
Joe Perches | bf4daf1 | 2015-09-09 15:37:50 -0700 | [diff] [blame] | 2073 | my $commit_log_possible_stack_dump = 0; |
Joe Perches | 2a076f4 | 2015-04-16 12:44:28 -0700 | [diff] [blame] | 2074 | my $commit_log_long_line = 0; |
Joe Perches | e518e9a | 2015-06-25 15:03:27 -0700 | [diff] [blame] | 2075 | my $commit_log_has_diff = 0; |
Joe Perches | 13f1937 | 2014-08-06 16:10:59 -0700 | [diff] [blame] | 2076 | my $reported_maintainer_file = 0; |
Pasi Savanainen | fa64205d | 2012-10-04 17:13:29 -0700 | [diff] [blame] | 2077 | my $non_utf8_charset = 0; |
| 2078 | |
Joe Perches | 365dd4e | 2014-08-06 16:10:42 -0700 | [diff] [blame] | 2079 | my $last_blank_line = 0; |
Joe Perches | 5e4f6ba | 2014-12-10 15:52:05 -0800 | [diff] [blame] | 2080 | my $last_coalesced_string_linenr = -1; |
Joe Perches | 365dd4e | 2014-08-06 16:10:42 -0700 | [diff] [blame] | 2081 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 2082 | our @report = (); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 2083 | our $cnt_lines = 0; |
| 2084 | our $cnt_error = 0; |
| 2085 | our $cnt_warn = 0; |
| 2086 | our $cnt_chk = 0; |
| 2087 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2088 | # Trace the real file/line as we go. |
| 2089 | my $realfile = ''; |
| 2090 | my $realline = 0; |
| 2091 | my $realcnt = 0; |
| 2092 | my $here = ''; |
| 2093 | my $in_comment = 0; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2094 | my $comment_edge = 0; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2095 | my $first_line = 0; |
Wolfram Sang | 1e85572 | 2009-01-06 14:41:24 -0800 | [diff] [blame] | 2096 | my $p1_prefix = ''; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2097 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 2098 | my $prev_values = 'E'; |
| 2099 | |
| 2100 | # suppression flags |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2101 | my %suppress_ifbraces; |
Andy Whitcroft | 170d3a2 | 2008-10-15 22:02:30 -0700 | [diff] [blame] | 2102 | my %suppress_whiletrailers; |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 2103 | my %suppress_export; |
Andy Whitcroft | 3e469cd | 2012-01-10 15:10:01 -0800 | [diff] [blame] | 2104 | my $suppress_statement = 0; |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 2105 | |
Joe Perches | 7e51f19 | 2013-09-11 14:23:57 -0700 | [diff] [blame] | 2106 | my %signatures = (); |
Joe Perches | 323c126 | 2012-12-17 16:02:07 -0800 | [diff] [blame] | 2107 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2108 | # Pre-scan the patch sanitizing the lines. |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2109 | # Pre-scan the patch looking for any __setup documentation. |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2110 | # |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2111 | my @setup_docs = (); |
| 2112 | my $setup_docs = 0; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2113 | |
Joe Perches | d8b0771 | 2013-11-12 15:10:06 -0800 | [diff] [blame] | 2114 | my $camelcase_file_seeded = 0; |
| 2115 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2116 | sanitise_line_reset(); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2117 | my $line; |
| 2118 | foreach my $rawline (@rawlines) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2119 | $linenr++; |
| 2120 | $line = $rawline; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2121 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2122 | push(@fixed, $rawline) if ($fix); |
| 2123 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2124 | if ($rawline=~/^\+\+\+\s+(\S+)/) { |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2125 | $setup_docs = 0; |
| 2126 | if ($1 =~ m@Documentation/kernel-parameters.txt$@) { |
| 2127 | $setup_docs = 1; |
| 2128 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2129 | #next; |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2130 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2131 | if ($rawline=~/^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) { |
| 2132 | $realline=$1-1; |
| 2133 | if (defined $2) { |
| 2134 | $realcnt=$3+1; |
| 2135 | } else { |
| 2136 | $realcnt=1+1; |
| 2137 | } |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 2138 | $in_comment = 0; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2139 | |
| 2140 | # Guestimate if this is a continuing comment. Run |
| 2141 | # the context looking for a comment "edge". If this |
| 2142 | # edge is a close comment then we must be in a comment |
| 2143 | # at context start. |
| 2144 | my $edge; |
Andy Whitcroft | 01fa914 | 2008-10-15 22:02:19 -0700 | [diff] [blame] | 2145 | my $cnt = $realcnt; |
| 2146 | for (my $ln = $linenr + 1; $cnt > 0; $ln++) { |
| 2147 | next if (defined $rawlines[$ln - 1] && |
| 2148 | $rawlines[$ln - 1] =~ /^-/); |
| 2149 | $cnt--; |
| 2150 | #print "RAW<$rawlines[$ln - 1]>\n"; |
Andy Whitcroft | 721c1cb | 2009-01-06 14:41:16 -0800 | [diff] [blame] | 2151 | last if (!defined $rawlines[$ln - 1]); |
Andy Whitcroft | fae17da | 2009-01-06 14:41:20 -0800 | [diff] [blame] | 2152 | if ($rawlines[$ln - 1] =~ m@(/\*|\*/)@ && |
| 2153 | $rawlines[$ln - 1] !~ m@"[^"]*(?:/\*|\*/)[^"]*"@) { |
| 2154 | ($edge) = $1; |
| 2155 | last; |
| 2156 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2157 | } |
| 2158 | if (defined $edge && $edge eq '*/') { |
| 2159 | $in_comment = 1; |
| 2160 | } |
| 2161 | |
| 2162 | # Guestimate if this is a continuing comment. If this |
| 2163 | # is the start of a diff block and this line starts |
| 2164 | # ' *' then it is very likely a comment. |
| 2165 | if (!defined $edge && |
Andy Whitcroft | 83242e0 | 2009-01-06 14:41:17 -0800 | [diff] [blame] | 2166 | $rawlines[$linenr] =~ m@^.\s*(?:\*\*+| \*)(?:\s|$)@) |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2167 | { |
| 2168 | $in_comment = 1; |
| 2169 | } |
| 2170 | |
| 2171 | ##print "COMMENT:$in_comment edge<$edge> $rawline\n"; |
| 2172 | sanitise_line_reset($in_comment); |
| 2173 | |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 2174 | } elsif ($realcnt && $rawline =~ /^(?:\+| |$)/) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2175 | # Standardise the strings and chars within the input to |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 2176 | # simplify matching -- only bother with positive lines. |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2177 | $line = sanitise_line($rawline); |
| 2178 | } |
| 2179 | push(@lines, $line); |
| 2180 | |
| 2181 | if ($realcnt > 1) { |
| 2182 | $realcnt-- if ($line =~ /^(?:\+| |$)/); |
| 2183 | } else { |
| 2184 | $realcnt = 0; |
| 2185 | } |
| 2186 | |
| 2187 | #print "==>$rawline\n"; |
| 2188 | #print "-->$line\n"; |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2189 | |
| 2190 | if ($setup_docs && $line =~ /^\+/) { |
| 2191 | push(@setup_docs, $line); |
| 2192 | } |
| 2193 | } |
| 2194 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 2195 | $prefix = ''; |
| 2196 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2197 | $realcnt = 0; |
| 2198 | $linenr = 0; |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2199 | $fixlinenr = -1; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2200 | foreach my $line (@lines) { |
| 2201 | $linenr++; |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2202 | $fixlinenr++; |
Joe Perches | 1b5539b | 2013-09-11 14:24:03 -0700 | [diff] [blame] | 2203 | my $sline = $line; #copy of $line |
| 2204 | $sline =~ s/$;/ /g; #with comments as spaces |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2205 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2206 | my $rawline = $rawlines[$linenr - 1]; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 2207 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2208 | #extract the line range in the file after the patch is applied |
Joe Perches | e518e9a | 2015-06-25 15:03:27 -0700 | [diff] [blame] | 2209 | if (!$in_commit_log && |
| 2210 | $line =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) { |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2211 | $is_patch = 1; |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 2212 | $first_line = $linenr + 1; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2213 | $realline=$1-1; |
| 2214 | if (defined $2) { |
| 2215 | $realcnt=$3+1; |
| 2216 | } else { |
| 2217 | $realcnt=1+1; |
| 2218 | } |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2219 | annotate_reset(); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 2220 | $prev_values = 'E'; |
| 2221 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2222 | %suppress_ifbraces = (); |
Andy Whitcroft | 170d3a2 | 2008-10-15 22:02:30 -0700 | [diff] [blame] | 2223 | %suppress_whiletrailers = (); |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 2224 | %suppress_export = (); |
Andy Whitcroft | 3e469cd | 2012-01-10 15:10:01 -0800 | [diff] [blame] | 2225 | $suppress_statement = 0; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2226 | next; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2227 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 2228 | # track the line number as we move through the hunk, note that |
| 2229 | # new versions of GNU diff omit the leading space on completely |
| 2230 | # blank context lines so we need to count that too. |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2231 | } elsif ($line =~ /^( |\+|$)/) { |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2232 | $realline++; |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 2233 | $realcnt-- if ($realcnt != 0); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2234 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 2235 | # Measure the line length and indent. |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2236 | ($length, $indent) = line_stats($rawline); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2237 | |
| 2238 | # Track the previous line. |
| 2239 | ($prevline, $stashline) = ($stashline, $line); |
| 2240 | ($previndent, $stashindent) = ($stashindent, $indent); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2241 | ($prevrawline, $stashrawline) = ($stashrawline, $rawline); |
| 2242 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2243 | #warn "line<$line>\n"; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 2244 | |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 2245 | } elsif ($realcnt == 1) { |
| 2246 | $realcnt--; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2247 | } |
| 2248 | |
Andy Whitcroft | cc77cdc | 2009-10-26 16:50:13 -0700 | [diff] [blame] | 2249 | my $hunk_line = ($realcnt != 0); |
| 2250 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 2251 | $here = "#$linenr: " if (!$file); |
| 2252 | $here = "#$realline: " if ($file); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2253 | |
Joe Perches | 2ac73b4 | 2014-06-04 16:12:05 -0700 | [diff] [blame] | 2254 | my $found_file = 0; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2255 | # extract the filename as it passes |
Rabin Vincent | 3bf9a00 | 2010-10-26 14:23:16 -0700 | [diff] [blame] | 2256 | if ($line =~ /^diff --git.*?(\S+)$/) { |
| 2257 | $realfile = $1; |
Joe Perches | 2b7ab45 | 2013-11-12 15:10:14 -0800 | [diff] [blame] | 2258 | $realfile =~ s@^([^/]*)/@@ if (!$file); |
Joe Perches | 270c49a | 2012-01-10 15:09:50 -0800 | [diff] [blame] | 2259 | $in_commit_log = 0; |
Joe Perches | 2ac73b4 | 2014-06-04 16:12:05 -0700 | [diff] [blame] | 2260 | $found_file = 1; |
Rabin Vincent | 3bf9a00 | 2010-10-26 14:23:16 -0700 | [diff] [blame] | 2261 | } elsif ($line =~ /^\+\+\+\s+(\S+)/) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2262 | $realfile = $1; |
Joe Perches | 2b7ab45 | 2013-11-12 15:10:14 -0800 | [diff] [blame] | 2263 | $realfile =~ s@^([^/]*)/@@ if (!$file); |
Joe Perches | 270c49a | 2012-01-10 15:09:50 -0800 | [diff] [blame] | 2264 | $in_commit_log = 0; |
Wolfram Sang | 1e85572 | 2009-01-06 14:41:24 -0800 | [diff] [blame] | 2265 | |
| 2266 | $p1_prefix = $1; |
Andy Whitcroft | e2f7aa4 | 2009-02-27 14:03:06 -0800 | [diff] [blame] | 2267 | if (!$file && $tree && $p1_prefix ne '' && |
| 2268 | -e "$root/$p1_prefix") { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2269 | WARN("PATCH_PREFIX", |
| 2270 | "patch prefix '$p1_prefix' exists, appears to be a -p0 patch\n"); |
Wolfram Sang | 1e85572 | 2009-01-06 14:41:24 -0800 | [diff] [blame] | 2271 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2272 | |
Andy Whitcroft | c1ab332 | 2008-10-15 22:02:20 -0700 | [diff] [blame] | 2273 | if ($realfile =~ m@^include/asm/@) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2274 | ERROR("MODIFIED_INCLUDE_ASM", |
| 2275 | "do not modify files in include/asm, change architecture specific files in include/asm-<architecture>\n" . "$here$rawline\n"); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2276 | } |
Joe Perches | 2ac73b4 | 2014-06-04 16:12:05 -0700 | [diff] [blame] | 2277 | $found_file = 1; |
| 2278 | } |
| 2279 | |
Joe Perches | 34d8815 | 2015-06-25 15:03:05 -0700 | [diff] [blame] | 2280 | #make up the handle for any error we report on this line |
| 2281 | if ($showfile) { |
| 2282 | $prefix = "$realfile:$realline: " |
| 2283 | } elsif ($emacs) { |
Joe Perches | 7d3a9f6 | 2015-09-09 15:37:39 -0700 | [diff] [blame] | 2284 | if ($file) { |
| 2285 | $prefix = "$filename:$realline: "; |
| 2286 | } else { |
| 2287 | $prefix = "$filename:$linenr: "; |
| 2288 | } |
Joe Perches | 34d8815 | 2015-06-25 15:03:05 -0700 | [diff] [blame] | 2289 | } |
| 2290 | |
Joe Perches | 2ac73b4 | 2014-06-04 16:12:05 -0700 | [diff] [blame] | 2291 | if ($found_file) { |
Joe Perches | 7bd7e48 | 2015-09-09 15:37:44 -0700 | [diff] [blame] | 2292 | if ($realfile =~ m@^(?:drivers/net/|net/|drivers/staging/)@) { |
Joe Perches | 2ac73b4 | 2014-06-04 16:12:05 -0700 | [diff] [blame] | 2293 | $check = 1; |
| 2294 | } else { |
| 2295 | $check = $check_orig; |
| 2296 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 2297 | next; |
| 2298 | } |
| 2299 | |
Randy Dunlap | 389834b | 2007-06-08 13:47:03 -0700 | [diff] [blame] | 2300 | $here .= "FILE: $realfile:$realline:" if ($realcnt != 0); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2301 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2302 | my $hereline = "$here\n$rawline\n"; |
| 2303 | my $herecurr = "$here\n$rawline\n"; |
| 2304 | my $hereprev = "$here\n$prevrawline\n$rawline\n"; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2305 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 2306 | $cnt_lines++ if ($realcnt != 0); |
| 2307 | |
Joe Perches | e518e9a | 2015-06-25 15:03:27 -0700 | [diff] [blame] | 2308 | # Check if the commit log has what seems like a diff which can confuse patch |
| 2309 | if ($in_commit_log && !$commit_log_has_diff && |
| 2310 | (($line =~ m@^\s+diff\b.*a/[\w/]+@ && |
| 2311 | $line =~ m@^\s+diff\b.*a/([\w/]+)\s+b/$1\b@) || |
| 2312 | $line =~ m@^\s*(?:\-\-\-\s+a/|\+\+\+\s+b/)@ || |
| 2313 | $line =~ m/^\s*\@\@ \-\d+,\d+ \+\d+,\d+ \@\@/)) { |
| 2314 | ERROR("DIFF_IN_COMMIT_MSG", |
| 2315 | "Avoid using diff content in the commit message - patch(1) might not work\n" . $herecurr); |
| 2316 | $commit_log_has_diff = 1; |
| 2317 | } |
| 2318 | |
Rabin Vincent | 3bf9a00 | 2010-10-26 14:23:16 -0700 | [diff] [blame] | 2319 | # Check for incorrect file permissions |
| 2320 | if ($line =~ /^new (file )?mode.*[7531]\d{0,2}$/) { |
| 2321 | my $permhere = $here . "FILE: $realfile\n"; |
Joe Perches | 04db4d2 | 2013-04-29 16:18:14 -0700 | [diff] [blame] | 2322 | if ($realfile !~ m@scripts/@ && |
| 2323 | $realfile !~ /\.(py|pl|awk|sh)$/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2324 | ERROR("EXECUTE_PERMISSIONS", |
| 2325 | "do not set execute permissions for source files\n" . $permhere); |
Rabin Vincent | 3bf9a00 | 2010-10-26 14:23:16 -0700 | [diff] [blame] | 2326 | } |
| 2327 | } |
| 2328 | |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2329 | # Check the patch for a signoff: |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 2330 | if ($line =~ /^\s*signed-off-by:/i) { |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 2331 | $signoff++; |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 2332 | $in_commit_log = 0; |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2333 | } |
| 2334 | |
Joe Perches | e0d975b | 2014-12-10 15:51:49 -0800 | [diff] [blame] | 2335 | # Check if MAINTAINERS is being updated. If so, there's probably no need to |
| 2336 | # emit the "does MAINTAINERS need updating?" message on file add/move/delete |
| 2337 | if ($line =~ /^\s*MAINTAINERS\s*\|/) { |
| 2338 | $reported_maintainer_file = 1; |
| 2339 | } |
| 2340 | |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2341 | # Check signature styles |
Joe Perches | 270c49a | 2012-01-10 15:09:50 -0800 | [diff] [blame] | 2342 | if (!$in_header_lines && |
Joe Perches | ce0338df3c | 2012-07-30 14:41:18 -0700 | [diff] [blame] | 2343 | $line =~ /^(\s*)([a-z0-9_-]+by:|$signature_tags)(\s*)(.*)/i) { |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2344 | my $space_before = $1; |
| 2345 | my $sign_off = $2; |
| 2346 | my $space_after = $3; |
| 2347 | my $email = $4; |
| 2348 | my $ucfirst_sign_off = ucfirst(lc($sign_off)); |
| 2349 | |
Joe Perches | ce0338df3c | 2012-07-30 14:41:18 -0700 | [diff] [blame] | 2350 | if ($sign_off !~ /$signature_tags/) { |
| 2351 | WARN("BAD_SIGN_OFF", |
| 2352 | "Non-standard signature: $sign_off\n" . $herecurr); |
| 2353 | } |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2354 | if (defined $space_before && $space_before ne "") { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2355 | if (WARN("BAD_SIGN_OFF", |
| 2356 | "Do not use whitespace before $ucfirst_sign_off\n" . $herecurr) && |
| 2357 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2358 | $fixed[$fixlinenr] = |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2359 | "$ucfirst_sign_off $email"; |
| 2360 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2361 | } |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2362 | if ($sign_off =~ /-by:$/i && $sign_off ne $ucfirst_sign_off) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2363 | if (WARN("BAD_SIGN_OFF", |
| 2364 | "'$ucfirst_sign_off' is the preferred signature form\n" . $herecurr) && |
| 2365 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2366 | $fixed[$fixlinenr] = |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2367 | "$ucfirst_sign_off $email"; |
| 2368 | } |
| 2369 | |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2370 | } |
| 2371 | if (!defined $space_after || $space_after ne " ") { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2372 | if (WARN("BAD_SIGN_OFF", |
| 2373 | "Use a single space after $ucfirst_sign_off\n" . $herecurr) && |
| 2374 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2375 | $fixed[$fixlinenr] = |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2376 | "$ucfirst_sign_off $email"; |
| 2377 | } |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2378 | } |
| 2379 | |
| 2380 | my ($email_name, $email_address, $comment) = parse_email($email); |
| 2381 | my $suggested_email = format_email(($email_name, $email_address)); |
| 2382 | if ($suggested_email eq "") { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2383 | ERROR("BAD_SIGN_OFF", |
| 2384 | "Unrecognized email address: '$email'\n" . $herecurr); |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2385 | } else { |
| 2386 | my $dequoted = $suggested_email; |
| 2387 | $dequoted =~ s/^"//; |
| 2388 | $dequoted =~ s/" </ </; |
| 2389 | # Don't force email to have quotes |
| 2390 | # Allow just an angle bracketed address |
| 2391 | if ("$dequoted$comment" ne $email && |
| 2392 | "<$email_address>$comment" ne $email && |
| 2393 | "$suggested_email$comment" ne $email) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2394 | WARN("BAD_SIGN_OFF", |
| 2395 | "email address '$email' might be better as '$suggested_email$comment'\n" . $herecurr); |
Joe Perches | 2011247 | 2011-07-25 17:13:23 -0700 | [diff] [blame] | 2396 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2397 | } |
Joe Perches | 7e51f19 | 2013-09-11 14:23:57 -0700 | [diff] [blame] | 2398 | |
| 2399 | # Check for duplicate signatures |
| 2400 | my $sig_nospace = $line; |
| 2401 | $sig_nospace =~ s/\s//g; |
| 2402 | $sig_nospace = lc($sig_nospace); |
| 2403 | if (defined $signatures{$sig_nospace}) { |
| 2404 | WARN("BAD_SIGN_OFF", |
| 2405 | "Duplicate signature\n" . $herecurr); |
| 2406 | } else { |
| 2407 | $signatures{$sig_nospace} = 1; |
| 2408 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2409 | } |
| 2410 | |
Joe Perches | a2fe16b | 2015-02-13 14:39:02 -0800 | [diff] [blame] | 2411 | # Check email subject for common tools that don't need to be mentioned |
| 2412 | if ($in_header_lines && |
| 2413 | $line =~ /^Subject:.*\b(?:checkpatch|sparse|smatch)\b[^:]/i) { |
| 2414 | WARN("EMAIL_SUBJECT", |
| 2415 | "A patch subject line should describe the change not the tool that found it\n" . $herecurr); |
| 2416 | } |
| 2417 | |
Joe Perches | 9b3189e | 2014-06-04 16:12:10 -0700 | [diff] [blame] | 2418 | # Check for old stable address |
| 2419 | if ($line =~ /^\s*cc:\s*.*<?\bstable\@kernel\.org\b>?.*$/i) { |
| 2420 | ERROR("STABLE_ADDRESS", |
| 2421 | "The 'stable' address should be 'stable\@vger.kernel.org'\n" . $herecurr); |
| 2422 | } |
| 2423 | |
Christopher Covington | 7ebd05e | 2014-04-03 14:49:31 -0700 | [diff] [blame] | 2424 | # Check for unwanted Gerrit info |
| 2425 | if ($in_commit_log && $line =~ /^\s*change-id:/i) { |
| 2426 | ERROR("GERRIT_CHANGE_ID", |
| 2427 | "Remove Gerrit Change-Id's before submitting upstream.\n" . $herecurr); |
| 2428 | } |
| 2429 | |
Joe Perches | 369c8dd | 2015-11-06 16:31:34 -0800 | [diff] [blame] | 2430 | # Check if the commit log is in a possible stack dump |
| 2431 | if ($in_commit_log && !$commit_log_possible_stack_dump && |
| 2432 | ($line =~ /^\s*(?:WARNING:|BUG:)/ || |
| 2433 | $line =~ /^\s*\[\s*\d+\.\d{6,6}\s*\]/ || |
| 2434 | # timestamp |
| 2435 | $line =~ /^\s*\[\<[0-9a-fA-F]{8,}\>\]/)) { |
| 2436 | # stack dump address |
| 2437 | $commit_log_possible_stack_dump = 1; |
| 2438 | } |
| 2439 | |
Joe Perches | 2a076f4 | 2015-04-16 12:44:28 -0700 | [diff] [blame] | 2440 | # Check for line lengths > 75 in commit log, warn once |
| 2441 | if ($in_commit_log && !$commit_log_long_line && |
Joe Perches | 369c8dd | 2015-11-06 16:31:34 -0800 | [diff] [blame] | 2442 | length($line) > 75 && |
| 2443 | !($line =~ /^\s*[a-zA-Z0-9_\/\.]+\s+\|\s+\d+/ || |
| 2444 | # file delta changes |
| 2445 | $line =~ /^\s*(?:[\w\.\-]+\/)++[\w\.\-]+:/ || |
| 2446 | # filename then : |
| 2447 | $line =~ /^\s*(?:Fixes:|Link:)/i || |
| 2448 | # A Fixes: or Link: line |
| 2449 | $commit_log_possible_stack_dump)) { |
Joe Perches | 2a076f4 | 2015-04-16 12:44:28 -0700 | [diff] [blame] | 2450 | WARN("COMMIT_LOG_LONG_LINE", |
| 2451 | "Possible unwrapped commit description (prefer a maximum 75 chars per line)\n" . $herecurr); |
| 2452 | $commit_log_long_line = 1; |
| 2453 | } |
| 2454 | |
Joe Perches | bf4daf1 | 2015-09-09 15:37:50 -0700 | [diff] [blame] | 2455 | # Reset possible stack dump if a blank line is found |
Joe Perches | 369c8dd | 2015-11-06 16:31:34 -0800 | [diff] [blame] | 2456 | if ($in_commit_log && $commit_log_possible_stack_dump && |
| 2457 | $line =~ /^\s*$/) { |
| 2458 | $commit_log_possible_stack_dump = 0; |
| 2459 | } |
Joe Perches | bf4daf1 | 2015-09-09 15:37:50 -0700 | [diff] [blame] | 2460 | |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 2461 | # Check for git id commit length and improperly formed commit descriptions |
Joe Perches | 369c8dd | 2015-11-06 16:31:34 -0800 | [diff] [blame] | 2462 | if ($in_commit_log && !$commit_log_possible_stack_dump && |
Joe Perches | aab38f5 | 2016-08-02 14:04:36 -0700 | [diff] [blame] | 2463 | $line !~ /^\s*(?:Link|Patchwork|http|https|BugLink):/i && |
Joe Perches | fe043ea | 2015-09-09 15:37:25 -0700 | [diff] [blame] | 2464 | ($line =~ /\bcommit\s+[0-9a-f]{5,}\b/i || |
Joe Perches | aab38f5 | 2016-08-02 14:04:36 -0700 | [diff] [blame] | 2465 | ($line =~ /(?:\s|^)[0-9a-f]{12,40}(?:[\s"'\(\[]|$)/i && |
Joe Perches | 369c8dd | 2015-11-06 16:31:34 -0800 | [diff] [blame] | 2466 | $line !~ /[\<\[][0-9a-f]{12,40}[\>\]]/i && |
| 2467 | $line !~ /\bfixes:\s*[0-9a-f]{12,40}/i))) { |
Joe Perches | fe043ea | 2015-09-09 15:37:25 -0700 | [diff] [blame] | 2468 | my $init_char = "c"; |
| 2469 | my $orig_commit = ""; |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 2470 | my $short = 1; |
| 2471 | my $long = 0; |
| 2472 | my $case = 1; |
| 2473 | my $space = 1; |
| 2474 | my $hasdesc = 0; |
Joe Perches | 19c146a | 2015-02-13 14:39:00 -0800 | [diff] [blame] | 2475 | my $hasparens = 0; |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 2476 | my $id = '0123456789ab'; |
| 2477 | my $orig_desc = "commit description"; |
| 2478 | my $description = ""; |
| 2479 | |
Joe Perches | fe043ea | 2015-09-09 15:37:25 -0700 | [diff] [blame] | 2480 | if ($line =~ /\b(c)ommit\s+([0-9a-f]{5,})\b/i) { |
| 2481 | $init_char = $1; |
| 2482 | $orig_commit = lc($2); |
| 2483 | } elsif ($line =~ /\b([0-9a-f]{12,40})\b/i) { |
| 2484 | $orig_commit = lc($1); |
| 2485 | } |
| 2486 | |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 2487 | $short = 0 if ($line =~ /\bcommit\s+[0-9a-f]{12,40}/i); |
| 2488 | $long = 1 if ($line =~ /\bcommit\s+[0-9a-f]{41,}/i); |
| 2489 | $space = 0 if ($line =~ /\bcommit [0-9a-f]/i); |
| 2490 | $case = 0 if ($line =~ /\b[Cc]ommit\s+[0-9a-f]{5,40}[^A-F]/); |
| 2491 | if ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)"\)/i) { |
| 2492 | $orig_desc = $1; |
Joe Perches | 19c146a | 2015-02-13 14:39:00 -0800 | [diff] [blame] | 2493 | $hasparens = 1; |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 2494 | } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s*$/i && |
| 2495 | defined $rawlines[$linenr] && |
| 2496 | $rawlines[$linenr] =~ /^\s*\("([^"]+)"\)/) { |
| 2497 | $orig_desc = $1; |
Joe Perches | 19c146a | 2015-02-13 14:39:00 -0800 | [diff] [blame] | 2498 | $hasparens = 1; |
Joe Perches | b671fde | 2015-02-13 14:38:41 -0800 | [diff] [blame] | 2499 | } elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("[^"]+$/i && |
| 2500 | defined $rawlines[$linenr] && |
| 2501 | $rawlines[$linenr] =~ /^\s*[^"]+"\)/) { |
| 2502 | $line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)$/i; |
| 2503 | $orig_desc = $1; |
| 2504 | $rawlines[$linenr] =~ /^\s*([^"]+)"\)/; |
| 2505 | $orig_desc .= " " . $1; |
Joe Perches | 19c146a | 2015-02-13 14:39:00 -0800 | [diff] [blame] | 2506 | $hasparens = 1; |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 2507 | } |
| 2508 | |
| 2509 | ($id, $description) = git_commit_info($orig_commit, |
| 2510 | $id, $orig_desc); |
| 2511 | |
Joe Perches | 19c146a | 2015-02-13 14:39:00 -0800 | [diff] [blame] | 2512 | if ($short || $long || $space || $case || ($orig_desc ne $description) || !$hasparens) { |
Joe Perches | 0d7835f | 2015-02-13 14:38:35 -0800 | [diff] [blame] | 2513 | ERROR("GIT_COMMIT_ID", |
| 2514 | "Please use git commit description style 'commit <12+ chars of sha1> (\"<title line>\")' - ie: '${init_char}ommit $id (\"$description\")'\n" . $herecurr); |
| 2515 | } |
Joe Perches | d311cd4 | 2014-08-06 16:10:57 -0700 | [diff] [blame] | 2516 | } |
| 2517 | |
Joe Perches | 13f1937 | 2014-08-06 16:10:59 -0700 | [diff] [blame] | 2518 | # Check for added, moved or deleted files |
| 2519 | if (!$reported_maintainer_file && !$in_commit_log && |
| 2520 | ($line =~ /^(?:new|deleted) file mode\s*\d+\s*$/ || |
| 2521 | $line =~ /^rename (?:from|to) [\w\/\.\-]+\s*$/ || |
| 2522 | ($line =~ /\{\s*([\w\/\.\-]*)\s*\=\>\s*([\w\/\.\-]*)\s*\}/ && |
| 2523 | (defined($1) || defined($2))))) { |
| 2524 | $reported_maintainer_file = 1; |
| 2525 | WARN("FILE_PATH_CHANGES", |
| 2526 | "added, moved or deleted file(s), does MAINTAINERS need updating?\n" . $herecurr); |
| 2527 | } |
| 2528 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 2529 | # Check for wrappage within a valid hunk of the file |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 2530 | if ($realcnt != 0 && $line !~ m{^(?:\+|-| |\\ No newline|$)}) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2531 | ERROR("CORRUPTED_PATCH", |
| 2532 | "patch seems to be corrupt (line wrapped?)\n" . |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 2533 | $herecurr) if (!$emitted_corrupt++); |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2534 | } |
| 2535 | |
Andy Whitcroft | 6ecd967 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 2536 | # Check for absolute kernel paths. |
| 2537 | if ($tree) { |
| 2538 | while ($line =~ m{(?:^|\s)(/\S*)}g) { |
| 2539 | my $file = $1; |
| 2540 | |
| 2541 | if ($file =~ m{^(.*?)(?::\d+)+:?$} && |
| 2542 | check_absolute_file($1, $herecurr)) { |
| 2543 | # |
| 2544 | } else { |
| 2545 | check_absolute_file($file, $herecurr); |
| 2546 | } |
| 2547 | } |
| 2548 | } |
| 2549 | |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 2550 | # UTF-8 regex found at http://www.w3.org/International/questions/qa-forms-utf-8.en.php |
| 2551 | if (($realfile =~ /^$/ || $line =~ /^\+/) && |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 2552 | $rawline !~ m/^$UTF8*$/) { |
| 2553 | my ($utf8_prefix) = ($rawline =~ /^($UTF8*)/); |
| 2554 | |
| 2555 | my $blank = copy_spacing($rawline); |
| 2556 | my $ptr = substr($blank, 0, length($utf8_prefix)) . "^"; |
| 2557 | my $hereptr = "$hereline$ptr\n"; |
| 2558 | |
Joe Perches | 34d9921 | 2011-07-25 17:13:26 -0700 | [diff] [blame] | 2559 | CHK("INVALID_UTF8", |
| 2560 | "Invalid UTF-8, patch and commit message should be encoded in UTF-8\n" . $hereptr); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 2561 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2562 | |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 2563 | # Check if it's the start of a commit log |
| 2564 | # (not a header line and we haven't seen the patch filename) |
| 2565 | if ($in_header_lines && $realfile =~ /^$/ && |
Joe Perches | 29ee1b0 | 2014-08-06 16:10:35 -0700 | [diff] [blame] | 2566 | !($rawline =~ /^\s+\S/ || |
| 2567 | $rawline =~ /^(commit\b|from\b|[\w-]+:).*$/i)) { |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 2568 | $in_header_lines = 0; |
| 2569 | $in_commit_log = 1; |
Allen Hubbe | ed43c4e | 2016-08-02 14:04:45 -0700 | [diff] [blame] | 2570 | $has_commit_log = 1; |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 2571 | } |
| 2572 | |
Pasi Savanainen | fa64205d | 2012-10-04 17:13:29 -0700 | [diff] [blame] | 2573 | # Check if there is UTF-8 in a commit log when a mail header has explicitly |
| 2574 | # declined it, i.e defined some charset where it is missing. |
| 2575 | if ($in_header_lines && |
| 2576 | $rawline =~ /^Content-Type:.+charset="(.+)".*$/ && |
| 2577 | $1 !~ /utf-8/i) { |
| 2578 | $non_utf8_charset = 1; |
| 2579 | } |
| 2580 | |
| 2581 | if ($in_commit_log && $non_utf8_charset && $realfile =~ /^$/ && |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 2582 | $rawline =~ /$NON_ASCII_UTF8/) { |
Pasi Savanainen | fa64205d | 2012-10-04 17:13:29 -0700 | [diff] [blame] | 2583 | WARN("UTF8_BEFORE_PATCH", |
Joe Perches | 15662b3 | 2011-10-31 17:13:12 -0700 | [diff] [blame] | 2584 | "8-bit UTF-8 used in possible commit log\n" . $herecurr); |
| 2585 | } |
| 2586 | |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 2587 | # Check for various typo / spelling mistakes |
Joe Perches | 66d7a38 | 2015-04-16 12:44:08 -0700 | [diff] [blame] | 2588 | if (defined($misspellings) && |
| 2589 | ($in_commit_log || $line =~ /^(?:\+|Subject:)/i)) { |
Joe Perches | ebfd7d6 | 2015-04-16 12:44:14 -0700 | [diff] [blame] | 2590 | while ($rawline =~ /(?:^|[^a-z@])($misspellings)(?:\b|$|[^a-z@])/gi) { |
Kees Cook | 66b47b4 | 2014-10-13 15:51:57 -0700 | [diff] [blame] | 2591 | my $typo = $1; |
| 2592 | my $typo_fix = $spelling_fix{lc($typo)}; |
| 2593 | $typo_fix = ucfirst($typo_fix) if ($typo =~ /^[A-Z]/); |
| 2594 | $typo_fix = uc($typo_fix) if ($typo =~ /^[A-Z]+$/); |
| 2595 | my $msg_type = \&WARN; |
| 2596 | $msg_type = \&CHK if ($file); |
| 2597 | if (&{$msg_type}("TYPO_SPELLING", |
| 2598 | "'$typo' may be misspelled - perhaps '$typo_fix'?\n" . $herecurr) && |
| 2599 | $fix) { |
| 2600 | $fixed[$fixlinenr] =~ s/(^|[^A-Za-z@])($typo)($|[^A-Za-z@])/$1$typo_fix$3/; |
| 2601 | } |
| 2602 | } |
| 2603 | } |
| 2604 | |
Andy Whitcroft | 30670854 | 2008-10-15 22:02:28 -0700 | [diff] [blame] | 2605 | # ignore non-hunk lines and lines being removed |
| 2606 | next if (!$hunk_line || $line =~ /^-/); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 2607 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2608 | #trailing whitespace |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 2609 | if ($line =~ /^\+.*\015/) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2610 | my $herevet = "$here\n" . cat_vet($rawline) . "\n"; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 2611 | if (ERROR("DOS_LINE_ENDINGS", |
| 2612 | "DOS line endings\n" . $herevet) && |
| 2613 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2614 | $fixed[$fixlinenr] =~ s/[\s\015]+$//; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 2615 | } |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2616 | } elsif ($rawline =~ /^\+.*\S\s+$/ || $rawline =~ /^\+\s+$/) { |
| 2617 | my $herevet = "$here\n" . cat_vet($rawline) . "\n"; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2618 | if (ERROR("TRAILING_WHITESPACE", |
| 2619 | "trailing whitespace\n" . $herevet) && |
| 2620 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2621 | $fixed[$fixlinenr] =~ s/\s+$//; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2622 | } |
| 2623 | |
Andy Whitcroft | d2c0a23 | 2010-10-26 14:23:12 -0700 | [diff] [blame] | 2624 | $rpt_cleaners = 1; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2625 | } |
Andy Whitcroft | 5368df20 | 2008-10-15 22:02:27 -0700 | [diff] [blame] | 2626 | |
Josh Triplett | 4783f89 | 2013-11-12 15:10:12 -0800 | [diff] [blame] | 2627 | # Check for FSF mailing addresses. |
Alexander Duyck | 109d8cb | 2014-01-23 15:54:50 -0800 | [diff] [blame] | 2628 | if ($rawline =~ /\bwrite to the Free/i || |
Joe Perches | 3e2232f | 2014-01-23 15:54:48 -0800 | [diff] [blame] | 2629 | $rawline =~ /\b59\s+Temple\s+Pl/i || |
| 2630 | $rawline =~ /\b51\s+Franklin\s+St/i) { |
Josh Triplett | 4783f89 | 2013-11-12 15:10:12 -0800 | [diff] [blame] | 2631 | my $herevet = "$here\n" . cat_vet($rawline) . "\n"; |
| 2632 | my $msg_type = \&ERROR; |
| 2633 | $msg_type = \&CHK if ($file); |
| 2634 | &{$msg_type}("FSF_MAILING_ADDRESS", |
Joe Perches | 3e2232f | 2014-01-23 15:54:48 -0800 | [diff] [blame] | 2635 | "Do not include the paragraph about writing to the Free Software Foundation's mailing address from the sample GPL notice. The FSF has changed addresses in the past, and may do so again. Linux already includes a copy of the GPL.\n" . $herevet) |
Josh Triplett | 4783f89 | 2013-11-12 15:10:12 -0800 | [diff] [blame] | 2636 | } |
| 2637 | |
Andi Kleen | 3354957 | 2010-05-24 14:33:29 -0700 | [diff] [blame] | 2638 | # check for Kconfig help text having a real description |
Andy Whitcroft | 9fe287d7 | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 2639 | # Only applies when adding the entry originally, after that we do not have |
| 2640 | # sufficient context to determine whether it is indeed long enough. |
Andi Kleen | 3354957 | 2010-05-24 14:33:29 -0700 | [diff] [blame] | 2641 | if ($realfile =~ /Kconfig/ && |
Joe Perches | 8d73e0e | 2014-08-06 16:10:46 -0700 | [diff] [blame] | 2642 | $line =~ /^\+\s*config\s+/) { |
Andi Kleen | 3354957 | 2010-05-24 14:33:29 -0700 | [diff] [blame] | 2643 | my $length = 0; |
Andy Whitcroft | 9fe287d7 | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 2644 | my $cnt = $realcnt; |
| 2645 | my $ln = $linenr + 1; |
| 2646 | my $f; |
Andy Whitcroft | a138580 | 2012-01-10 15:10:03 -0800 | [diff] [blame] | 2647 | my $is_start = 0; |
Andy Whitcroft | 9fe287d7 | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 2648 | my $is_end = 0; |
Andy Whitcroft | a138580 | 2012-01-10 15:10:03 -0800 | [diff] [blame] | 2649 | for (; $cnt > 0 && defined $lines[$ln - 1]; $ln++) { |
Andy Whitcroft | 9fe287d7 | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 2650 | $f = $lines[$ln - 1]; |
| 2651 | $cnt-- if ($lines[$ln - 1] !~ /^-/); |
| 2652 | $is_end = $lines[$ln - 1] =~ /^\+/; |
Andy Whitcroft | 9fe287d7 | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 2653 | |
| 2654 | next if ($f =~ /^-/); |
Joe Perches | 8d73e0e | 2014-08-06 16:10:46 -0700 | [diff] [blame] | 2655 | last if (!$file && $f =~ /^\@\@/); |
Andy Whitcroft | a138580 | 2012-01-10 15:10:03 -0800 | [diff] [blame] | 2656 | |
Joe Perches | 8d73e0e | 2014-08-06 16:10:46 -0700 | [diff] [blame] | 2657 | if ($lines[$ln - 1] =~ /^\+\s*(?:bool|tristate)\s*\"/) { |
Andy Whitcroft | a138580 | 2012-01-10 15:10:03 -0800 | [diff] [blame] | 2658 | $is_start = 1; |
Joe Perches | 8d73e0e | 2014-08-06 16:10:46 -0700 | [diff] [blame] | 2659 | } elsif ($lines[$ln - 1] =~ /^\+\s*(?:---)?help(?:---)?$/) { |
Andy Whitcroft | a138580 | 2012-01-10 15:10:03 -0800 | [diff] [blame] | 2660 | $length = -1; |
| 2661 | } |
| 2662 | |
Andy Whitcroft | 9fe287d7 | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 2663 | $f =~ s/^.//; |
Andi Kleen | 3354957 | 2010-05-24 14:33:29 -0700 | [diff] [blame] | 2664 | $f =~ s/#.*//; |
| 2665 | $f =~ s/^\s+//; |
| 2666 | next if ($f =~ /^$/); |
Andy Whitcroft | 9fe287d7 | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 2667 | if ($f =~ /^\s*config\s/) { |
| 2668 | $is_end = 1; |
| 2669 | last; |
| 2670 | } |
Andi Kleen | 3354957 | 2010-05-24 14:33:29 -0700 | [diff] [blame] | 2671 | $length++; |
| 2672 | } |
Vadim Bendebury | 5619327 | 2014-10-13 15:51:48 -0700 | [diff] [blame] | 2673 | if ($is_start && $is_end && $length < $min_conf_desc_length) { |
| 2674 | WARN("CONFIG_DESCRIPTION", |
| 2675 | "please write a paragraph that describes the config symbol fully\n" . $herecurr); |
| 2676 | } |
Andy Whitcroft | a138580 | 2012-01-10 15:10:03 -0800 | [diff] [blame] | 2677 | #print "is_start<$is_start> is_end<$is_end> length<$length>\n"; |
Andi Kleen | 3354957 | 2010-05-24 14:33:29 -0700 | [diff] [blame] | 2678 | } |
| 2679 | |
Kees Cook | 1ba8dfd | 2012-12-17 16:01:48 -0800 | [diff] [blame] | 2680 | # discourage the addition of CONFIG_EXPERIMENTAL in Kconfig. |
| 2681 | if ($realfile =~ /Kconfig/ && |
| 2682 | $line =~ /.\s*depends on\s+.*\bEXPERIMENTAL\b/) { |
| 2683 | WARN("CONFIG_EXPERIMENTAL", |
| 2684 | "Use of CONFIG_EXPERIMENTAL is deprecated. For alternatives, see https://lkml.org/lkml/2012/10/23/580\n"); |
| 2685 | } |
| 2686 | |
Christoph Jaeger | 327953e | 2015-02-13 14:38:29 -0800 | [diff] [blame] | 2687 | # discourage the use of boolean for type definition attributes of Kconfig options |
| 2688 | if ($realfile =~ /Kconfig/ && |
| 2689 | $line =~ /^\+\s*\bboolean\b/) { |
| 2690 | WARN("CONFIG_TYPE_BOOLEAN", |
| 2691 | "Use of boolean is deprecated, please use bool instead.\n" . $herecurr); |
| 2692 | } |
| 2693 | |
Arnaud Lacombe | c68e587 | 2011-08-15 01:07:14 -0400 | [diff] [blame] | 2694 | if (($realfile =~ /Makefile.*/ || $realfile =~ /Kbuild.*/) && |
| 2695 | ($line =~ /\+(EXTRA_[A-Z]+FLAGS).*/)) { |
| 2696 | my $flag = $1; |
| 2697 | my $replacement = { |
| 2698 | 'EXTRA_AFLAGS' => 'asflags-y', |
| 2699 | 'EXTRA_CFLAGS' => 'ccflags-y', |
| 2700 | 'EXTRA_CPPFLAGS' => 'cppflags-y', |
| 2701 | 'EXTRA_LDFLAGS' => 'ldflags-y', |
| 2702 | }; |
| 2703 | |
| 2704 | WARN("DEPRECATED_VARIABLE", |
| 2705 | "Use of $flag is deprecated, please use \`$replacement->{$flag} instead.\n" . $herecurr) if ($replacement->{$flag}); |
| 2706 | } |
| 2707 | |
Rob Herring | bff5da4 | 2014-01-23 15:54:51 -0800 | [diff] [blame] | 2708 | # check for DT compatible documentation |
Florian Vaussard | 7dd05b3 | 2014-04-03 14:49:26 -0700 | [diff] [blame] | 2709 | if (defined $root && |
| 2710 | (($realfile =~ /\.dtsi?$/ && $line =~ /^\+\s*compatible\s*=\s*\"/) || |
| 2711 | ($realfile =~ /\.[ch]$/ && $line =~ /^\+.*\.compatible\s*=\s*\"/))) { |
| 2712 | |
Rob Herring | bff5da4 | 2014-01-23 15:54:51 -0800 | [diff] [blame] | 2713 | my @compats = $rawline =~ /\"([a-zA-Z0-9\-\,\.\+_]+)\"/g; |
| 2714 | |
Florian Vaussard | cc93319 | 2014-04-03 14:49:27 -0700 | [diff] [blame] | 2715 | my $dt_path = $root . "/Documentation/devicetree/bindings/"; |
| 2716 | my $vp_file = $dt_path . "vendor-prefixes.txt"; |
| 2717 | |
Rob Herring | bff5da4 | 2014-01-23 15:54:51 -0800 | [diff] [blame] | 2718 | foreach my $compat (@compats) { |
| 2719 | my $compat2 = $compat; |
Rob Herring | 185d566 | 2014-06-04 16:12:03 -0700 | [diff] [blame] | 2720 | $compat2 =~ s/\,[a-zA-Z0-9]*\-/\,<\.\*>\-/; |
| 2721 | my $compat3 = $compat; |
| 2722 | $compat3 =~ s/\,([a-z]*)[0-9]*\-/\,$1<\.\*>\-/; |
| 2723 | `grep -Erq "$compat|$compat2|$compat3" $dt_path`; |
Rob Herring | bff5da4 | 2014-01-23 15:54:51 -0800 | [diff] [blame] | 2724 | if ( $? >> 8 ) { |
| 2725 | WARN("UNDOCUMENTED_DT_STRING", |
| 2726 | "DT compatible string \"$compat\" appears un-documented -- check $dt_path\n" . $herecurr); |
| 2727 | } |
| 2728 | |
Florian Vaussard | 4fbf32a | 2014-04-03 14:49:25 -0700 | [diff] [blame] | 2729 | next if $compat !~ /^([a-zA-Z0-9\-]+)\,/; |
| 2730 | my $vendor = $1; |
Florian Vaussard | cc93319 | 2014-04-03 14:49:27 -0700 | [diff] [blame] | 2731 | `grep -Eq "^$vendor\\b" $vp_file`; |
Rob Herring | bff5da4 | 2014-01-23 15:54:51 -0800 | [diff] [blame] | 2732 | if ( $? >> 8 ) { |
| 2733 | WARN("UNDOCUMENTED_DT_STRING", |
Florian Vaussard | cc93319 | 2014-04-03 14:49:27 -0700 | [diff] [blame] | 2734 | "DT compatible string vendor \"$vendor\" appears un-documented -- check $vp_file\n" . $herecurr); |
Rob Herring | bff5da4 | 2014-01-23 15:54:51 -0800 | [diff] [blame] | 2735 | } |
| 2736 | } |
| 2737 | } |
| 2738 | |
Andy Whitcroft | 5368df20 | 2008-10-15 22:02:27 -0700 | [diff] [blame] | 2739 | # check we are in a valid source file if not then ignore this hunk |
Geert Uytterhoeven | de4c924 | 2014-10-13 15:51:46 -0700 | [diff] [blame] | 2740 | next if ($realfile !~ /\.(h|c|s|S|pl|sh|dtsi|dts)$/); |
Andy Whitcroft | 5368df20 | 2008-10-15 22:02:27 -0700 | [diff] [blame] | 2741 | |
Joe Perches | 47e0c88 | 2015-06-25 15:02:57 -0700 | [diff] [blame] | 2742 | # line length limit (with some exclusions) |
| 2743 | # |
| 2744 | # There are a few types of lines that may extend beyond $max_line_length: |
| 2745 | # logging functions like pr_info that end in a string |
| 2746 | # lines with a single string |
| 2747 | # #defines that are a single string |
| 2748 | # |
| 2749 | # There are 3 different line length message types: |
| 2750 | # LONG_LINE_COMMENT a comment starts before but extends beyond $max_linelength |
| 2751 | # LONG_LINE_STRING a string starts before but extends beyond $max_line_length |
| 2752 | # LONG_LINE all other lines longer than $max_line_length |
| 2753 | # |
| 2754 | # if LONG_LINE is ignored, the other 2 types are also ignored |
| 2755 | # |
| 2756 | |
Joe Perches | b4749e9 | 2015-07-17 16:24:01 -0700 | [diff] [blame] | 2757 | if ($line =~ /^\+/ && $length > $max_line_length) { |
Joe Perches | 47e0c88 | 2015-06-25 15:02:57 -0700 | [diff] [blame] | 2758 | my $msg_type = "LONG_LINE"; |
| 2759 | |
| 2760 | # Check the allowed long line types first |
| 2761 | |
| 2762 | # logging functions that end in a string that starts |
| 2763 | # before $max_line_length |
| 2764 | if ($line =~ /^\+\s*$logFunctions\s*\(\s*(?:(?:KERN_\S+\s*|[^"]*))?($String\s*(?:|,|\)\s*;)\s*)$/ && |
| 2765 | length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { |
| 2766 | $msg_type = ""; |
| 2767 | |
| 2768 | # lines with only strings (w/ possible termination) |
| 2769 | # #defines with only strings |
| 2770 | } elsif ($line =~ /^\+\s*$String\s*(?:\s*|,|\)\s*;)\s*$/ || |
| 2771 | $line =~ /^\+\s*#\s*define\s+\w+\s+$String$/) { |
| 2772 | $msg_type = ""; |
| 2773 | |
Joe Perches | d560a5f | 2016-08-02 14:04:31 -0700 | [diff] [blame] | 2774 | # EFI_GUID is another special case |
| 2775 | } elsif ($line =~ /^\+.*\bEFI_GUID\s*\(/) { |
| 2776 | $msg_type = ""; |
| 2777 | |
Joe Perches | 47e0c88 | 2015-06-25 15:02:57 -0700 | [diff] [blame] | 2778 | # Otherwise set the alternate message types |
| 2779 | |
| 2780 | # a comment starts before $max_line_length |
| 2781 | } elsif ($line =~ /($;[\s$;]*)$/ && |
| 2782 | length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { |
| 2783 | $msg_type = "LONG_LINE_COMMENT" |
| 2784 | |
| 2785 | # a quoted string starts before $max_line_length |
| 2786 | } elsif ($sline =~ /\s*($String(?:\s*(?:\\|,\s*|\)\s*;\s*))?)$/ && |
| 2787 | length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) { |
| 2788 | $msg_type = "LONG_LINE_STRING" |
| 2789 | } |
| 2790 | |
| 2791 | if ($msg_type ne "" && |
| 2792 | (show_type("LONG_LINE") || show_type($msg_type))) { |
| 2793 | WARN($msg_type, |
| 2794 | "line over $max_line_length characters\n" . $herecurr); |
| 2795 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2796 | } |
| 2797 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 2798 | # check for adding lines without a newline. |
| 2799 | if ($line =~ /^\+/ && defined $lines[$linenr] && $lines[$linenr] =~ /^\\ No newline at end of file/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2800 | WARN("MISSING_EOF_NEWLINE", |
| 2801 | "adding a line without newline at end of file\n" . $herecurr); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 2802 | } |
| 2803 | |
Mike Frysinger | 42e41c5 | 2009-09-21 17:04:40 -0700 | [diff] [blame] | 2804 | # Blackfin: use hi/lo macros |
| 2805 | if ($realfile =~ m@arch/blackfin/.*\.S$@) { |
| 2806 | if ($line =~ /\.[lL][[:space:]]*=.*&[[:space:]]*0x[fF][fF][fF][fF]/) { |
| 2807 | my $herevet = "$here\n" . cat_vet($line) . "\n"; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2808 | ERROR("LO_MACRO", |
| 2809 | "use the LO() macro, not (... & 0xFFFF)\n" . $herevet); |
Mike Frysinger | 42e41c5 | 2009-09-21 17:04:40 -0700 | [diff] [blame] | 2810 | } |
| 2811 | if ($line =~ /\.[hH][[:space:]]*=.*>>[[:space:]]*16/) { |
| 2812 | my $herevet = "$here\n" . cat_vet($line) . "\n"; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 2813 | ERROR("HI_MACRO", |
| 2814 | "use the HI() macro, not (... >> 16)\n" . $herevet); |
Mike Frysinger | 42e41c5 | 2009-09-21 17:04:40 -0700 | [diff] [blame] | 2815 | } |
| 2816 | } |
| 2817 | |
Andy Whitcroft | b9ea10d | 2008-10-15 22:02:24 -0700 | [diff] [blame] | 2818 | # check we are in a valid source file C or perl if not then ignore this hunk |
Geert Uytterhoeven | de4c924 | 2014-10-13 15:51:46 -0700 | [diff] [blame] | 2819 | next if ($realfile !~ /\.(h|c|pl|dtsi|dts)$/); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2820 | |
| 2821 | # at the beginning of a line any tabs must come first and anything |
| 2822 | # more than 8 must use tabs. |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 2823 | if ($rawline =~ /^\+\s* \t\s*\S/ || |
| 2824 | $rawline =~ /^\+\s* \s*/) { |
| 2825 | my $herevet = "$here\n" . cat_vet($rawline) . "\n"; |
Andy Whitcroft | d2c0a23 | 2010-10-26 14:23:12 -0700 | [diff] [blame] | 2826 | $rpt_cleaners = 1; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2827 | if (ERROR("CODE_INDENT", |
| 2828 | "code indent should use tabs where possible\n" . $herevet) && |
| 2829 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2830 | $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2831 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 2832 | } |
| 2833 | |
Alberto Panizzo | 08e4436 | 2010-03-05 13:43:54 -0800 | [diff] [blame] | 2834 | # check for space before tabs. |
| 2835 | if ($rawline =~ /^\+/ && $rawline =~ / \t/) { |
| 2836 | my $herevet = "$here\n" . cat_vet($rawline) . "\n"; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2837 | if (WARN("SPACE_BEFORE_TAB", |
| 2838 | "please, no space before tabs\n" . $herevet) && |
| 2839 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2840 | while ($fixed[$fixlinenr] =~ |
Joe Perches | d2207cc | 2014-10-13 15:51:53 -0700 | [diff] [blame] | 2841 | s/(^\+.*) {8,8}\t/$1\t\t/) {} |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2842 | while ($fixed[$fixlinenr] =~ |
Joe Perches | c76f4cb | 2014-01-23 15:54:46 -0800 | [diff] [blame] | 2843 | s/(^\+.*) +\t/$1\t/) {} |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2844 | } |
Alberto Panizzo | 08e4436 | 2010-03-05 13:43:54 -0800 | [diff] [blame] | 2845 | } |
| 2846 | |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 2847 | # check for && or || at the start of a line |
| 2848 | if ($rawline =~ /^\+\s*(&&|\|\|)/) { |
| 2849 | CHK("LOGICAL_CONTINUATIONS", |
| 2850 | "Logical continuations should be on the previous line\n" . $hereprev); |
| 2851 | } |
| 2852 | |
Joe Perches | a91e899 | 2016-05-20 17:04:05 -0700 | [diff] [blame] | 2853 | # check indentation starts on a tab stop |
| 2854 | if ($^V && $^V ge 5.10.0 && |
| 2855 | $sline =~ /^\+\t+( +)(?:$c90_Keywords\b|\{\s*$|\}\s*(?:else\b|while\b|\s*$))/) { |
| 2856 | my $indent = length($1); |
| 2857 | if ($indent % 8) { |
| 2858 | if (WARN("TABSTOP", |
| 2859 | "Statements should start on a tabstop\n" . $herecurr) && |
| 2860 | $fix) { |
| 2861 | $fixed[$fixlinenr] =~ s@(^\+\t+) +@$1 . "\t" x ($indent/8)@e; |
| 2862 | } |
| 2863 | } |
| 2864 | } |
| 2865 | |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 2866 | # check multi-line statement indentation matches previous line |
| 2867 | if ($^V && $^V ge 5.10.0 && |
Joe Perches | 91cb519 | 2014-04-03 14:49:32 -0700 | [diff] [blame] | 2868 | $prevline =~ /^\+([ \t]*)((?:$c90_Keywords(?:\s+if)\s*)|(?:$Declare\s*)?(?:$Ident|\(\s*\*\s*$Ident\s*\))\s*|$Ident\s*=\s*$Ident\s*)\(.*(\&\&|\|\||,)\s*$/) { |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 2869 | $prevline =~ /^\+(\t*)(.*)$/; |
| 2870 | my $oldindent = $1; |
| 2871 | my $rest = $2; |
| 2872 | |
| 2873 | my $pos = pos_last_openparen($rest); |
| 2874 | if ($pos >= 0) { |
Joe Perches | b34a26f | 2012-07-30 14:41:16 -0700 | [diff] [blame] | 2875 | $line =~ /^(\+| )([ \t]*)/; |
| 2876 | my $newindent = $2; |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 2877 | |
| 2878 | my $goodtabindent = $oldindent . |
| 2879 | "\t" x ($pos / 8) . |
| 2880 | " " x ($pos % 8); |
| 2881 | my $goodspaceindent = $oldindent . " " x $pos; |
| 2882 | |
| 2883 | if ($newindent ne $goodtabindent && |
| 2884 | $newindent ne $goodspaceindent) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2885 | |
| 2886 | if (CHK("PARENTHESIS_ALIGNMENT", |
| 2887 | "Alignment should match open parenthesis\n" . $hereprev) && |
| 2888 | $fix && $line =~ /^\+/) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2889 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2890 | s/^\+[ \t]*/\+$goodtabindent/; |
| 2891 | } |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 2892 | } |
| 2893 | } |
| 2894 | } |
| 2895 | |
Joe Perches | 6ab3a97 | 2015-04-16 12:44:05 -0700 | [diff] [blame] | 2896 | # check for space after cast like "(int) foo" or "(struct foo) bar" |
| 2897 | # avoid checking a few false positives: |
| 2898 | # "sizeof(<type>)" or "__alignof__(<type>)" |
| 2899 | # function pointer declarations like "(*foo)(int) = bar;" |
| 2900 | # structure definitions like "(struct foo) { 0 };" |
| 2901 | # multiline macros that define functions |
| 2902 | # known attributes or the __attribute__ keyword |
| 2903 | if ($line =~ /^\+(.*)\(\s*$Type\s*\)([ \t]++)((?![={]|\\$|$Attribute|__attribute__))/ && |
| 2904 | (!defined($1) || $1 !~ /\b(?:sizeof|__alignof__)\s*$/)) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2905 | if (CHK("SPACING", |
Joe Perches | f27c95d | 2014-08-06 16:10:52 -0700 | [diff] [blame] | 2906 | "No space is necessary after a cast\n" . $herecurr) && |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2907 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 2908 | $fixed[$fixlinenr] =~ |
Joe Perches | f27c95d | 2014-08-06 16:10:52 -0700 | [diff] [blame] | 2909 | s/(\(\s*$Type\s*\))[ \t]+/$1/; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 2910 | } |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 2911 | } |
| 2912 | |
Joe Perches | 86406b1 | 2015-09-09 15:37:41 -0700 | [diff] [blame] | 2913 | # Block comment styles |
| 2914 | # Networking with an initial /* |
Joe Perches | 0588060 | 2012-10-04 17:13:35 -0700 | [diff] [blame] | 2915 | if ($realfile =~ m@^(drivers/net/|net/)@ && |
Joe Perches | fdb4bcd | 2013-07-03 15:05:23 -0700 | [diff] [blame] | 2916 | $prevrawline =~ /^\+[ \t]*\/\*[ \t]*$/ && |
Joe Perches | 85ad978 | 2014-04-03 14:49:20 -0700 | [diff] [blame] | 2917 | $rawline =~ /^\+[ \t]*\*/ && |
| 2918 | $realline > 2) { |
Joe Perches | 0588060 | 2012-10-04 17:13:35 -0700 | [diff] [blame] | 2919 | WARN("NETWORKING_BLOCK_COMMENT_STYLE", |
| 2920 | "networking block comments don't use an empty /* line, use /* Comment...\n" . $hereprev); |
| 2921 | } |
| 2922 | |
Joe Perches | 86406b1 | 2015-09-09 15:37:41 -0700 | [diff] [blame] | 2923 | # Block comments use * on subsequent lines |
| 2924 | if ($prevline =~ /$;[ \t]*$/ && #ends in comment |
| 2925 | $prevrawline =~ /^\+.*?\/\*/ && #starting /* |
Joe Perches | a605e32 | 2013-07-03 15:05:24 -0700 | [diff] [blame] | 2926 | $prevrawline !~ /\*\/[ \t]*$/ && #no trailing */ |
Joe Perches | 61135e9 | 2013-09-11 14:23:59 -0700 | [diff] [blame] | 2927 | $rawline =~ /^\+/ && #line is new |
Joe Perches | a605e32 | 2013-07-03 15:05:24 -0700 | [diff] [blame] | 2928 | $rawline !~ /^\+[ \t]*\*/) { #no leading * |
Joe Perches | 86406b1 | 2015-09-09 15:37:41 -0700 | [diff] [blame] | 2929 | WARN("BLOCK_COMMENT_STYLE", |
| 2930 | "Block comments use * on subsequent lines\n" . $hereprev); |
Joe Perches | a605e32 | 2013-07-03 15:05:24 -0700 | [diff] [blame] | 2931 | } |
| 2932 | |
Joe Perches | 86406b1 | 2015-09-09 15:37:41 -0700 | [diff] [blame] | 2933 | # Block comments use */ on trailing lines |
| 2934 | if ($rawline !~ m@^\+[ \t]*\*/[ \t]*$@ && #trailing */ |
Joe Perches | c24f9f1 | 2012-11-08 15:53:29 -0800 | [diff] [blame] | 2935 | $rawline !~ m@^\+.*/\*.*\*/[ \t]*$@ && #inline /*...*/ |
| 2936 | $rawline !~ m@^\+.*\*{2,}/[ \t]*$@ && #trailing **/ |
| 2937 | $rawline =~ m@^\+[ \t]*.+\*\/[ \t]*$@) { #non blank */ |
Joe Perches | 86406b1 | 2015-09-09 15:37:41 -0700 | [diff] [blame] | 2938 | WARN("BLOCK_COMMENT_STYLE", |
| 2939 | "Block comments use a trailing */ on a separate line\n" . $herecurr); |
Joe Perches | 0588060 | 2012-10-04 17:13:35 -0700 | [diff] [blame] | 2940 | } |
| 2941 | |
Joe Perches | 7f61919 | 2014-08-06 16:10:39 -0700 | [diff] [blame] | 2942 | # check for missing blank lines after struct/union declarations |
| 2943 | # with exceptions for various attributes and macros |
| 2944 | if ($prevline =~ /^[\+ ]};?\s*$/ && |
| 2945 | $line =~ /^\+/ && |
| 2946 | !($line =~ /^\+\s*$/ || |
| 2947 | $line =~ /^\+\s*EXPORT_SYMBOL/ || |
| 2948 | $line =~ /^\+\s*MODULE_/i || |
| 2949 | $line =~ /^\+\s*\#\s*(?:end|elif|else)/ || |
| 2950 | $line =~ /^\+[a-z_]*init/ || |
| 2951 | $line =~ /^\+\s*(?:static\s+)?[A-Z_]*ATTR/ || |
| 2952 | $line =~ /^\+\s*DECLARE/ || |
| 2953 | $line =~ /^\+\s*__setup/)) { |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 2954 | if (CHK("LINE_SPACING", |
| 2955 | "Please use a blank line after function/struct/union/enum declarations\n" . $hereprev) && |
| 2956 | $fix) { |
Joe Perches | f2d7e4d | 2014-08-06 16:11:07 -0700 | [diff] [blame] | 2957 | fix_insert_line($fixlinenr, "\+"); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 2958 | } |
Joe Perches | 7f61919 | 2014-08-06 16:10:39 -0700 | [diff] [blame] | 2959 | } |
| 2960 | |
Joe Perches | 365dd4e | 2014-08-06 16:10:42 -0700 | [diff] [blame] | 2961 | # check for multiple consecutive blank lines |
| 2962 | if ($prevline =~ /^[\+ ]\s*$/ && |
| 2963 | $line =~ /^\+\s*$/ && |
| 2964 | $last_blank_line != ($linenr - 1)) { |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 2965 | if (CHK("LINE_SPACING", |
| 2966 | "Please don't use multiple blank lines\n" . $hereprev) && |
| 2967 | $fix) { |
Joe Perches | f2d7e4d | 2014-08-06 16:11:07 -0700 | [diff] [blame] | 2968 | fix_delete_line($fixlinenr, $rawline); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 2969 | } |
| 2970 | |
Joe Perches | 365dd4e | 2014-08-06 16:10:42 -0700 | [diff] [blame] | 2971 | $last_blank_line = $linenr; |
| 2972 | } |
| 2973 | |
Joe Perches | 3b617e3 | 2014-04-03 14:49:28 -0700 | [diff] [blame] | 2974 | # check for missing blank lines after declarations |
Joe Perches | 3f7bac0 | 2014-06-04 16:12:04 -0700 | [diff] [blame] | 2975 | if ($sline =~ /^\+\s+\S/ && #Not at char 1 |
| 2976 | # actual declarations |
| 2977 | ($prevline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || |
Joe Perches | 5a4e1fd | 2014-08-06 16:10:33 -0700 | [diff] [blame] | 2978 | # function pointer declarations |
| 2979 | $prevline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || |
Joe Perches | 3f7bac0 | 2014-06-04 16:12:04 -0700 | [diff] [blame] | 2980 | # foo bar; where foo is some local typedef or #define |
| 2981 | $prevline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || |
| 2982 | # known declaration macros |
| 2983 | $prevline =~ /^\+\s+$declaration_macros/) && |
| 2984 | # for "else if" which can look like "$Ident $Ident" |
| 2985 | !($prevline =~ /^\+\s+$c90_Keywords\b/ || |
| 2986 | # other possible extensions of declaration lines |
| 2987 | $prevline =~ /(?:$Compare|$Assignment|$Operators)\s*$/ || |
| 2988 | # not starting a section or a macro "\" extended line |
| 2989 | $prevline =~ /(?:\{\s*|\\)$/) && |
| 2990 | # looks like a declaration |
| 2991 | !($sline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ || |
Joe Perches | 5a4e1fd | 2014-08-06 16:10:33 -0700 | [diff] [blame] | 2992 | # function pointer declarations |
| 2993 | $sline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ || |
Joe Perches | 3f7bac0 | 2014-06-04 16:12:04 -0700 | [diff] [blame] | 2994 | # foo bar; where foo is some local typedef or #define |
| 2995 | $sline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ || |
| 2996 | # known declaration macros |
| 2997 | $sline =~ /^\+\s+$declaration_macros/ || |
| 2998 | # start of struct or union or enum |
Joe Perches | 3b617e3 | 2014-04-03 14:49:28 -0700 | [diff] [blame] | 2999 | $sline =~ /^\+\s+(?:union|struct|enum|typedef)\b/ || |
Joe Perches | 3f7bac0 | 2014-06-04 16:12:04 -0700 | [diff] [blame] | 3000 | # start or end of block or continuation of declaration |
| 3001 | $sline =~ /^\+\s+(?:$|[\{\}\.\#\"\?\:\(\[])/ || |
| 3002 | # bitfield continuation |
| 3003 | $sline =~ /^\+\s+$Ident\s*:\s*\d+\s*[,;]/ || |
| 3004 | # other possible extensions of declaration lines |
| 3005 | $sline =~ /^\+\s+\(?\s*(?:$Compare|$Assignment|$Operators)/) && |
| 3006 | # indentation of previous and current line are the same |
| 3007 | (($prevline =~ /\+(\s+)\S/) && $sline =~ /^\+$1\S/)) { |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 3008 | if (WARN("LINE_SPACING", |
| 3009 | "Missing a blank line after declarations\n" . $hereprev) && |
| 3010 | $fix) { |
Joe Perches | f2d7e4d | 2014-08-06 16:11:07 -0700 | [diff] [blame] | 3011 | fix_insert_line($fixlinenr, "\+"); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 3012 | } |
Joe Perches | 3b617e3 | 2014-04-03 14:49:28 -0700 | [diff] [blame] | 3013 | } |
| 3014 | |
Raffaele Recalcati | 5f7ddae | 2010-08-09 17:20:59 -0700 | [diff] [blame] | 3015 | # check for spaces at the beginning of a line. |
Andy Whitcroft | 6b4c5be | 2010-10-26 14:23:11 -0700 | [diff] [blame] | 3016 | # Exceptions: |
| 3017 | # 1) within comments |
| 3018 | # 2) indented preprocessor commands |
| 3019 | # 3) hanging labels |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3020 | if ($rawline =~ /^\+ / && $line !~ /^\+ *(?:$;|#|$Ident:)/) { |
Raffaele Recalcati | 5f7ddae | 2010-08-09 17:20:59 -0700 | [diff] [blame] | 3021 | my $herevet = "$here\n" . cat_vet($rawline) . "\n"; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3022 | if (WARN("LEADING_SPACE", |
| 3023 | "please, no spaces at the start of a line\n" . $herevet) && |
| 3024 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3025 | $fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3026 | } |
Raffaele Recalcati | 5f7ddae | 2010-08-09 17:20:59 -0700 | [diff] [blame] | 3027 | } |
| 3028 | |
Andy Whitcroft | b9ea10d | 2008-10-15 22:02:24 -0700 | [diff] [blame] | 3029 | # check we are in a valid C source file if not then ignore this hunk |
| 3030 | next if ($realfile !~ /\.(h|c)$/); |
| 3031 | |
Joe Perches | 032a4c0 | 2014-08-06 16:10:29 -0700 | [diff] [blame] | 3032 | # check indentation of any line with a bare else |
Joe Perches | 840080a | 2014-10-13 15:51:59 -0700 | [diff] [blame] | 3033 | # (but not if it is a multiple line "if (foo) return bar; else return baz;") |
Joe Perches | 032a4c0 | 2014-08-06 16:10:29 -0700 | [diff] [blame] | 3034 | # if the previous line is a break or return and is indented 1 tab more... |
| 3035 | if ($sline =~ /^\+([\t]+)(?:}[ \t]*)?else(?:[ \t]*{)?\s*$/) { |
| 3036 | my $tabs = length($1) + 1; |
Joe Perches | 840080a | 2014-10-13 15:51:59 -0700 | [diff] [blame] | 3037 | if ($prevline =~ /^\+\t{$tabs,$tabs}break\b/ || |
| 3038 | ($prevline =~ /^\+\t{$tabs,$tabs}return\b/ && |
| 3039 | defined $lines[$linenr] && |
| 3040 | $lines[$linenr] !~ /^[ \+]\t{$tabs,$tabs}return/)) { |
Joe Perches | 032a4c0 | 2014-08-06 16:10:29 -0700 | [diff] [blame] | 3041 | WARN("UNNECESSARY_ELSE", |
| 3042 | "else is not generally useful after a break or return\n" . $hereprev); |
| 3043 | } |
| 3044 | } |
| 3045 | |
Joe Perches | c00df19 | 2014-08-06 16:11:01 -0700 | [diff] [blame] | 3046 | # check indentation of a line with a break; |
| 3047 | # if the previous line is a goto or return and is indented the same # of tabs |
| 3048 | if ($sline =~ /^\+([\t]+)break\s*;\s*$/) { |
| 3049 | my $tabs = $1; |
| 3050 | if ($prevline =~ /^\+$tabs(?:goto|return)\b/) { |
| 3051 | WARN("UNNECESSARY_BREAK", |
| 3052 | "break is not useful after a goto or return\n" . $hereprev); |
| 3053 | } |
| 3054 | } |
| 3055 | |
Kees Cook | 1ba8dfd | 2012-12-17 16:01:48 -0800 | [diff] [blame] | 3056 | # discourage the addition of CONFIG_EXPERIMENTAL in #if(def). |
| 3057 | if ($line =~ /^\+\s*\#\s*if.*\bCONFIG_EXPERIMENTAL\b/) { |
| 3058 | WARN("CONFIG_EXPERIMENTAL", |
| 3059 | "Use of CONFIG_EXPERIMENTAL is deprecated. For alternatives, see https://lkml.org/lkml/2012/10/23/580\n"); |
| 3060 | } |
| 3061 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3062 | # check for RCS/CVS revision markers |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3063 | if ($rawline =~ /^\+.*\$(Revision|Log|Id)(?:\$|)/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3064 | WARN("CVS_KEYWORD", |
| 3065 | "CVS style keyword markers, these will _not_ be updated\n". $herecurr); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3066 | } |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 3067 | |
Mike Frysinger | 42e41c5 | 2009-09-21 17:04:40 -0700 | [diff] [blame] | 3068 | # Blackfin: don't use __builtin_bfin_[cs]sync |
| 3069 | if ($line =~ /__builtin_bfin_csync/) { |
| 3070 | my $herevet = "$here\n" . cat_vet($line) . "\n"; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3071 | ERROR("CSYNC", |
| 3072 | "use the CSYNC() macro in asm/blackfin.h\n" . $herevet); |
Mike Frysinger | 42e41c5 | 2009-09-21 17:04:40 -0700 | [diff] [blame] | 3073 | } |
| 3074 | if ($line =~ /__builtin_bfin_ssync/) { |
| 3075 | my $herevet = "$here\n" . cat_vet($line) . "\n"; |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3076 | ERROR("SSYNC", |
| 3077 | "use the SSYNC() macro in asm/blackfin.h\n" . $herevet); |
Mike Frysinger | 42e41c5 | 2009-09-21 17:04:40 -0700 | [diff] [blame] | 3078 | } |
| 3079 | |
Joe Perches | 56e77d7 | 2013-02-21 16:44:14 -0800 | [diff] [blame] | 3080 | # check for old HOTPLUG __dev<foo> section markings |
| 3081 | if ($line =~ /\b(__dev(init|exit)(data|const|))\b/) { |
| 3082 | WARN("HOTPLUG_SECTION", |
| 3083 | "Using $1 is unnecessary\n" . $herecurr); |
| 3084 | } |
| 3085 | |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3086 | # Check for potential 'bare' types |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3087 | my ($stat, $cond, $line_nr_next, $remain_next, $off_next, |
| 3088 | $realline_next); |
Andy Whitcroft | 3e469cd | 2012-01-10 15:10:01 -0800 | [diff] [blame] | 3089 | #print "LINE<$line>\n"; |
| 3090 | if ($linenr >= $suppress_statement && |
Joe Perches | 1b5539b | 2013-09-11 14:24:03 -0700 | [diff] [blame] | 3091 | $realcnt && $sline =~ /.\s*\S/) { |
Andy Whitcroft | 170d3a2 | 2008-10-15 22:02:30 -0700 | [diff] [blame] | 3092 | ($stat, $cond, $line_nr_next, $remain_next, $off_next) = |
Andy Whitcroft | f5fe35d | 2008-07-23 21:29:03 -0700 | [diff] [blame] | 3093 | ctx_statement_block($linenr, $realcnt, 0); |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 3094 | $stat =~ s/\n./\n /g; |
| 3095 | $cond =~ s/\n./\n /g; |
| 3096 | |
Andy Whitcroft | 3e469cd | 2012-01-10 15:10:01 -0800 | [diff] [blame] | 3097 | #print "linenr<$linenr> <$stat>\n"; |
| 3098 | # If this statement has no statement boundaries within |
| 3099 | # it there is no point in retrying a statement scan |
| 3100 | # until we hit end of it. |
| 3101 | my $frag = $stat; $frag =~ s/;+\s*$//; |
| 3102 | if ($frag !~ /(?:{|;)/) { |
| 3103 | #print "skip<$line_nr_next>\n"; |
| 3104 | $suppress_statement = $line_nr_next; |
| 3105 | } |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 3106 | |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3107 | # Find the real next line. |
| 3108 | $realline_next = $line_nr_next; |
| 3109 | if (defined $realline_next && |
| 3110 | (!defined $lines[$realline_next - 1] || |
| 3111 | substr($lines[$realline_next - 1], $off_next) =~ /^\s*$/)) { |
| 3112 | $realline_next++; |
| 3113 | } |
| 3114 | |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 3115 | my $s = $stat; |
| 3116 | $s =~ s/{.*$//s; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3117 | |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3118 | # Ignore goto labels. |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 3119 | if ($s =~ /$Ident:\*$/s) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3120 | |
| 3121 | # Ignore functions being called |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 3122 | } elsif ($s =~ /^.\s*$Ident\s*\(/s) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3123 | |
Andy Whitcroft | 463f286 | 2009-09-21 17:04:34 -0700 | [diff] [blame] | 3124 | } elsif ($s =~ /^.\s*else\b/s) { |
| 3125 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3126 | # declarations always start with types |
Andy Whitcroft | d250658 | 2008-07-23 21:29:09 -0700 | [diff] [blame] | 3127 | } elsif ($prev_values eq 'E' && $s =~ /^.\s*(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?((?:\s*$Ident)+?)\b(?:\s+$Sparse)?\s*\**\s*(?:$Ident|\(\*[^\)]*\))(?:\s*$Modifier)?\s*(?:;|=|,|\()/s) { |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3128 | my $type = $1; |
| 3129 | $type =~ s/\s+/ /g; |
| 3130 | possible($type, "A:" . $s); |
| 3131 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3132 | # definitions in global scope can only start with types |
Andy Whitcroft | a6a84062 | 2008-10-15 22:02:30 -0700 | [diff] [blame] | 3133 | } elsif ($s =~ /^.(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?($Ident)\b\s*(?!:)/s) { |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3134 | possible($1, "B:" . $s); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3135 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3136 | |
| 3137 | # any (foo ... *) is a pointer cast, and foo is a type |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3138 | while ($s =~ /\(($Ident)(?:\s+$Sparse)*[\s\*]+\s*\)/sg) { |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3139 | possible($1, "C:" . $s); |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3140 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3141 | |
| 3142 | # Check for any sort of function declaration. |
| 3143 | # int foo(something bar, other baz); |
| 3144 | # void (*store_gdt)(x86_descr_ptr *); |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 3145 | if ($prev_values eq 'E' && $s =~ /^(.(?:typedef\s*)?(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*(?:\b$Ident|\(\*\s*$Ident\))\s*)\(/s) { |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3146 | my ($name_len) = length($1); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3147 | |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3148 | my $ctx = $s; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3149 | substr($ctx, 0, $name_len + 1, ''); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3150 | $ctx =~ s/\)[^\)]*$//; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3151 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3152 | for my $arg (split(/\s*,\s*/, $ctx)) { |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3153 | if ($arg =~ /^(?:const\s+)?($Ident)(?:\s+$Sparse)*\s*\**\s*(:?\b$Ident)?$/s || $arg =~ /^($Ident)$/s) { |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3154 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3155 | possible($1, "D:" . $s); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3156 | } |
| 3157 | } |
| 3158 | } |
| 3159 | |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3160 | } |
| 3161 | |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3162 | # |
| 3163 | # Checks which may be anchored in the context. |
| 3164 | # |
| 3165 | |
| 3166 | # Check for switch () and associated case and default |
| 3167 | # statements should be at the same indent. |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3168 | if ($line=~/\bswitch\s*\(.*\)/) { |
| 3169 | my $err = ''; |
| 3170 | my $sep = ''; |
| 3171 | my @ctx = ctx_block_outer($linenr, $realcnt); |
| 3172 | shift(@ctx); |
| 3173 | for my $ctx (@ctx) { |
| 3174 | my ($clen, $cindent) = line_stats($ctx); |
| 3175 | if ($ctx =~ /^\+\s*(case\s+|default:)/ && |
| 3176 | $indent != $cindent) { |
| 3177 | $err .= "$sep$ctx\n"; |
| 3178 | $sep = ''; |
| 3179 | } else { |
| 3180 | $sep = "[...]\n"; |
| 3181 | } |
| 3182 | } |
| 3183 | if ($err ne '') { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3184 | ERROR("SWITCH_CASE_INDENT_LEVEL", |
| 3185 | "switch and case should be at the same indent\n$hereline$err"); |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 3186 | } |
| 3187 | } |
| 3188 | |
| 3189 | # if/while/etc brace do not go on next line, unless defining a do while loop, |
| 3190 | # or if that brace on the next line is for something else |
Joe Perches | 0fe3dc2 | 2014-08-06 16:11:16 -0700 | [diff] [blame] | 3191 | if ($line =~ /(.*)\b((?:if|while|for|switch|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b|else\b)/ && $line !~ /^.\s*\#/) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3192 | my $pre_ctx = "$1$2"; |
| 3193 | |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3194 | my ($level, @ctx) = ctx_statement_level($linenr, $realcnt, 0); |
Joe Perches | 8eef05d | 2012-02-03 15:20:39 -0800 | [diff] [blame] | 3195 | |
| 3196 | if ($line =~ /^\+\t{6,}/) { |
| 3197 | WARN("DEEP_INDENTATION", |
| 3198 | "Too many leading tabs - consider code refactoring\n" . $herecurr); |
| 3199 | } |
| 3200 | |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 3201 | my $ctx_cnt = $realcnt - $#ctx - 1; |
| 3202 | my $ctx = join("\n", @ctx); |
| 3203 | |
Andy Whitcroft | 548596d | 2008-07-23 21:29:01 -0700 | [diff] [blame] | 3204 | my $ctx_ln = $linenr; |
| 3205 | my $ctx_skip = $realcnt; |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 3206 | |
Andy Whitcroft | 548596d | 2008-07-23 21:29:01 -0700 | [diff] [blame] | 3207 | while ($ctx_skip > $ctx_cnt || ($ctx_skip == $ctx_cnt && |
| 3208 | defined $lines[$ctx_ln - 1] && |
| 3209 | $lines[$ctx_ln - 1] =~ /^-/)) { |
| 3210 | ##print "SKIP<$ctx_skip> CNT<$ctx_cnt>\n"; |
| 3211 | $ctx_skip-- if (!defined $lines[$ctx_ln - 1] || $lines[$ctx_ln - 1] !~ /^-/); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3212 | $ctx_ln++; |
| 3213 | } |
Andy Whitcroft | 548596d | 2008-07-23 21:29:01 -0700 | [diff] [blame] | 3214 | |
Andy Whitcroft | 5321016 | 2008-07-23 21:29:03 -0700 | [diff] [blame] | 3215 | #print "realcnt<$realcnt> ctx_cnt<$ctx_cnt>\n"; |
| 3216 | #print "pre<$pre_ctx>\nline<$line>\nctx<$ctx>\nnext<$lines[$ctx_ln - 1]>\n"; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3217 | |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 3218 | if ($ctx !~ /{\s*/ && defined($lines[$ctx_ln - 1]) && $lines[$ctx_ln - 1] =~ /^\+\s*{/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3219 | ERROR("OPEN_BRACE", |
| 3220 | "that open brace { should be on the previous line\n" . |
Andy Whitcroft | 01464f3 | 2010-10-26 14:23:19 -0700 | [diff] [blame] | 3221 | "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3222 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3223 | if ($level == 0 && $pre_ctx !~ /}\s*while\s*\($/ && |
| 3224 | $ctx =~ /\)\s*\;\s*$/ && |
| 3225 | defined $lines[$ctx_ln - 1]) |
| 3226 | { |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3227 | my ($nlength, $nindent) = line_stats($lines[$ctx_ln - 1]); |
| 3228 | if ($nindent > $indent) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3229 | WARN("TRAILING_SEMICOLON", |
| 3230 | "trailing semicolon indicates no statements, indent implies otherwise\n" . |
Andy Whitcroft | 01464f3 | 2010-10-26 14:23:19 -0700 | [diff] [blame] | 3231 | "$here\n$ctx\n$rawlines[$ctx_ln - 1]\n"); |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3232 | } |
| 3233 | } |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3234 | } |
| 3235 | |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3236 | # Check relative indent for conditionals and blocks. |
Joe Perches | 0fe3dc2 | 2014-08-06 16:11:16 -0700 | [diff] [blame] | 3237 | if ($line =~ /\b(?:(?:if|while|for|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b)/ && $line !~ /^.\s*#/ && $line !~ /\}\s*while\s*/) { |
Andy Whitcroft | 3e469cd | 2012-01-10 15:10:01 -0800 | [diff] [blame] | 3238 | ($stat, $cond, $line_nr_next, $remain_next, $off_next) = |
| 3239 | ctx_statement_block($linenr, $realcnt, 0) |
| 3240 | if (!defined $stat); |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3241 | my ($s, $c) = ($stat, $cond); |
| 3242 | |
| 3243 | substr($s, 0, length($c), ''); |
| 3244 | |
Joe Perches | 9f5af48 | 2015-09-09 15:37:30 -0700 | [diff] [blame] | 3245 | # remove inline comments |
| 3246 | $s =~ s/$;/ /g; |
| 3247 | $c =~ s/$;/ /g; |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3248 | |
| 3249 | # Find out how long the conditional actually is. |
Andy Whitcroft | 6f779c1 | 2008-10-15 22:02:27 -0700 | [diff] [blame] | 3250 | my @newlines = ($c =~ /\n/gs); |
| 3251 | my $cond_lines = 1 + $#newlines; |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3252 | |
Joe Perches | 9f5af48 | 2015-09-09 15:37:30 -0700 | [diff] [blame] | 3253 | # Make sure we remove the line prefixes as we have |
| 3254 | # none on the first line, and are going to readd them |
| 3255 | # where necessary. |
| 3256 | $s =~ s/\n./\n/gs; |
| 3257 | while ($s =~ /\n\s+\\\n/) { |
| 3258 | $cond_lines += $s =~ s/\n\s+\\\n/\n/g; |
| 3259 | } |
| 3260 | |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3261 | # We want to check the first line inside the block |
| 3262 | # starting at the end of the conditional, so remove: |
| 3263 | # 1) any blank line termination |
| 3264 | # 2) any opening brace { on end of the line |
| 3265 | # 3) any do (...) { |
| 3266 | my $continuation = 0; |
| 3267 | my $check = 0; |
| 3268 | $s =~ s/^.*\bdo\b//; |
| 3269 | $s =~ s/^\s*{//; |
| 3270 | if ($s =~ s/^\s*\\//) { |
| 3271 | $continuation = 1; |
| 3272 | } |
Andy Whitcroft | 9bd49ef | 2008-10-15 22:02:22 -0700 | [diff] [blame] | 3273 | if ($s =~ s/^\s*?\n//) { |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3274 | $check = 1; |
| 3275 | $cond_lines++; |
| 3276 | } |
| 3277 | |
| 3278 | # Also ignore a loop construct at the end of a |
| 3279 | # preprocessor statement. |
| 3280 | if (($prevline =~ /^.\s*#\s*define\s/ || |
| 3281 | $prevline =~ /\\\s*$/) && $continuation == 0) { |
| 3282 | $check = 0; |
| 3283 | } |
| 3284 | |
Andy Whitcroft | 9bd49ef | 2008-10-15 22:02:22 -0700 | [diff] [blame] | 3285 | my $cond_ptr = -1; |
Andy Whitcroft | 740504c | 2008-10-15 22:02:35 -0700 | [diff] [blame] | 3286 | $continuation = 0; |
Andy Whitcroft | 9bd49ef | 2008-10-15 22:02:22 -0700 | [diff] [blame] | 3287 | while ($cond_ptr != $cond_lines) { |
| 3288 | $cond_ptr = $cond_lines; |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3289 | |
Andy Whitcroft | f16fa28 | 2008-10-15 22:02:32 -0700 | [diff] [blame] | 3290 | # If we see an #else/#elif then the code |
| 3291 | # is not linear. |
| 3292 | if ($s =~ /^\s*\#\s*(?:else|elif)/) { |
| 3293 | $check = 0; |
| 3294 | } |
| 3295 | |
Andy Whitcroft | 9bd49ef | 2008-10-15 22:02:22 -0700 | [diff] [blame] | 3296 | # Ignore: |
| 3297 | # 1) blank lines, they should be at 0, |
| 3298 | # 2) preprocessor lines, and |
| 3299 | # 3) labels. |
Andy Whitcroft | 740504c | 2008-10-15 22:02:35 -0700 | [diff] [blame] | 3300 | if ($continuation || |
| 3301 | $s =~ /^\s*?\n/ || |
Andy Whitcroft | 9bd49ef | 2008-10-15 22:02:22 -0700 | [diff] [blame] | 3302 | $s =~ /^\s*#\s*?/ || |
| 3303 | $s =~ /^\s*$Ident\s*:/) { |
Andy Whitcroft | 740504c | 2008-10-15 22:02:35 -0700 | [diff] [blame] | 3304 | $continuation = ($s =~ /^.*?\\\n/) ? 1 : 0; |
Andy Whitcroft | 30dad6e | 2009-09-21 17:04:36 -0700 | [diff] [blame] | 3305 | if ($s =~ s/^.*?\n//) { |
| 3306 | $cond_lines++; |
| 3307 | } |
Andy Whitcroft | 9bd49ef | 2008-10-15 22:02:22 -0700 | [diff] [blame] | 3308 | } |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3309 | } |
| 3310 | |
| 3311 | my (undef, $sindent) = line_stats("+" . $s); |
| 3312 | my $stat_real = raw_line($linenr, $cond_lines); |
| 3313 | |
| 3314 | # Check if either of these lines are modified, else |
| 3315 | # this is not this patch's fault. |
| 3316 | if (!defined($stat_real) || |
| 3317 | $stat !~ /^\+/ && $stat_real !~ /^\+/) { |
| 3318 | $check = 0; |
| 3319 | } |
| 3320 | if (defined($stat_real) && $cond_lines > 1) { |
| 3321 | $stat_real = "[...]\n$stat_real"; |
| 3322 | } |
| 3323 | |
Andy Whitcroft | 9bd49ef | 2008-10-15 22:02:22 -0700 | [diff] [blame] | 3324 | #print "line<$line> prevline<$prevline> indent<$indent> sindent<$sindent> check<$check> continuation<$continuation> s<$s> cond_lines<$cond_lines> stat_real<$stat_real> stat<$stat>\n"; |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3325 | |
Joe Perches | 9f5af48 | 2015-09-09 15:37:30 -0700 | [diff] [blame] | 3326 | if ($check && $s ne '' && |
| 3327 | (($sindent % 8) != 0 || |
| 3328 | ($sindent < $indent) || |
| 3329 | ($sindent > $indent + 8))) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3330 | WARN("SUSPECT_CODE_INDENT", |
| 3331 | "suspect code indent for conditional statements ($indent, $sindent)\n" . $herecurr . "$stat_real\n"); |
Andy Whitcroft | 4d001e4 | 2008-10-15 22:02:21 -0700 | [diff] [blame] | 3332 | } |
| 3333 | } |
| 3334 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3335 | # Track the 'values' across context and added lines. |
| 3336 | my $opline = $line; $opline =~ s/^./ /; |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 3337 | my ($curr_values, $curr_vars) = |
| 3338 | annotate_values($opline . "\n", $prev_values); |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3339 | $curr_values = $prev_values . $curr_values; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3340 | if ($dbg_values) { |
| 3341 | my $outline = $opline; $outline =~ s/\t/ /g; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3342 | print "$linenr > .$outline\n"; |
| 3343 | print "$linenr > $curr_values\n"; |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 3344 | print "$linenr > $curr_vars\n"; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3345 | } |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3346 | $prev_values = substr($curr_values, -1); |
| 3347 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3348 | #ignore lines not being added |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3349 | next if ($line =~ /^[^\+]/); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3350 | |
Joe Perches | a1ce18e | 2016-03-15 14:58:03 -0700 | [diff] [blame] | 3351 | # check for declarations of signed or unsigned without int |
Joe Perches | c844711 | 2016-08-02 14:04:42 -0700 | [diff] [blame] | 3352 | while ($line =~ m{\b($Declare)\s*(?!char\b|short\b|int\b|long\b)\s*($Ident)?\s*[=,;\[\)\(]}g) { |
Joe Perches | a1ce18e | 2016-03-15 14:58:03 -0700 | [diff] [blame] | 3353 | my $type = $1; |
| 3354 | my $var = $2; |
Joe Perches | 207a8e8 | 2016-03-15 14:58:06 -0700 | [diff] [blame] | 3355 | $var = "" if (!defined $var); |
| 3356 | if ($type =~ /^(?:(?:$Storage|$Inline|$Attribute)\s+)*((?:un)?signed)((?:\s*\*)*)\s*$/) { |
Joe Perches | a1ce18e | 2016-03-15 14:58:03 -0700 | [diff] [blame] | 3357 | my $sign = $1; |
| 3358 | my $pointer = $2; |
| 3359 | |
| 3360 | $pointer = "" if (!defined $pointer); |
| 3361 | |
| 3362 | if (WARN("UNSPECIFIED_INT", |
| 3363 | "Prefer '" . trim($sign) . " int" . rtrim($pointer) . "' to bare use of '$sign" . rtrim($pointer) . "'\n" . $herecurr) && |
| 3364 | $fix) { |
| 3365 | my $decl = trim($sign) . " int "; |
Joe Perches | 207a8e8 | 2016-03-15 14:58:06 -0700 | [diff] [blame] | 3366 | my $comp_pointer = $pointer; |
| 3367 | $comp_pointer =~ s/\s//g; |
| 3368 | $decl .= $comp_pointer; |
| 3369 | $decl = rtrim($decl) if ($var eq ""); |
| 3370 | $fixed[$fixlinenr] =~ s@\b$sign\s*\Q$pointer\E\s*$var\b@$decl$var@; |
Joe Perches | a1ce18e | 2016-03-15 14:58:03 -0700 | [diff] [blame] | 3371 | } |
| 3372 | } |
| 3373 | } |
| 3374 | |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3375 | # TEST: allow direct testing of the type matcher. |
Andy Whitcroft | 7429c69 | 2008-07-23 21:29:06 -0700 | [diff] [blame] | 3376 | if ($dbg_type) { |
| 3377 | if ($line =~ /^.\s*$Declare\s*$/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3378 | ERROR("TEST_TYPE", |
| 3379 | "TEST: is type\n" . $herecurr); |
Andy Whitcroft | 7429c69 | 2008-07-23 21:29:06 -0700 | [diff] [blame] | 3380 | } elsif ($dbg_type > 1 && $line =~ /^.+($Declare)/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3381 | ERROR("TEST_NOT_TYPE", |
| 3382 | "TEST: is not type ($1 is)\n". $herecurr); |
Andy Whitcroft | 7429c69 | 2008-07-23 21:29:06 -0700 | [diff] [blame] | 3383 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3384 | next; |
| 3385 | } |
Andy Whitcroft | a1ef277 | 2008-10-15 22:02:17 -0700 | [diff] [blame] | 3386 | # TEST: allow direct testing of the attribute matcher. |
| 3387 | if ($dbg_attr) { |
Andy Whitcroft | 9360b0e | 2009-02-27 14:03:08 -0800 | [diff] [blame] | 3388 | if ($line =~ /^.\s*$Modifier\s*$/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3389 | ERROR("TEST_ATTR", |
| 3390 | "TEST: is attr\n" . $herecurr); |
Andy Whitcroft | 9360b0e | 2009-02-27 14:03:08 -0800 | [diff] [blame] | 3391 | } elsif ($dbg_attr > 1 && $line =~ /^.+($Modifier)/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3392 | ERROR("TEST_NOT_ATTR", |
| 3393 | "TEST: is not attr ($1 is)\n". $herecurr); |
Andy Whitcroft | a1ef277 | 2008-10-15 22:02:17 -0700 | [diff] [blame] | 3394 | } |
| 3395 | next; |
| 3396 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3397 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 3398 | # check for initialisation to aggregates open brace on the next line |
Andy Whitcroft | 99423c2 | 2009-10-26 16:50:15 -0700 | [diff] [blame] | 3399 | if ($line =~ /^.\s*{/ && |
| 3400 | $prevline =~ /(?:^|[^=])=\s*$/) { |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 3401 | if (ERROR("OPEN_BRACE", |
| 3402 | "that open brace { should be on the previous line\n" . $hereprev) && |
Joe Perches | f2d7e4d | 2014-08-06 16:11:07 -0700 | [diff] [blame] | 3403 | $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { |
| 3404 | fix_delete_line($fixlinenr - 1, $prevrawline); |
| 3405 | fix_delete_line($fixlinenr, $rawline); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 3406 | my $fixedline = $prevrawline; |
| 3407 | $fixedline =~ s/\s*=\s*$/ = {/; |
Joe Perches | f2d7e4d | 2014-08-06 16:11:07 -0700 | [diff] [blame] | 3408 | fix_insert_line($fixlinenr, $fixedline); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 3409 | $fixedline = $line; |
| 3410 | $fixedline =~ s/^(.\s*){\s*/$1/; |
Joe Perches | f2d7e4d | 2014-08-06 16:11:07 -0700 | [diff] [blame] | 3411 | fix_insert_line($fixlinenr, $fixedline); |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 3412 | } |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 3413 | } |
| 3414 | |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3415 | # |
| 3416 | # Checks which are anchored on the added line. |
| 3417 | # |
| 3418 | |
| 3419 | # check for malformed paths in #include statements (uses RAW line) |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3420 | if ($rawline =~ m{^.\s*\#\s*include\s+[<"](.*)[">]}) { |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3421 | my $path = $1; |
| 3422 | if ($path =~ m{//}) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3423 | ERROR("MALFORMED_INCLUDE", |
Joe Perches | 495e9d8 | 2012-12-20 15:05:37 -0800 | [diff] [blame] | 3424 | "malformed #include filename\n" . $herecurr); |
| 3425 | } |
| 3426 | if ($path =~ "^uapi/" && $realfile =~ m@\binclude/uapi/@) { |
| 3427 | ERROR("UAPI_INCLUDE", |
| 3428 | "No #include in ...include/uapi/... should use a uapi/ path prefix\n" . $herecurr); |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3429 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3430 | } |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3431 | |
| 3432 | # no C99 // comments |
| 3433 | if ($line =~ m{//}) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3434 | if (ERROR("C99_COMMENTS", |
| 3435 | "do not use C99 // comments\n" . $herecurr) && |
| 3436 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3437 | my $line = $fixed[$fixlinenr]; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3438 | if ($line =~ /\/\/(.*)$/) { |
| 3439 | my $comment = trim($1); |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3440 | $fixed[$fixlinenr] =~ s@\/\/(.*)$@/\* $comment \*/@; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3441 | } |
| 3442 | } |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3443 | } |
| 3444 | # Remove C99 comments. |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3445 | $line =~ s@//.*@@; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3446 | $opline =~ s@//.*@@; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3447 | |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3448 | # EXPORT_SYMBOL should immediately follow the thing it is exporting, consider |
| 3449 | # the whole statement. |
| 3450 | #print "APW <$lines[$realline_next - 1]>\n"; |
| 3451 | if (defined $realline_next && |
| 3452 | exists $lines[$realline_next - 1] && |
| 3453 | !defined $suppress_export{$realline_next} && |
| 3454 | ($lines[$realline_next - 1] =~ /EXPORT_SYMBOL.*\((.*)\)/ || |
| 3455 | $lines[$realline_next - 1] =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { |
Andy Whitcroft | 3cbf62d | 2010-10-26 14:23:18 -0700 | [diff] [blame] | 3456 | # Handle definitions which produce identifiers with |
| 3457 | # a prefix: |
| 3458 | # XXX(foo); |
| 3459 | # EXPORT_SYMBOL(something_foo); |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3460 | my $name = $1; |
Andy Whitcroft | 87a5387 | 2012-01-10 15:10:04 -0800 | [diff] [blame] | 3461 | if ($stat =~ /^(?:.\s*}\s*\n)?.([A-Z_]+)\s*\(\s*($Ident)/ && |
Andy Whitcroft | 3cbf62d | 2010-10-26 14:23:18 -0700 | [diff] [blame] | 3462 | $name =~ /^${Ident}_$2/) { |
| 3463 | #print "FOO C name<$name>\n"; |
| 3464 | $suppress_export{$realline_next} = 1; |
| 3465 | |
| 3466 | } elsif ($stat !~ /(?: |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3467 | \n.}\s*$| |
Andy Whitcroft | 4801205 | 2008-10-15 22:02:34 -0700 | [diff] [blame] | 3468 | ^.DEFINE_$Ident\(\Q$name\E\)| |
| 3469 | ^.DECLARE_$Ident\(\Q$name\E\)| |
| 3470 | ^.LIST_HEAD\(\Q$name\E\)| |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3471 | ^.(?:$Storage\s+)?$Type\s*\(\s*\*\s*\Q$name\E\s*\)\s*\(| |
| 3472 | \b\Q$name\E(?:\s+$Attribute)*\s*(?:;|=|\[|\() |
Andy Whitcroft | 4801205 | 2008-10-15 22:02:34 -0700 | [diff] [blame] | 3473 | )/x) { |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3474 | #print "FOO A<$lines[$realline_next - 1]> stat<$stat> name<$name>\n"; |
| 3475 | $suppress_export{$realline_next} = 2; |
| 3476 | } else { |
| 3477 | $suppress_export{$realline_next} = 1; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3478 | } |
| 3479 | } |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3480 | if (!defined $suppress_export{$linenr} && |
| 3481 | $prevline =~ /^.\s*$/ && |
| 3482 | ($line =~ /EXPORT_SYMBOL.*\((.*)\)/ || |
| 3483 | $line =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) { |
| 3484 | #print "FOO B <$lines[$linenr - 1]>\n"; |
| 3485 | $suppress_export{$linenr} = 2; |
| 3486 | } |
| 3487 | if (defined $suppress_export{$linenr} && |
| 3488 | $suppress_export{$linenr} == 2) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3489 | WARN("EXPORT_SYMBOL", |
| 3490 | "EXPORT_SYMBOL(foo); should immediately follow its function/variable\n" . $herecurr); |
Andy Whitcroft | 2b474a1 | 2009-10-26 16:50:16 -0700 | [diff] [blame] | 3491 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3492 | |
Joe Eloff | 5150bda | 2010-08-09 17:21:00 -0700 | [diff] [blame] | 3493 | # check for global initialisers. |
Joe Perches | 6d32f7a | 2015-11-06 16:31:37 -0800 | [diff] [blame] | 3494 | if ($line =~ /^\+$Type\s*$Ident(?:\s+$Modifier)*\s*=\s*($zero_initializer)\s*;/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3495 | if (ERROR("GLOBAL_INITIALISERS", |
Joe Perches | 6d32f7a | 2015-11-06 16:31:37 -0800 | [diff] [blame] | 3496 | "do not initialise globals to $1\n" . $herecurr) && |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3497 | $fix) { |
Joe Perches | 6d32f7a | 2015-11-06 16:31:37 -0800 | [diff] [blame] | 3498 | $fixed[$fixlinenr] =~ s/(^.$Type\s*$Ident(?:\s+$Modifier)*)\s*=\s*$zero_initializer\s*;/$1;/; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3499 | } |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 3500 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3501 | # check for static initialisers. |
Joe Perches | 6d32f7a | 2015-11-06 16:31:37 -0800 | [diff] [blame] | 3502 | if ($line =~ /^\+.*\bstatic\s.*=\s*($zero_initializer)\s*;/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3503 | if (ERROR("INITIALISED_STATIC", |
Joe Perches | 6d32f7a | 2015-11-06 16:31:37 -0800 | [diff] [blame] | 3504 | "do not initialise statics to $1\n" . |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3505 | $herecurr) && |
| 3506 | $fix) { |
Joe Perches | 6d32f7a | 2015-11-06 16:31:37 -0800 | [diff] [blame] | 3507 | $fixed[$fixlinenr] =~ s/(\bstatic\s.*?)\s*=\s*$zero_initializer\s*;/$1;/; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3508 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3509 | } |
| 3510 | |
Joe Perches | 1813087 | 2014-08-06 16:11:22 -0700 | [diff] [blame] | 3511 | # check for misordered declarations of char/short/int/long with signed/unsigned |
| 3512 | while ($sline =~ m{(\b$TypeMisordered\b)}g) { |
| 3513 | my $tmp = trim($1); |
| 3514 | WARN("MISORDERED_TYPE", |
| 3515 | "type '$tmp' should be specified in [[un]signed] [short|int|long|long long] order\n" . $herecurr); |
| 3516 | } |
| 3517 | |
Joe Perches | cb710ec | 2010-10-26 14:23:20 -0700 | [diff] [blame] | 3518 | # check for static const char * arrays. |
| 3519 | if ($line =~ /\bstatic\s+const\s+char\s*\*\s*(\w+)\s*\[\s*\]\s*=\s*/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3520 | WARN("STATIC_CONST_CHAR_ARRAY", |
| 3521 | "static const char * array should probably be static const char * const\n" . |
Joe Perches | cb710ec | 2010-10-26 14:23:20 -0700 | [diff] [blame] | 3522 | $herecurr); |
| 3523 | } |
| 3524 | |
| 3525 | # check for static char foo[] = "bar" declarations. |
| 3526 | if ($line =~ /\bstatic\s+char\s+(\w+)\s*\[\s*\]\s*=\s*"/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3527 | WARN("STATIC_CONST_CHAR_ARRAY", |
| 3528 | "static char array declaration should probably be static const char\n" . |
Joe Perches | cb710ec | 2010-10-26 14:23:20 -0700 | [diff] [blame] | 3529 | $herecurr); |
| 3530 | } |
| 3531 | |
Joe Perches | ab7e23f | 2015-04-16 12:44:22 -0700 | [diff] [blame] | 3532 | # check for const <foo> const where <foo> is not a pointer or array type |
| 3533 | if ($sline =~ /\bconst\s+($BasicType)\s+const\b/) { |
| 3534 | my $found = $1; |
| 3535 | if ($sline =~ /\bconst\s+\Q$found\E\s+const\b\s*\*/) { |
| 3536 | WARN("CONST_CONST", |
| 3537 | "'const $found const *' should probably be 'const $found * const'\n" . $herecurr); |
| 3538 | } elsif ($sline !~ /\bconst\s+\Q$found\E\s+const\s+\w+\s*\[/) { |
| 3539 | WARN("CONST_CONST", |
| 3540 | "'const $found const' should probably be 'const $found'\n" . $herecurr); |
| 3541 | } |
| 3542 | } |
| 3543 | |
Joe Perches | 9b0fa60 | 2014-04-03 14:49:18 -0700 | [diff] [blame] | 3544 | # check for non-global char *foo[] = {"bar", ...} declarations. |
| 3545 | if ($line =~ /^.\s+(?:static\s+|const\s+)?char\s+\*\s*\w+\s*\[\s*\]\s*=\s*\{/) { |
| 3546 | WARN("STATIC_CONST_CHAR_ARRAY", |
| 3547 | "char * array declaration might be better as static const\n" . |
| 3548 | $herecurr); |
| 3549 | } |
| 3550 | |
Joe Perches | b598b67 | 2015-04-16 12:44:36 -0700 | [diff] [blame] | 3551 | # check for sizeof(foo)/sizeof(foo[0]) that could be ARRAY_SIZE(foo) |
| 3552 | if ($line =~ m@\bsizeof\s*\(\s*($Lval)\s*\)@) { |
| 3553 | my $array = $1; |
| 3554 | if ($line =~ m@\b(sizeof\s*\(\s*\Q$array\E\s*\)\s*/\s*sizeof\s*\(\s*\Q$array\E\s*\[\s*0\s*\]\s*\))@) { |
| 3555 | my $array_div = $1; |
| 3556 | if (WARN("ARRAY_SIZE", |
| 3557 | "Prefer ARRAY_SIZE($array)\n" . $herecurr) && |
| 3558 | $fix) { |
| 3559 | $fixed[$fixlinenr] =~ s/\Q$array_div\E/ARRAY_SIZE($array)/; |
| 3560 | } |
| 3561 | } |
| 3562 | } |
| 3563 | |
Joe Perches | b36190c | 2014-01-27 17:07:18 -0800 | [diff] [blame] | 3564 | # check for function declarations without arguments like "int foo()" |
| 3565 | if ($line =~ /(\b$Type\s+$Ident)\s*\(\s*\)/) { |
| 3566 | if (ERROR("FUNCTION_WITHOUT_ARGS", |
| 3567 | "Bad function definition - $1() should probably be $1(void)\n" . $herecurr) && |
| 3568 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3569 | $fixed[$fixlinenr] =~ s/(\b($Type)\s+($Ident))\s*\(\s*\)/$2 $3(void)/; |
Joe Perches | b36190c | 2014-01-27 17:07:18 -0800 | [diff] [blame] | 3570 | } |
| 3571 | } |
| 3572 | |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3573 | # check for new typedefs, only function parameters and sparse annotations |
| 3574 | # make sense. |
| 3575 | if ($line =~ /\btypedef\s/ && |
Andy Whitcroft | 8054576 | 2009-01-06 14:41:26 -0800 | [diff] [blame] | 3576 | $line !~ /\btypedef\s+$Type\s*\(\s*\*?$Ident\s*\)\s*\(/ && |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3577 | $line !~ /\btypedef\s+$Type\s+$Ident\s*\(/ && |
Andy Whitcroft | 8ed22ca | 2008-10-15 22:02:32 -0700 | [diff] [blame] | 3578 | $line !~ /\b$typeTypedefs\b/ && |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3579 | $line !~ /\b__bitwise(?:__|)\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3580 | WARN("NEW_TYPEDEFS", |
| 3581 | "do not add new typedefs\n" . $herecurr); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3582 | } |
| 3583 | |
| 3584 | # * goes on variable not on type |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3585 | # (char*[ const]) |
Andy Whitcroft | bfcb2cc | 2012-01-10 15:10:15 -0800 | [diff] [blame] | 3586 | while ($line =~ m{(\($NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)\))}g) { |
| 3587 | #print "AA<$1>\n"; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3588 | my ($ident, $from, $to) = ($1, $2, $2); |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 3589 | |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3590 | # Should start with a space. |
| 3591 | $to =~ s/^(\S)/ $1/; |
| 3592 | # Should not end with a space. |
| 3593 | $to =~ s/\s+$//; |
| 3594 | # '*'s should not have spaces between. |
Andy Whitcroft | f9a0b3d | 2009-01-15 13:51:05 -0800 | [diff] [blame] | 3595 | while ($to =~ s/\*\s+\*/\*\*/) { |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3596 | } |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 3597 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3598 | ## print "1: from<$from> to<$to> ident<$ident>\n"; |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3599 | if ($from ne $to) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3600 | if (ERROR("POINTER_LOCATION", |
| 3601 | "\"(foo$from)\" should be \"(foo$to)\"\n" . $herecurr) && |
| 3602 | $fix) { |
| 3603 | my $sub_from = $ident; |
| 3604 | my $sub_to = $ident; |
| 3605 | $sub_to =~ s/\Q$from\E/$to/; |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3606 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3607 | s@\Q$sub_from\E@$sub_to@; |
| 3608 | } |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3609 | } |
Andy Whitcroft | bfcb2cc | 2012-01-10 15:10:15 -0800 | [diff] [blame] | 3610 | } |
| 3611 | while ($line =~ m{(\b$NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)($Ident))}g) { |
| 3612 | #print "BB<$1>\n"; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3613 | my ($match, $from, $to, $ident) = ($1, $2, $2, $3); |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 3614 | |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3615 | # Should start with a space. |
| 3616 | $to =~ s/^(\S)/ $1/; |
| 3617 | # Should not end with a space. |
| 3618 | $to =~ s/\s+$//; |
| 3619 | # '*'s should not have spaces between. |
Andy Whitcroft | f9a0b3d | 2009-01-15 13:51:05 -0800 | [diff] [blame] | 3620 | while ($to =~ s/\*\s+\*/\*\*/) { |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3621 | } |
| 3622 | # Modifiers should have spaces. |
| 3623 | $to =~ s/(\b$Modifier$)/$1 /; |
| 3624 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3625 | ## print "2: from<$from> to<$to> ident<$ident>\n"; |
Andy Whitcroft | 667026e | 2009-02-27 14:03:08 -0800 | [diff] [blame] | 3626 | if ($from ne $to && $ident !~ /^$Modifier$/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3627 | if (ERROR("POINTER_LOCATION", |
| 3628 | "\"foo${from}bar\" should be \"foo${to}bar\"\n" . $herecurr) && |
| 3629 | $fix) { |
| 3630 | |
| 3631 | my $sub_from = $match; |
| 3632 | my $sub_to = $match; |
| 3633 | $sub_to =~ s/\Q$from\E/$to/; |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3634 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3635 | s@\Q$sub_from\E@$sub_to@; |
| 3636 | } |
Andy Whitcroft | 6586386 | 2009-01-06 14:41:21 -0800 | [diff] [blame] | 3637 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3638 | } |
| 3639 | |
Joe Perches | 9d3e3c7 | 2015-09-09 15:37:27 -0700 | [diff] [blame] | 3640 | # avoid BUG() or BUG_ON() |
| 3641 | if ($line =~ /\b(?:BUG|BUG_ON)\b/) { |
| 3642 | my $msg_type = \&WARN; |
| 3643 | $msg_type = \&CHK if ($file); |
| 3644 | &{$msg_type}("AVOID_BUG", |
| 3645 | "Avoid crashing the kernel - try using WARN_ON & recovery code rather than BUG() or BUG_ON()\n" . $herecurr); |
| 3646 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3647 | |
Joe Perches | 9d3e3c7 | 2015-09-09 15:37:27 -0700 | [diff] [blame] | 3648 | # avoid LINUX_VERSION_CODE |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3649 | if ($line =~ /\bLINUX_VERSION_CODE\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3650 | WARN("LINUX_VERSION_CODE", |
| 3651 | "LINUX_VERSION_CODE should be avoided, code should be for the version to which it is merged\n" . $herecurr); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3652 | } |
| 3653 | |
Joe Perches | 1744122 | 2011-06-15 15:08:17 -0700 | [diff] [blame] | 3654 | # check for uses of printk_ratelimit |
| 3655 | if ($line =~ /\bprintk_ratelimit\s*\(/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3656 | WARN("PRINTK_RATELIMITED", |
Joe Perches | 101ee68 | 2015-02-13 14:38:54 -0800 | [diff] [blame] | 3657 | "Prefer printk_ratelimited or pr_<level>_ratelimited to printk_ratelimit\n" . $herecurr); |
Joe Perches | 1744122 | 2011-06-15 15:08:17 -0700 | [diff] [blame] | 3658 | } |
| 3659 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3660 | # printk should use KERN_* levels. Note that follow on printk's on the |
| 3661 | # same line do not need a level, so we use the current block context |
| 3662 | # to try and find and validate the current printk. In summary the current |
Lucas De Marchi | 25985ed | 2011-03-30 22:57:33 -0300 | [diff] [blame] | 3663 | # printk includes all preceding printk's which have no newline on the end. |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3664 | # we assume the first bad printk is the one to report. |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 3665 | if ($line =~ /\bprintk\((?!KERN_)\s*"/) { |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3666 | my $ok = 0; |
| 3667 | for (my $ln = $linenr - 1; $ln >= $first_line; $ln--) { |
| 3668 | #print "CHECK<$lines[$ln - 1]\n"; |
Lucas De Marchi | 25985ed | 2011-03-30 22:57:33 -0300 | [diff] [blame] | 3669 | # we have a preceding printk if it ends |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3670 | # with "\n" ignore it, else it is to blame |
| 3671 | if ($lines[$ln - 1] =~ m{\bprintk\(}) { |
| 3672 | if ($rawlines[$ln - 1] !~ m{\\n"}) { |
| 3673 | $ok = 1; |
| 3674 | } |
| 3675 | last; |
| 3676 | } |
| 3677 | } |
| 3678 | if ($ok == 0) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 3679 | WARN("PRINTK_WITHOUT_KERN_LEVEL", |
| 3680 | "printk() should include KERN_ facility level\n" . $herecurr); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3681 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3682 | } |
| 3683 | |
Joe Perches | 243f380 | 2012-05-31 16:26:09 -0700 | [diff] [blame] | 3684 | if ($line =~ /\bprintk\s*\(\s*KERN_([A-Z]+)/) { |
| 3685 | my $orig = $1; |
| 3686 | my $level = lc($orig); |
| 3687 | $level = "warn" if ($level eq "warning"); |
Joe Perches | 8f26b83 | 2012-10-04 17:13:32 -0700 | [diff] [blame] | 3688 | my $level2 = $level; |
| 3689 | $level2 = "dbg" if ($level eq "debug"); |
Joe Perches | 243f380 | 2012-05-31 16:26:09 -0700 | [diff] [blame] | 3690 | WARN("PREFER_PR_LEVEL", |
Yogesh Chaudhari | daa8b05 | 2014-04-03 14:49:23 -0700 | [diff] [blame] | 3691 | "Prefer [subsystem eg: netdev]_$level2([subsystem]dev, ... then dev_$level2(dev, ... then pr_$level(... to printk(KERN_$orig ...\n" . $herecurr); |
Joe Perches | 243f380 | 2012-05-31 16:26:09 -0700 | [diff] [blame] | 3692 | } |
| 3693 | |
| 3694 | if ($line =~ /\bpr_warning\s*\(/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3695 | if (WARN("PREFER_PR_LEVEL", |
| 3696 | "Prefer pr_warn(... to pr_warning(...\n" . $herecurr) && |
| 3697 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3698 | $fixed[$fixlinenr] =~ |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 3699 | s/\bpr_warning\b/pr_warn/; |
| 3700 | } |
Joe Perches | 243f380 | 2012-05-31 16:26:09 -0700 | [diff] [blame] | 3701 | } |
| 3702 | |
Joe Perches | dc13931 | 2013-02-21 16:44:13 -0800 | [diff] [blame] | 3703 | if ($line =~ /\bdev_printk\s*\(\s*KERN_([A-Z]+)/) { |
| 3704 | my $orig = $1; |
| 3705 | my $level = lc($orig); |
| 3706 | $level = "warn" if ($level eq "warning"); |
| 3707 | $level = "dbg" if ($level eq "debug"); |
| 3708 | WARN("PREFER_DEV_LEVEL", |
| 3709 | "Prefer dev_$level(... to dev_printk(KERN_$orig, ...\n" . $herecurr); |
| 3710 | } |
| 3711 | |
Andy Lutomirski | 91c9afa | 2015-04-16 12:44:44 -0700 | [diff] [blame] | 3712 | # ENOSYS means "bad syscall nr" and nothing else. This will have a small |
| 3713 | # number of false positives, but assembly files are not checked, so at |
| 3714 | # least the arch entry code will not trigger this warning. |
| 3715 | if ($line =~ /\bENOSYS\b/) { |
| 3716 | WARN("ENOSYS", |
| 3717 | "ENOSYS means 'invalid syscall nr' and nothing else\n" . $herecurr); |
| 3718 | } |
| 3719 | |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3720 | # function brace can't be on same line, except for #defines of do while, |
| 3721 | # or if closed on same line |
Joe Perches | 8d18247 | 2014-08-06 16:11:12 -0700 | [diff] [blame] | 3722 | if (($line=~/$Type\s*$Ident\(.*\).*\s*{/) and |
Eddie Kovsky | 4e5d56b | 2015-09-09 15:37:52 -0700 | [diff] [blame] | 3723 | !($line=~/\#\s*define.*do\s\{/) and !($line=~/}/)) { |
Joe Perches | 8d18247 | 2014-08-06 16:11:12 -0700 | [diff] [blame] | 3724 | if (ERROR("OPEN_BRACE", |
| 3725 | "open brace '{' following function declarations go on the next line\n" . $herecurr) && |
| 3726 | $fix) { |
| 3727 | fix_delete_line($fixlinenr, $rawline); |
| 3728 | my $fixed_line = $rawline; |
| 3729 | $fixed_line =~ /(^..*$Type\s*$Ident\(.*\)\s*){(.*)$/; |
| 3730 | my $line1 = $1; |
| 3731 | my $line2 = $2; |
| 3732 | fix_insert_line($fixlinenr, ltrim($line1)); |
| 3733 | fix_insert_line($fixlinenr, "\+{"); |
| 3734 | if ($line2 !~ /^\s*$/) { |
| 3735 | fix_insert_line($fixlinenr, "\+\t" . trim($line2)); |
| 3736 | } |
| 3737 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3738 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3739 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3740 | # open braces for enum, union and struct go on the same line. |
| 3741 | if ($line =~ /^.\s*{/ && |
| 3742 | $prevline =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident)?\s*$/) { |
Joe Perches | 8d18247 | 2014-08-06 16:11:12 -0700 | [diff] [blame] | 3743 | if (ERROR("OPEN_BRACE", |
| 3744 | "open brace '{' following $1 go on the same line\n" . $hereprev) && |
| 3745 | $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { |
| 3746 | fix_delete_line($fixlinenr - 1, $prevrawline); |
| 3747 | fix_delete_line($fixlinenr, $rawline); |
| 3748 | my $fixedline = rtrim($prevrawline) . " {"; |
| 3749 | fix_insert_line($fixlinenr, $fixedline); |
| 3750 | $fixedline = $rawline; |
| 3751 | $fixedline =~ s/^(.\s*){\s*/$1\t/; |
| 3752 | if ($fixedline !~ /^\+\s*$/) { |
| 3753 | fix_insert_line($fixlinenr, $fixedline); |
| 3754 | } |
| 3755 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 3756 | } |
| 3757 | |
Andy Whitcroft | 0c73b4e | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 3758 | # missing space after union, struct or enum definition |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3759 | if ($line =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident){1,2}[=\{]/) { |
| 3760 | if (WARN("SPACING", |
| 3761 | "missing space after $1 definition\n" . $herecurr) && |
| 3762 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3763 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3764 | s/^(.\s*(?:typedef\s+)?(?:enum|union|struct)(?:\s+$Ident){1,2})([=\{])/$1 $2/; |
| 3765 | } |
Andy Whitcroft | 0c73b4e | 2010-10-26 14:23:15 -0700 | [diff] [blame] | 3766 | } |
| 3767 | |
Joe Perches | 31070b5 | 2014-01-23 15:54:49 -0800 | [diff] [blame] | 3768 | # Function pointer declarations |
| 3769 | # check spacing between type, funcptr, and args |
| 3770 | # canonical declaration is "type (*funcptr)(args...)" |
Joe Perches | 91f72e9 | 2014-04-03 14:49:12 -0700 | [diff] [blame] | 3771 | if ($line =~ /^.\s*($Declare)\((\s*)\*(\s*)($Ident)(\s*)\)(\s*)\(/) { |
Joe Perches | 31070b5 | 2014-01-23 15:54:49 -0800 | [diff] [blame] | 3772 | my $declare = $1; |
| 3773 | my $pre_pointer_space = $2; |
| 3774 | my $post_pointer_space = $3; |
| 3775 | my $funcname = $4; |
| 3776 | my $post_funcname_space = $5; |
| 3777 | my $pre_args_space = $6; |
| 3778 | |
Joe Perches | 91f72e9 | 2014-04-03 14:49:12 -0700 | [diff] [blame] | 3779 | # the $Declare variable will capture all spaces after the type |
| 3780 | # so check it for a missing trailing missing space but pointer return types |
| 3781 | # don't need a space so don't warn for those. |
| 3782 | my $post_declare_space = ""; |
| 3783 | if ($declare =~ /(\s+)$/) { |
| 3784 | $post_declare_space = $1; |
| 3785 | $declare = rtrim($declare); |
| 3786 | } |
| 3787 | if ($declare !~ /\*$/ && $post_declare_space =~ /^$/) { |
Joe Perches | 31070b5 | 2014-01-23 15:54:49 -0800 | [diff] [blame] | 3788 | WARN("SPACING", |
| 3789 | "missing space after return type\n" . $herecurr); |
Joe Perches | 91f72e9 | 2014-04-03 14:49:12 -0700 | [diff] [blame] | 3790 | $post_declare_space = " "; |
Joe Perches | 31070b5 | 2014-01-23 15:54:49 -0800 | [diff] [blame] | 3791 | } |
| 3792 | |
| 3793 | # unnecessary space "type (*funcptr)(args...)" |
Joe Perches | 91f72e9 | 2014-04-03 14:49:12 -0700 | [diff] [blame] | 3794 | # This test is not currently implemented because these declarations are |
| 3795 | # equivalent to |
| 3796 | # int foo(int bar, ...) |
| 3797 | # and this is form shouldn't/doesn't generate a checkpatch warning. |
| 3798 | # |
| 3799 | # elsif ($declare =~ /\s{2,}$/) { |
| 3800 | # WARN("SPACING", |
| 3801 | # "Multiple spaces after return type\n" . $herecurr); |
| 3802 | # } |
Joe Perches | 31070b5 | 2014-01-23 15:54:49 -0800 | [diff] [blame] | 3803 | |
| 3804 | # unnecessary space "type ( *funcptr)(args...)" |
| 3805 | if (defined $pre_pointer_space && |
| 3806 | $pre_pointer_space =~ /^\s/) { |
| 3807 | WARN("SPACING", |
| 3808 | "Unnecessary space after function pointer open parenthesis\n" . $herecurr); |
| 3809 | } |
| 3810 | |
| 3811 | # unnecessary space "type (* funcptr)(args...)" |
| 3812 | if (defined $post_pointer_space && |
| 3813 | $post_pointer_space =~ /^\s/) { |
| 3814 | WARN("SPACING", |
| 3815 | "Unnecessary space before function pointer name\n" . $herecurr); |
| 3816 | } |
| 3817 | |
| 3818 | # unnecessary space "type (*funcptr )(args...)" |
| 3819 | if (defined $post_funcname_space && |
| 3820 | $post_funcname_space =~ /^\s/) { |
| 3821 | WARN("SPACING", |
| 3822 | "Unnecessary space after function pointer name\n" . $herecurr); |
| 3823 | } |
| 3824 | |
| 3825 | # unnecessary space "type (*funcptr) (args...)" |
| 3826 | if (defined $pre_args_space && |
| 3827 | $pre_args_space =~ /^\s/) { |
| 3828 | WARN("SPACING", |
| 3829 | "Unnecessary space before function pointer arguments\n" . $herecurr); |
| 3830 | } |
| 3831 | |
| 3832 | if (show_type("SPACING") && $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3833 | $fixed[$fixlinenr] =~ |
Joe Perches | 91f72e9 | 2014-04-03 14:49:12 -0700 | [diff] [blame] | 3834 | s/^(.\s*)$Declare\s*\(\s*\*\s*$Ident\s*\)\s*\(/$1 . $declare . $post_declare_space . '(*' . $funcname . ')('/ex; |
Joe Perches | 31070b5 | 2014-01-23 15:54:49 -0800 | [diff] [blame] | 3835 | } |
| 3836 | } |
| 3837 | |
Andy Whitcroft | 8d31cfc | 2008-07-23 21:29:02 -0700 | [diff] [blame] | 3838 | # check for spacing round square brackets; allowed: |
| 3839 | # 1. with a type on the left -- int [] a; |
Andy Whitcroft | fe2a7db | 2008-10-15 22:02:15 -0700 | [diff] [blame] | 3840 | # 2. at the beginning of a line for slice initialisers -- [0...10] = 5, |
| 3841 | # 3. inside a curly brace -- = { [0...10] = 5 } |
Andy Whitcroft | 8d31cfc | 2008-07-23 21:29:02 -0700 | [diff] [blame] | 3842 | while ($line =~ /(.*?\s)\[/g) { |
| 3843 | my ($where, $prefix) = ($-[1], $1); |
| 3844 | if ($prefix !~ /$Type\s+$/ && |
Andy Whitcroft | fe2a7db | 2008-10-15 22:02:15 -0700 | [diff] [blame] | 3845 | ($where != 0 || $prefix !~ /^.\s+$/) && |
Andy Whitcroft | daebc53 | 2012-03-23 15:02:17 -0700 | [diff] [blame] | 3846 | $prefix !~ /[{,]\s+$/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3847 | if (ERROR("BRACKET_SPACE", |
| 3848 | "space prohibited before open square bracket '['\n" . $herecurr) && |
| 3849 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3850 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3851 | s/^(\+.*?)\s+\[/$1\[/; |
| 3852 | } |
Andy Whitcroft | 8d31cfc | 2008-07-23 21:29:02 -0700 | [diff] [blame] | 3853 | } |
| 3854 | } |
| 3855 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 3856 | # check for spaces between functions and their parentheses. |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3857 | while ($line =~ /($Ident)\s+\(/g) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3858 | my $name = $1; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3859 | my $ctx_before = substr($line, 0, $-[1]); |
| 3860 | my $ctx = "$ctx_before$name"; |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3861 | |
| 3862 | # Ignore those directives where spaces _are_ permitted. |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3863 | if ($name =~ /^(?: |
| 3864 | if|for|while|switch|return|case| |
| 3865 | volatile|__volatile__| |
| 3866 | __attribute__|format|__extension__| |
| 3867 | asm|__asm__)$/x) |
| 3868 | { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3869 | # cpp #define statements have non-optional spaces, ie |
| 3870 | # if there is a space between the name and the open |
| 3871 | # parenthesis it is simply not a parameter group. |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3872 | } elsif ($ctx_before =~ /^.\s*\#\s*define\s*$/) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3873 | |
| 3874 | # cpp #elif statement condition may start with a ( |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 3875 | } elsif ($ctx =~ /^.\s*\#\s*elif\s*$/) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3876 | |
| 3877 | # If this whole things ends with a type its most |
| 3878 | # likely a typedef for a function. |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3879 | } elsif ($ctx =~ /$Type$/) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 3880 | |
| 3881 | } else { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3882 | if (WARN("SPACING", |
| 3883 | "space prohibited between function name and open parenthesis '('\n" . $herecurr) && |
| 3884 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 3885 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3886 | s/\b$name\s+\(/$name\(/; |
| 3887 | } |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3888 | } |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 3889 | } |
Eric Nelson | 9a4cad4 | 2012-05-31 16:26:09 -0700 | [diff] [blame] | 3890 | |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 3891 | # Check operator spacing. |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3892 | if (!($line=~/\#\s*include/)) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3893 | my $fixed_line = ""; |
| 3894 | my $line_fixed = 0; |
| 3895 | |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3896 | my $ops = qr{ |
| 3897 | <<=|>>=|<=|>=|==|!=| |
| 3898 | \+=|-=|\*=|\/=|%=|\^=|\|=|&=| |
| 3899 | =>|->|<<|>>|<|>|=|!|~| |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 3900 | &&|\|\||,|\^|\+\+|--|&|\||\+|-|\*|\/|%| |
Joe Perches | 8473162 | 2013-11-12 15:10:05 -0800 | [diff] [blame] | 3901 | \?:|\?|: |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 3902 | }x; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3903 | my @elements = split(/($ops|;)/, $opline); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3904 | |
| 3905 | ## print("element count: <" . $#elements . ">\n"); |
| 3906 | ## foreach my $el (@elements) { |
| 3907 | ## print("el: <$el>\n"); |
| 3908 | ## } |
| 3909 | |
| 3910 | my @fix_elements = (); |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 3911 | my $off = 0; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3912 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3913 | foreach my $el (@elements) { |
| 3914 | push(@fix_elements, substr($rawline, $off, length($el))); |
| 3915 | $off += length($el); |
| 3916 | } |
| 3917 | |
| 3918 | $off = 0; |
| 3919 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3920 | my $blank = copy_spacing($opline); |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 3921 | my $last_after = -1; |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3922 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3923 | for (my $n = 0; $n < $#elements; $n += 2) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3924 | |
| 3925 | my $good = $fix_elements[$n] . $fix_elements[$n + 1]; |
| 3926 | |
| 3927 | ## print("n: <$n> good: <$good>\n"); |
| 3928 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3929 | $off += length($elements[$n]); |
| 3930 | |
Lucas De Marchi | 25985ed | 2011-03-30 22:57:33 -0300 | [diff] [blame] | 3931 | # Pick up the preceding and succeeding characters. |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3932 | my $ca = substr($opline, 0, $off); |
| 3933 | my $cc = ''; |
| 3934 | if (length($opline) >= ($off + length($elements[$n + 1]))) { |
| 3935 | $cc = substr($opline, $off + length($elements[$n + 1])); |
| 3936 | } |
| 3937 | my $cb = "$ca$;$cc"; |
| 3938 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3939 | my $a = ''; |
| 3940 | $a = 'V' if ($elements[$n] ne ''); |
| 3941 | $a = 'W' if ($elements[$n] =~ /\s$/); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3942 | $a = 'C' if ($elements[$n] =~ /$;$/); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3943 | $a = 'B' if ($elements[$n] =~ /(\[|\()$/); |
| 3944 | $a = 'O' if ($elements[$n] eq ''); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 3945 | $a = 'E' if ($ca =~ /^\s*$/); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3946 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3947 | my $op = $elements[$n + 1]; |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3948 | |
| 3949 | my $c = ''; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3950 | if (defined $elements[$n + 2]) { |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3951 | $c = 'V' if ($elements[$n + 2] ne ''); |
| 3952 | $c = 'W' if ($elements[$n + 2] =~ /^\s/); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3953 | $c = 'C' if ($elements[$n + 2] =~ /^$;/); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3954 | $c = 'B' if ($elements[$n + 2] =~ /^(\)|\]|;)/); |
| 3955 | $c = 'O' if ($elements[$n + 2] eq ''); |
Andy Whitcroft | 8b1b337 | 2009-01-06 14:41:27 -0800 | [diff] [blame] | 3956 | $c = 'E' if ($elements[$n + 2] =~ /^\s*\\$/); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3957 | } else { |
| 3958 | $c = 'E'; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3959 | } |
| 3960 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 3961 | my $ctx = "${a}x${c}"; |
| 3962 | |
| 3963 | my $at = "(ctx:$ctx)"; |
| 3964 | |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3965 | my $ptr = substr($blank, 0, $off) . "^"; |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 3966 | my $hereptr = "$hereline$ptr\n"; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3967 | |
Andy Whitcroft | 74048ed | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 3968 | # Pull out the value of this operator. |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 3969 | my $op_type = substr($curr_values, $off + 1, 1); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3970 | |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 3971 | # Get the full operator variant. |
| 3972 | my $opv = $op . substr($curr_vars, $off, 1); |
| 3973 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 3974 | # Ignore operators passed as parameters. |
| 3975 | if ($op_type ne 'V' && |
Sam Bobroff | d7fe806 | 2015-04-16 12:44:39 -0700 | [diff] [blame] | 3976 | $ca =~ /\s$/ && $cc =~ /^\s*[,\)]/) { |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 3977 | |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3978 | # # Ignore comments |
| 3979 | # } elsif ($op =~ /^$;+$/) { |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 3980 | |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 3981 | # ; should have either the end of line or a space or \ after it |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 3982 | } elsif ($op eq ';') { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 3983 | if ($ctx !~ /.x[WEBC]/ && |
| 3984 | $cc !~ /^\\/ && $cc !~ /^;/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3985 | if (ERROR("SPACING", |
| 3986 | "space required after that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 3987 | $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 3988 | $line_fixed = 1; |
| 3989 | } |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 3990 | } |
| 3991 | |
| 3992 | # // is a comment |
| 3993 | } elsif ($op eq '//') { |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 3994 | |
Joe Perches | b00e481 | 2014-04-03 14:49:33 -0700 | [diff] [blame] | 3995 | # : when part of a bitfield |
| 3996 | } elsif ($opv eq ':B') { |
| 3997 | # skip the bitfield test for now |
| 3998 | |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 3999 | # No spaces for: |
| 4000 | # -> |
Joe Perches | b00e481 | 2014-04-03 14:49:33 -0700 | [diff] [blame] | 4001 | } elsif ($op eq '->') { |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 4002 | if ($ctx =~ /Wx.|.xW/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4003 | if (ERROR("SPACING", |
| 4004 | "spaces prohibited around that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4005 | $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4006 | if (defined $fix_elements[$n + 2]) { |
| 4007 | $fix_elements[$n + 2] =~ s/^\s+//; |
| 4008 | } |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4009 | $line_fixed = 1; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4010 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4011 | } |
| 4012 | |
Joe Perches | 2381097 | 2014-12-10 15:51:32 -0800 | [diff] [blame] | 4013 | # , must not have a space before and must have a space on the right. |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4014 | } elsif ($op eq ',') { |
Joe Perches | 2381097 | 2014-12-10 15:51:32 -0800 | [diff] [blame] | 4015 | my $rtrim_before = 0; |
| 4016 | my $space_after = 0; |
| 4017 | if ($ctx =~ /Wx./) { |
| 4018 | if (ERROR("SPACING", |
| 4019 | "space prohibited before that '$op' $at\n" . $hereptr)) { |
| 4020 | $line_fixed = 1; |
| 4021 | $rtrim_before = 1; |
| 4022 | } |
| 4023 | } |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4024 | if ($ctx !~ /.x[WEC]/ && $cc !~ /^}/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4025 | if (ERROR("SPACING", |
| 4026 | "space required after that '$op' $at\n" . $hereptr)) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4027 | $line_fixed = 1; |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4028 | $last_after = $n; |
Joe Perches | 2381097 | 2014-12-10 15:51:32 -0800 | [diff] [blame] | 4029 | $space_after = 1; |
| 4030 | } |
| 4031 | } |
| 4032 | if ($rtrim_before || $space_after) { |
| 4033 | if ($rtrim_before) { |
| 4034 | $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); |
| 4035 | } else { |
| 4036 | $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); |
| 4037 | } |
| 4038 | if ($space_after) { |
| 4039 | $good .= " "; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4040 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4041 | } |
| 4042 | |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 4043 | # '*' as part of a type definition -- reported already. |
Andy Whitcroft | 74048ed | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 4044 | } elsif ($opv eq '*_') { |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 4045 | #warn "'*' is part of type\n"; |
| 4046 | |
| 4047 | # unary operators should have a space before and |
| 4048 | # none after. May be left adjacent to another |
| 4049 | # unary operator, or a cast |
| 4050 | } elsif ($op eq '!' || $op eq '~' || |
Andy Whitcroft | 74048ed | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 4051 | $opv eq '*U' || $opv eq '-U' || |
Andy Whitcroft | 0d41386 | 2008-10-15 22:02:16 -0700 | [diff] [blame] | 4052 | $opv eq '&U' || $opv eq '&&U') { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4053 | if ($ctx !~ /[WEBC]x./ && $ca !~ /(?:\)|!|~|\*|-|\&|\||\+\+|\-\-|\{)$/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4054 | if (ERROR("SPACING", |
| 4055 | "space required before that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4056 | if ($n != $last_after + 2) { |
| 4057 | $good = $fix_elements[$n] . " " . ltrim($fix_elements[$n + 1]); |
| 4058 | $line_fixed = 1; |
| 4059 | } |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4060 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4061 | } |
Andy Whitcroft | a3340b3 | 2009-02-27 14:03:07 -0800 | [diff] [blame] | 4062 | if ($op eq '*' && $cc =~/\s*$Modifier\b/) { |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 4063 | # A unary '*' may be const |
| 4064 | |
| 4065 | } elsif ($ctx =~ /.xW/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4066 | if (ERROR("SPACING", |
| 4067 | "space prohibited after that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4068 | $good = $fix_elements[$n] . rtrim($fix_elements[$n + 1]); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4069 | if (defined $fix_elements[$n + 2]) { |
| 4070 | $fix_elements[$n + 2] =~ s/^\s+//; |
| 4071 | } |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4072 | $line_fixed = 1; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4073 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4074 | } |
| 4075 | |
| 4076 | # unary ++ and unary -- are allowed no space on one side. |
| 4077 | } elsif ($op eq '++' or $op eq '--') { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4078 | if ($ctx !~ /[WEOBC]x[^W]/ && $ctx !~ /[^W]x[WOBEC]/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4079 | if (ERROR("SPACING", |
| 4080 | "space required one side of that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4081 | $good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " "; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4082 | $line_fixed = 1; |
| 4083 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4084 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4085 | if ($ctx =~ /Wx[BE]/ || |
| 4086 | ($ctx =~ /Wx./ && $cc =~ /^;/)) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4087 | if (ERROR("SPACING", |
| 4088 | "space prohibited before that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4089 | $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4090 | $line_fixed = 1; |
| 4091 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 4092 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4093 | if ($ctx =~ /ExW/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4094 | if (ERROR("SPACING", |
| 4095 | "space prohibited after that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4096 | $good = $fix_elements[$n] . trim($fix_elements[$n + 1]); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4097 | if (defined $fix_elements[$n + 2]) { |
| 4098 | $fix_elements[$n + 2] =~ s/^\s+//; |
| 4099 | } |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4100 | $line_fixed = 1; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4101 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4102 | } |
| 4103 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4104 | # << and >> may either have or not have spaces both sides |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 4105 | } elsif ($op eq '<<' or $op eq '>>' or |
| 4106 | $op eq '&' or $op eq '^' or $op eq '|' or |
| 4107 | $op eq '+' or $op eq '-' or |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 4108 | $op eq '*' or $op eq '/' or |
| 4109 | $op eq '%') |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4110 | { |
Joe Perches | d2e025f | 2015-02-13 14:38:57 -0800 | [diff] [blame] | 4111 | if ($check) { |
| 4112 | if (defined $fix_elements[$n + 2] && $ctx !~ /[EW]x[EW]/) { |
| 4113 | if (CHK("SPACING", |
| 4114 | "spaces preferred around that '$op' $at\n" . $hereptr)) { |
| 4115 | $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; |
| 4116 | $fix_elements[$n + 2] =~ s/^\s+//; |
| 4117 | $line_fixed = 1; |
| 4118 | } |
| 4119 | } elsif (!defined $fix_elements[$n + 2] && $ctx !~ /Wx[OE]/) { |
| 4120 | if (CHK("SPACING", |
| 4121 | "space preferred before that '$op' $at\n" . $hereptr)) { |
| 4122 | $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]); |
| 4123 | $line_fixed = 1; |
| 4124 | } |
| 4125 | } |
| 4126 | } elsif ($ctx =~ /Wx[^WCE]|[^WCE]xW/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4127 | if (ERROR("SPACING", |
| 4128 | "need consistent spacing around '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4129 | $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; |
| 4130 | if (defined $fix_elements[$n + 2]) { |
| 4131 | $fix_elements[$n + 2] =~ s/^\s+//; |
| 4132 | } |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4133 | $line_fixed = 1; |
| 4134 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4135 | } |
| 4136 | |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 4137 | # A colon needs no spaces before when it is |
| 4138 | # terminating a case value or a label. |
| 4139 | } elsif ($opv eq ':C' || $opv eq ':L') { |
| 4140 | if ($ctx =~ /Wx./) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4141 | if (ERROR("SPACING", |
| 4142 | "space prohibited before that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4143 | $good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4144 | $line_fixed = 1; |
| 4145 | } |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 4146 | } |
| 4147 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4148 | # All the others need spaces both sides. |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4149 | } elsif ($ctx !~ /[EWC]x[CWE]/) { |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 4150 | my $ok = 0; |
| 4151 | |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4152 | # Ignore email addresses <foo@bar> |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 4153 | if (($op eq '<' && |
| 4154 | $cc =~ /^\S+\@\S+>/) || |
| 4155 | ($op eq '>' && |
| 4156 | $ca =~ /<\S+\@\S+$/)) |
| 4157 | { |
| 4158 | $ok = 1; |
| 4159 | } |
| 4160 | |
Joe Perches | e0df7e1 | 2015-04-16 12:44:53 -0700 | [diff] [blame] | 4161 | # for asm volatile statements |
| 4162 | # ignore a colon with another |
| 4163 | # colon immediately before or after |
| 4164 | if (($op eq ':') && |
| 4165 | ($ca =~ /:$/ || $cc =~ /^:/)) { |
| 4166 | $ok = 1; |
| 4167 | } |
| 4168 | |
Joe Perches | 8473162 | 2013-11-12 15:10:05 -0800 | [diff] [blame] | 4169 | # messages are ERROR, but ?: are CHK |
Andy Whitcroft | 1f65f94 | 2008-07-23 21:29:10 -0700 | [diff] [blame] | 4170 | if ($ok == 0) { |
Joe Perches | 8473162 | 2013-11-12 15:10:05 -0800 | [diff] [blame] | 4171 | my $msg_type = \&ERROR; |
| 4172 | $msg_type = \&CHK if (($op eq '?:' || $op eq '?' || $op eq ':') && $ctx =~ /VxV/); |
| 4173 | |
| 4174 | if (&{$msg_type}("SPACING", |
| 4175 | "spaces required around that '$op' $at\n" . $hereptr)) { |
Joe Perches | b34c648 | 2013-09-11 14:24:01 -0700 | [diff] [blame] | 4176 | $good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " "; |
| 4177 | if (defined $fix_elements[$n + 2]) { |
| 4178 | $fix_elements[$n + 2] =~ s/^\s+//; |
| 4179 | } |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4180 | $line_fixed = 1; |
| 4181 | } |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4182 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4183 | } |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 4184 | $off += length($elements[$n + 1]); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4185 | |
| 4186 | ## print("n: <$n> GOOD: <$good>\n"); |
| 4187 | |
| 4188 | $fixed_line = $fixed_line . $good; |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4189 | } |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4190 | |
| 4191 | if (($#elements % 2) == 0) { |
| 4192 | $fixed_line = $fixed_line . $fix_elements[$#elements]; |
| 4193 | } |
| 4194 | |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4195 | if ($fix && $line_fixed && $fixed_line ne $fixed[$fixlinenr]) { |
| 4196 | $fixed[$fixlinenr] = $fixed_line; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4197 | } |
| 4198 | |
| 4199 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4200 | } |
| 4201 | |
Joe Perches | 786b632 | 2013-07-03 15:05:32 -0700 | [diff] [blame] | 4202 | # check for whitespace before a non-naked semicolon |
Joe Perches | d2e248e | 2014-01-23 15:54:41 -0800 | [diff] [blame] | 4203 | if ($line =~ /^\+.*\S\s+;\s*$/) { |
Joe Perches | 786b632 | 2013-07-03 15:05:32 -0700 | [diff] [blame] | 4204 | if (WARN("SPACING", |
| 4205 | "space prohibited before semicolon\n" . $herecurr) && |
| 4206 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4207 | 1 while $fixed[$fixlinenr] =~ |
Joe Perches | 786b632 | 2013-07-03 15:05:32 -0700 | [diff] [blame] | 4208 | s/^(\+.*\S)\s+;/$1;/; |
| 4209 | } |
| 4210 | } |
| 4211 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 4212 | # check for multiple assignments |
| 4213 | if ($line =~ /^.\s*$Lval\s*=\s*$Lval\s*=(?!=)/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4214 | CHK("MULTIPLE_ASSIGNMENTS", |
| 4215 | "multiple assignments should be avoided\n" . $herecurr); |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 4216 | } |
| 4217 | |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4218 | ## # check for multiple declarations, allowing for a function declaration |
| 4219 | ## # continuation. |
| 4220 | ## if ($line =~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Ident.*/ && |
| 4221 | ## $line !~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Type\s*$Ident.*/) { |
| 4222 | ## |
| 4223 | ## # Remove any bracketed sections to ensure we do not |
| 4224 | ## # falsly report the parameters of functions. |
| 4225 | ## my $ln = $line; |
| 4226 | ## while ($ln =~ s/\([^\(\)]*\)//g) { |
| 4227 | ## } |
| 4228 | ## if ($ln =~ /,/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4229 | ## WARN("MULTIPLE_DECLARATION", |
| 4230 | ## "declaring multiple variables together should be avoided\n" . $herecurr); |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4231 | ## } |
| 4232 | ## } |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 4233 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4234 | #need space before brace following if, while, etc |
Geyslan G. Bem | 6b8c69e | 2016-03-15 14:58:09 -0700 | [diff] [blame] | 4235 | if (($line =~ /\(.*\)\{/ && $line !~ /\($Type\)\{/) || |
Eddie Kovsky | 4e5d56b | 2015-09-09 15:37:52 -0700 | [diff] [blame] | 4236 | $line =~ /do\{/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4237 | if (ERROR("SPACING", |
| 4238 | "space required before the open brace '{'\n" . $herecurr) && |
| 4239 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4240 | $fixed[$fixlinenr] =~ s/^(\+.*(?:do|\))){/$1 {/; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4241 | } |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 4242 | } |
| 4243 | |
Joe Perches | c4a62ef | 2013-07-03 15:05:28 -0700 | [diff] [blame] | 4244 | ## # check for blank lines before declarations |
| 4245 | ## if ($line =~ /^.\t+$Type\s+$Ident(?:\s*=.*)?;/ && |
| 4246 | ## $prevrawline =~ /^.\s*$/) { |
| 4247 | ## WARN("SPACING", |
| 4248 | ## "No blank lines before declarations\n" . $hereprev); |
| 4249 | ## } |
| 4250 | ## |
| 4251 | |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 4252 | # closing brace should have a space following it when it has anything |
| 4253 | # on the line |
| 4254 | if ($line =~ /}(?!(?:,|;|\)))\S/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 4255 | if (ERROR("SPACING", |
| 4256 | "space required after that close brace '}'\n" . $herecurr) && |
| 4257 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4258 | $fixed[$fixlinenr] =~ |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 4259 | s/}((?!(?:,|;|\)))\S)/} $1/; |
| 4260 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4261 | } |
| 4262 | |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4263 | # check spacing on square brackets |
| 4264 | if ($line =~ /\[\s/ && $line !~ /\[\s*$/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4265 | if (ERROR("SPACING", |
| 4266 | "space prohibited after that open square bracket '['\n" . $herecurr) && |
| 4267 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4268 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4269 | s/\[\s+/\[/; |
| 4270 | } |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4271 | } |
| 4272 | if ($line =~ /\s\]/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4273 | if (ERROR("SPACING", |
| 4274 | "space prohibited before that close square bracket ']'\n" . $herecurr) && |
| 4275 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4276 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4277 | s/\s+\]/\]/; |
| 4278 | } |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4279 | } |
| 4280 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4281 | # check spacing on parentheses |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 4282 | if ($line =~ /\(\s/ && $line !~ /\(\s*(?:\\)?$/ && |
| 4283 | $line !~ /for\s*\(\s+;/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4284 | if (ERROR("SPACING", |
| 4285 | "space prohibited after that open parenthesis '('\n" . $herecurr) && |
| 4286 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4287 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4288 | s/\(\s+/\(/; |
| 4289 | } |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4290 | } |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4291 | if ($line =~ /(\s+)\)/ && $line !~ /^.\s*\)/ && |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4292 | $line !~ /for\s*\(.*;\s+\)/ && |
| 4293 | $line !~ /:\s+\)/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4294 | if (ERROR("SPACING", |
| 4295 | "space prohibited before that close parenthesis ')'\n" . $herecurr) && |
| 4296 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4297 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4298 | s/\s+\)/\)/; |
| 4299 | } |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 4300 | } |
| 4301 | |
Joe Perches | e2826fd | 2014-08-06 16:10:48 -0700 | [diff] [blame] | 4302 | # check unnecessary parentheses around addressof/dereference single $Lvals |
| 4303 | # ie: &(foo->bar) should be &foo->bar and *(foo->bar) should be *foo->bar |
| 4304 | |
| 4305 | while ($line =~ /(?:[^&]&\s*|\*)\(\s*($Ident\s*(?:$Member\s*)+)\s*\)/g) { |
Joe Perches | ea4acbb | 2014-12-10 15:51:51 -0800 | [diff] [blame] | 4306 | my $var = $1; |
| 4307 | if (CHK("UNNECESSARY_PARENTHESES", |
| 4308 | "Unnecessary parentheses around $var\n" . $herecurr) && |
| 4309 | $fix) { |
| 4310 | $fixed[$fixlinenr] =~ s/\(\s*\Q$var\E\s*\)/$var/; |
| 4311 | } |
| 4312 | } |
| 4313 | |
| 4314 | # check for unnecessary parentheses around function pointer uses |
| 4315 | # ie: (foo->bar)(); should be foo->bar(); |
| 4316 | # but not "if (foo->bar) (" to avoid some false positives |
| 4317 | if ($line =~ /(\bif\s*|)(\(\s*$Ident\s*(?:$Member\s*)+\))[ \t]*\(/ && $1 !~ /^if/) { |
| 4318 | my $var = $2; |
| 4319 | if (CHK("UNNECESSARY_PARENTHESES", |
| 4320 | "Unnecessary parentheses around function pointer $var\n" . $herecurr) && |
| 4321 | $fix) { |
| 4322 | my $var2 = deparenthesize($var); |
| 4323 | $var2 =~ s/\s//g; |
| 4324 | $fixed[$fixlinenr] =~ s/\Q$var\E/$var2/; |
| 4325 | } |
| 4326 | } |
Joe Perches | e2826fd | 2014-08-06 16:10:48 -0700 | [diff] [blame] | 4327 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4328 | #goto labels aren't indented, allow a single space however |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 4329 | if ($line=~/^.\s+[A-Za-z\d_]+:(?![0-9]+)/ and |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4330 | !($line=~/^. [A-Za-z\d_]+:/) and !($line=~/^.\s+default:/)) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4331 | if (WARN("INDENTED_LABEL", |
| 4332 | "labels should not be indented\n" . $herecurr) && |
| 4333 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4334 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4335 | s/^(.)\s+/$1/; |
| 4336 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4337 | } |
| 4338 | |
Joe Perches | 5b9553a | 2014-04-03 14:49:21 -0700 | [diff] [blame] | 4339 | # return is not a function |
Joe Perches | 507e514 | 2013-11-12 15:10:13 -0800 | [diff] [blame] | 4340 | if (defined($stat) && $stat =~ /^.\s*return(\s*)\(/s) { |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4341 | my $spacing = $1; |
Joe Perches | 507e514 | 2013-11-12 15:10:13 -0800 | [diff] [blame] | 4342 | if ($^V && $^V ge 5.10.0 && |
Joe Perches | 5b9553a | 2014-04-03 14:49:21 -0700 | [diff] [blame] | 4343 | $stat =~ /^.\s*return\s*($balanced_parens)\s*;\s*$/) { |
| 4344 | my $value = $1; |
| 4345 | $value = deparenthesize($value); |
| 4346 | if ($value =~ m/^\s*$FuncArg\s*(?:\?|$)/) { |
| 4347 | ERROR("RETURN_PARENTHESES", |
| 4348 | "return is not a function, parentheses are not required\n" . $herecurr); |
| 4349 | } |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4350 | } elsif ($spacing !~ /\s+/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4351 | ERROR("SPACING", |
| 4352 | "space required before the open parenthesis '('\n" . $herecurr); |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4353 | } |
| 4354 | } |
Joe Perches | 507e514 | 2013-11-12 15:10:13 -0800 | [diff] [blame] | 4355 | |
Joe Perches | b43ae21 | 2014-06-23 13:22:07 -0700 | [diff] [blame] | 4356 | # unnecessary return in a void function |
| 4357 | # at end-of-function, with the previous line a single leading tab, then return; |
| 4358 | # and the line before that not a goto label target like "out:" |
| 4359 | if ($sline =~ /^[ \+]}\s*$/ && |
| 4360 | $prevline =~ /^\+\treturn\s*;\s*$/ && |
| 4361 | $linenr >= 3 && |
| 4362 | $lines[$linenr - 3] =~ /^[ +]/ && |
| 4363 | $lines[$linenr - 3] !~ /^[ +]\s*$Ident\s*:/) { |
Joe Perches | 9819cf2 | 2014-06-04 16:12:09 -0700 | [diff] [blame] | 4364 | WARN("RETURN_VOID", |
Joe Perches | b43ae21 | 2014-06-23 13:22:07 -0700 | [diff] [blame] | 4365 | "void function return statements are not generally useful\n" . $hereprev); |
| 4366 | } |
Joe Perches | 9819cf2 | 2014-06-04 16:12:09 -0700 | [diff] [blame] | 4367 | |
Joe Perches | 189248d | 2014-01-23 15:54:47 -0800 | [diff] [blame] | 4368 | # if statements using unnecessary parentheses - ie: if ((foo == bar)) |
| 4369 | if ($^V && $^V ge 5.10.0 && |
| 4370 | $line =~ /\bif\s*((?:\(\s*){2,})/) { |
| 4371 | my $openparens = $1; |
| 4372 | my $count = $openparens =~ tr@\(@\(@; |
| 4373 | my $msg = ""; |
| 4374 | if ($line =~ /\bif\s*(?:\(\s*){$count,$count}$LvalOrFunc\s*($Compare)\s*$LvalOrFunc(?:\s*\)){$count,$count}/) { |
| 4375 | my $comp = $4; #Not $1 because of $LvalOrFunc |
| 4376 | $msg = " - maybe == should be = ?" if ($comp eq "=="); |
| 4377 | WARN("UNNECESSARY_PARENTHESES", |
| 4378 | "Unnecessary parentheses$msg\n" . $herecurr); |
| 4379 | } |
| 4380 | } |
| 4381 | |
Joe Perches | c5595fa | 2015-09-09 15:37:58 -0700 | [diff] [blame] | 4382 | # comparisons with a constant or upper case identifier on the left |
| 4383 | # avoid cases like "foo + BAR < baz" |
| 4384 | # only fix matches surrounded by parentheses to avoid incorrect |
| 4385 | # conversions like "FOO < baz() + 5" being "misfixed" to "baz() > FOO + 5" |
| 4386 | if ($^V && $^V ge 5.10.0 && |
| 4387 | $line =~ /^\+(.*)\b($Constant|[A-Z_][A-Z0-9_]*)\s*($Compare)\s*($LvalOrFunc)/) { |
| 4388 | my $lead = $1; |
| 4389 | my $const = $2; |
| 4390 | my $comp = $3; |
| 4391 | my $to = $4; |
| 4392 | my $newcomp = $comp; |
Joe Perches | f39e176 | 2016-05-20 17:04:02 -0700 | [diff] [blame] | 4393 | if ($lead !~ /(?:$Operators|\.)\s*$/ && |
Joe Perches | c5595fa | 2015-09-09 15:37:58 -0700 | [diff] [blame] | 4394 | $to !~ /^(?:Constant|[A-Z_][A-Z0-9_]*)$/ && |
| 4395 | WARN("CONSTANT_COMPARISON", |
| 4396 | "Comparisons should place the constant on the right side of the test\n" . $herecurr) && |
| 4397 | $fix) { |
| 4398 | if ($comp eq "<") { |
| 4399 | $newcomp = ">"; |
| 4400 | } elsif ($comp eq "<=") { |
| 4401 | $newcomp = ">="; |
| 4402 | } elsif ($comp eq ">") { |
| 4403 | $newcomp = "<"; |
| 4404 | } elsif ($comp eq ">=") { |
| 4405 | $newcomp = "<="; |
| 4406 | } |
| 4407 | $fixed[$fixlinenr] =~ s/\(\s*\Q$const\E\s*$Compare\s*\Q$to\E\s*\)/($to $newcomp $const)/; |
| 4408 | } |
| 4409 | } |
| 4410 | |
Joe Perches | f34e4a4 | 2015-04-16 12:44:19 -0700 | [diff] [blame] | 4411 | # Return of what appears to be an errno should normally be negative |
| 4412 | if ($sline =~ /\breturn(?:\s*\(+\s*|\s+)(E[A-Z]+)(?:\s*\)+\s*|\s*)[;:,]/) { |
Andy Whitcroft | 53a3c44 | 2010-10-26 14:23:14 -0700 | [diff] [blame] | 4413 | my $name = $1; |
| 4414 | if ($name ne 'EOF' && $name ne 'ERROR') { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4415 | WARN("USE_NEGATIVE_ERRNO", |
Joe Perches | f34e4a4 | 2015-04-16 12:44:19 -0700 | [diff] [blame] | 4416 | "return of an errno should typically be negative (ie: return -$1)\n" . $herecurr); |
Andy Whitcroft | 53a3c44 | 2010-10-26 14:23:14 -0700 | [diff] [blame] | 4417 | } |
| 4418 | } |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4419 | |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4420 | # Need a space before open parenthesis after if, while etc |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4421 | if ($line =~ /\b(if|while|for|switch)\(/) { |
| 4422 | if (ERROR("SPACING", |
| 4423 | "space required before the open parenthesis '('\n" . $herecurr) && |
| 4424 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4425 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 4426 | s/\b(if|while|for|switch)\(/$1 \(/; |
| 4427 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4428 | } |
| 4429 | |
Andy Whitcroft | f5fe35d | 2008-07-23 21:29:03 -0700 | [diff] [blame] | 4430 | # Check for illegal assignment in if conditional -- and check for trailing |
| 4431 | # statements after the conditional. |
Andy Whitcroft | 170d3a2 | 2008-10-15 22:02:30 -0700 | [diff] [blame] | 4432 | if ($line =~ /do\s*(?!{)/) { |
Andy Whitcroft | 3e469cd | 2012-01-10 15:10:01 -0800 | [diff] [blame] | 4433 | ($stat, $cond, $line_nr_next, $remain_next, $off_next) = |
| 4434 | ctx_statement_block($linenr, $realcnt, 0) |
| 4435 | if (!defined $stat); |
Andy Whitcroft | 170d3a2 | 2008-10-15 22:02:30 -0700 | [diff] [blame] | 4436 | my ($stat_next) = ctx_statement_block($line_nr_next, |
| 4437 | $remain_next, $off_next); |
| 4438 | $stat_next =~ s/\n./\n /g; |
| 4439 | ##print "stat<$stat> stat_next<$stat_next>\n"; |
| 4440 | |
| 4441 | if ($stat_next =~ /^\s*while\b/) { |
| 4442 | # If the statement carries leading newlines, |
| 4443 | # then count those as offsets. |
| 4444 | my ($whitespace) = |
| 4445 | ($stat_next =~ /^((?:\s*\n[+-])*\s*)/s); |
| 4446 | my $offset = |
| 4447 | statement_rawlines($whitespace) - 1; |
| 4448 | |
| 4449 | $suppress_whiletrailers{$line_nr_next + |
| 4450 | $offset} = 1; |
| 4451 | } |
| 4452 | } |
| 4453 | if (!defined $suppress_whiletrailers{$linenr} && |
Joe Perches | c11230f | 2013-11-21 14:31:57 -0800 | [diff] [blame] | 4454 | defined($stat) && defined($cond) && |
Andy Whitcroft | 170d3a2 | 2008-10-15 22:02:30 -0700 | [diff] [blame] | 4455 | $line =~ /\b(?:if|while|for)\s*\(/ && $line !~ /^.\s*#/) { |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 4456 | my ($s, $c) = ($stat, $cond); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 4457 | |
Andy Whitcroft | b53c8e1 | 2009-01-06 14:41:29 -0800 | [diff] [blame] | 4458 | if ($c =~ /\bif\s*\(.*[^<>!=]=[^=].*/s) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4459 | ERROR("ASSIGN_IN_IF", |
| 4460 | "do not use assignment in if condition\n" . $herecurr); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 4461 | } |
| 4462 | |
| 4463 | # Find out what is on the end of the line after the |
| 4464 | # conditional. |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4465 | substr($s, 0, length($c), ''); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 4466 | $s =~ s/\n.*//g; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4467 | $s =~ s/$;//g; # Remove any comments |
Andy Whitcroft | 5321016 | 2008-07-23 21:29:03 -0700 | [diff] [blame] | 4468 | if (length($c) && $s !~ /^\s*{?\s*\\*\s*$/ && |
| 4469 | $c !~ /}\s*while\s*/) |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4470 | { |
Andy Whitcroft | bb44ad3 | 2008-10-15 22:02:34 -0700 | [diff] [blame] | 4471 | # Find out how long the conditional actually is. |
| 4472 | my @newlines = ($c =~ /\n/gs); |
| 4473 | my $cond_lines = 1 + $#newlines; |
Hidetoshi Seto | 42bdf74 | 2010-03-05 13:43:50 -0800 | [diff] [blame] | 4474 | my $stat_real = ''; |
Andy Whitcroft | bb44ad3 | 2008-10-15 22:02:34 -0700 | [diff] [blame] | 4475 | |
Hidetoshi Seto | 42bdf74 | 2010-03-05 13:43:50 -0800 | [diff] [blame] | 4476 | $stat_real = raw_line($linenr, $cond_lines) |
| 4477 | . "\n" if ($cond_lines); |
Andy Whitcroft | bb44ad3 | 2008-10-15 22:02:34 -0700 | [diff] [blame] | 4478 | if (defined($stat_real) && $cond_lines > 1) { |
| 4479 | $stat_real = "[...]\n$stat_real"; |
| 4480 | } |
| 4481 | |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4482 | ERROR("TRAILING_STATEMENTS", |
| 4483 | "trailing statements should be on next line\n" . $herecurr . $stat_real); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 4484 | } |
| 4485 | } |
| 4486 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4487 | # Check for bitwise tests written as boolean |
| 4488 | if ($line =~ / |
| 4489 | (?: |
| 4490 | (?:\[|\(|\&\&|\|\|) |
| 4491 | \s*0[xX][0-9]+\s* |
| 4492 | (?:\&\&|\|\|) |
| 4493 | | |
| 4494 | (?:\&\&|\|\|) |
| 4495 | \s*0[xX][0-9]+\s* |
| 4496 | (?:\&\&|\|\||\)|\]) |
| 4497 | )/x) |
| 4498 | { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4499 | WARN("HEXADECIMAL_BOOLEAN_TEST", |
| 4500 | "boolean test with hexadecimal, perhaps just 1 \& or \|?\n" . $herecurr); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4501 | } |
| 4502 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 4503 | # if and else should not have general statements after it |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4504 | if ($line =~ /^.\s*(?:}\s*)?else\b(.*)/) { |
| 4505 | my $s = $1; |
| 4506 | $s =~ s/$;//g; # Remove any comments |
| 4507 | if ($s !~ /^\s*(?:\sif|(?:{|)\s*\\?\s*$)/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4508 | ERROR("TRAILING_STATEMENTS", |
| 4509 | "trailing statements should be on next line\n" . $herecurr); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4510 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4511 | } |
Andy Whitcroft | 3966778 | 2009-01-15 13:51:06 -0800 | [diff] [blame] | 4512 | # if should not continue a brace |
| 4513 | if ($line =~ /}\s*if\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4514 | ERROR("TRAILING_STATEMENTS", |
Rasmus Villemoes | 048b123 | 2014-08-06 16:10:37 -0700 | [diff] [blame] | 4515 | "trailing statements should be on next line (or did you mean 'else if'?)\n" . |
Andy Whitcroft | 3966778 | 2009-01-15 13:51:06 -0800 | [diff] [blame] | 4516 | $herecurr); |
| 4517 | } |
Andy Whitcroft | a1080bf | 2008-10-15 22:02:25 -0700 | [diff] [blame] | 4518 | # case and default should not have general statements after them |
| 4519 | if ($line =~ /^.\s*(?:case\s*.*|default\s*):/g && |
| 4520 | $line !~ /\G(?: |
Andy Whitcroft | 3fef12d | 2008-10-15 22:02:36 -0700 | [diff] [blame] | 4521 | (?:\s*$;*)(?:\s*{)?(?:\s*$;*)(?:\s*\\)?\s*$| |
Andy Whitcroft | a1080bf | 2008-10-15 22:02:25 -0700 | [diff] [blame] | 4522 | \s*return\s+ |
| 4523 | )/xg) |
| 4524 | { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4525 | ERROR("TRAILING_STATEMENTS", |
| 4526 | "trailing statements should be on next line\n" . $herecurr); |
Andy Whitcroft | a1080bf | 2008-10-15 22:02:25 -0700 | [diff] [blame] | 4527 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4528 | |
| 4529 | # Check for }<nl>else {, these must be at the same |
| 4530 | # indent level to be relevant to each other. |
Joe Perches | 8b8856f | 2014-08-06 16:11:14 -0700 | [diff] [blame] | 4531 | if ($prevline=~/}\s*$/ and $line=~/^.\s*else\s*/ && |
| 4532 | $previndent == $indent) { |
| 4533 | if (ERROR("ELSE_AFTER_BRACE", |
| 4534 | "else should follow close brace '}'\n" . $hereprev) && |
| 4535 | $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { |
| 4536 | fix_delete_line($fixlinenr - 1, $prevrawline); |
| 4537 | fix_delete_line($fixlinenr, $rawline); |
| 4538 | my $fixedline = $prevrawline; |
| 4539 | $fixedline =~ s/}\s*$//; |
| 4540 | if ($fixedline !~ /^\+\s*$/) { |
| 4541 | fix_insert_line($fixlinenr, $fixedline); |
| 4542 | } |
| 4543 | $fixedline = $rawline; |
| 4544 | $fixedline =~ s/^(.\s*)else/$1} else/; |
| 4545 | fix_insert_line($fixlinenr, $fixedline); |
| 4546 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4547 | } |
| 4548 | |
Joe Perches | 8b8856f | 2014-08-06 16:11:14 -0700 | [diff] [blame] | 4549 | if ($prevline=~/}\s*$/ and $line=~/^.\s*while\s*/ && |
| 4550 | $previndent == $indent) { |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 4551 | my ($s, $c) = ctx_statement_block($linenr, $realcnt, 0); |
| 4552 | |
| 4553 | # Find out what is on the end of the line after the |
| 4554 | # conditional. |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4555 | substr($s, 0, length($c), ''); |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 4556 | $s =~ s/\n.*//g; |
| 4557 | |
| 4558 | if ($s =~ /^\s*;/) { |
Joe Perches | 8b8856f | 2014-08-06 16:11:14 -0700 | [diff] [blame] | 4559 | if (ERROR("WHILE_AFTER_BRACE", |
| 4560 | "while should follow close brace '}'\n" . $hereprev) && |
| 4561 | $fix && $prevline =~ /^\+/ && $line =~ /^\+/) { |
| 4562 | fix_delete_line($fixlinenr - 1, $prevrawline); |
| 4563 | fix_delete_line($fixlinenr, $rawline); |
| 4564 | my $fixedline = $prevrawline; |
| 4565 | my $trailing = $rawline; |
| 4566 | $trailing =~ s/^\+//; |
| 4567 | $trailing = trim($trailing); |
| 4568 | $fixedline =~ s/}\s*$/} $trailing/; |
| 4569 | fix_insert_line($fixlinenr, $fixedline); |
| 4570 | } |
Andy Whitcroft | c2fdda0 | 2008-02-08 04:20:54 -0800 | [diff] [blame] | 4571 | } |
| 4572 | } |
| 4573 | |
Joe Perches | 95e2c60 | 2013-07-03 15:05:20 -0700 | [diff] [blame] | 4574 | #Specific variable tests |
Joe Perches | 323c126 | 2012-12-17 16:02:07 -0800 | [diff] [blame] | 4575 | while ($line =~ m{($Constant|$Lval)}g) { |
| 4576 | my $var = $1; |
Joe Perches | 95e2c60 | 2013-07-03 15:05:20 -0700 | [diff] [blame] | 4577 | |
| 4578 | #gcc binary extension |
| 4579 | if ($var =~ /^$Binary$/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 4580 | if (WARN("GCC_BINARY_CONSTANT", |
| 4581 | "Avoid gcc v4.3+ binary constant extension: <$var>\n" . $herecurr) && |
| 4582 | $fix) { |
| 4583 | my $hexval = sprintf("0x%x", oct($var)); |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4584 | $fixed[$fixlinenr] =~ |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 4585 | s/\b$var\b/$hexval/; |
| 4586 | } |
Joe Perches | 95e2c60 | 2013-07-03 15:05:20 -0700 | [diff] [blame] | 4587 | } |
| 4588 | |
| 4589 | #CamelCase |
Joe Perches | 807bd26 | 2013-07-03 15:05:22 -0700 | [diff] [blame] | 4590 | if ($var !~ /^$Constant$/ && |
Joe Perches | be79794 | 2013-07-03 15:05:20 -0700 | [diff] [blame] | 4591 | $var =~ /[A-Z][a-z]|[a-z][A-Z]/ && |
Joe Perches | 22735ce | 2013-07-03 15:05:33 -0700 | [diff] [blame] | 4592 | #Ignore Page<foo> variants |
Joe Perches | 807bd26 | 2013-07-03 15:05:22 -0700 | [diff] [blame] | 4593 | $var !~ /^(?:Clear|Set|TestClear|TestSet|)Page[A-Z]/ && |
Joe Perches | 22735ce | 2013-07-03 15:05:33 -0700 | [diff] [blame] | 4594 | #Ignore SI style variants like nS, mV and dB (ie: max_uV, regulator_min_uA_show) |
Julius Werner | f512357 | 2014-12-10 15:51:54 -0800 | [diff] [blame] | 4595 | $var !~ /^(?:[a-z_]*?)_?[a-z][A-Z](?:_[a-z_]+)?$/ && |
| 4596 | #Ignore some three character SI units explicitly, like MiB and KHz |
| 4597 | $var !~ /^(?:[a-z_]*?)_?(?:[KMGT]iB|[KMGT]?Hz)(?:_[a-z_]+)?$/) { |
Joe Perches | 7e781f6 | 2013-09-11 14:23:55 -0700 | [diff] [blame] | 4598 | while ($var =~ m{($Ident)}g) { |
| 4599 | my $word = $1; |
| 4600 | next if ($word !~ /[A-Z][a-z]|[a-z][A-Z]/); |
Joe Perches | d8b0771 | 2013-11-12 15:10:06 -0800 | [diff] [blame] | 4601 | if ($check) { |
| 4602 | seed_camelcase_includes(); |
| 4603 | if (!$file && !$camelcase_file_seeded) { |
| 4604 | seed_camelcase_file($realfile); |
| 4605 | $camelcase_file_seeded = 1; |
| 4606 | } |
| 4607 | } |
Joe Perches | 7e781f6 | 2013-09-11 14:23:55 -0700 | [diff] [blame] | 4608 | if (!defined $camelcase{$word}) { |
| 4609 | $camelcase{$word} = 1; |
| 4610 | CHK("CAMELCASE", |
| 4611 | "Avoid CamelCase: <$word>\n" . $herecurr); |
| 4612 | } |
Joe Perches | 3445686 | 2013-07-03 15:05:34 -0700 | [diff] [blame] | 4613 | } |
Joe Perches | 323c126 | 2012-12-17 16:02:07 -0800 | [diff] [blame] | 4614 | } |
| 4615 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4616 | |
| 4617 | #no spaces allowed after \ in define |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 4618 | if ($line =~ /\#\s*define.*\\\s+$/) { |
| 4619 | if (WARN("WHITESPACE_AFTER_LINE_CONTINUATION", |
| 4620 | "Whitespace after \\ makes next lines useless\n" . $herecurr) && |
| 4621 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 4622 | $fixed[$fixlinenr] =~ s/\s+$//; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 4623 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4624 | } |
| 4625 | |
Fabian Frederick | 0e212e0 | 2015-04-16 12:44:25 -0700 | [diff] [blame] | 4626 | # warn if <asm/foo.h> is #included and <linux/foo.h> is available and includes |
| 4627 | # itself <asm/foo.h> (uses RAW line) |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4628 | if ($tree && $rawline =~ m{^.\s*\#\s*include\s*\<asm\/(.*)\.h\>}) { |
Andy Whitcroft | e09dec4 | 2008-10-15 22:02:20 -0700 | [diff] [blame] | 4629 | my $file = "$1.h"; |
| 4630 | my $checkfile = "include/linux/$file"; |
| 4631 | if (-f "$root/$checkfile" && |
| 4632 | $realfile ne $checkfile && |
Wolfram Sang | 7840a94 | 2010-08-09 17:20:57 -0700 | [diff] [blame] | 4633 | $1 !~ /$allowed_asm_includes/) |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4634 | { |
Fabian Frederick | 0e212e0 | 2015-04-16 12:44:25 -0700 | [diff] [blame] | 4635 | my $asminclude = `grep -Ec "#include\\s+<asm/$file>" $root/$checkfile`; |
| 4636 | if ($asminclude > 0) { |
| 4637 | if ($realfile =~ m{^arch/}) { |
| 4638 | CHK("ARCH_INCLUDE_LINUX", |
| 4639 | "Consider using #include <linux/$file> instead of <asm/$file>\n" . $herecurr); |
| 4640 | } else { |
| 4641 | WARN("INCLUDE_LINUX", |
| 4642 | "Use #include <linux/$file> instead of <asm/$file>\n" . $herecurr); |
| 4643 | } |
Andy Whitcroft | e09dec4 | 2008-10-15 22:02:20 -0700 | [diff] [blame] | 4644 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4645 | } |
| 4646 | } |
| 4647 | |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 4648 | # multi-statement macros should be enclosed in a do while loop, grab the |
| 4649 | # first statement and ensure its the whole macro if its not enclosed |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4650 | # in a known good container |
Andy Whitcroft | b8f96a31 | 2008-07-23 21:29:07 -0700 | [diff] [blame] | 4651 | if ($realfile !~ m@/vmlinux.lds.h$@ && |
| 4652 | $line =~ /^.\s*\#\s*define\s*$Ident(\()?/) { |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 4653 | my $ln = $linenr; |
| 4654 | my $cnt = $realcnt; |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4655 | my ($off, $dstat, $dcond, $rest); |
| 4656 | my $ctx = ''; |
Joe Perches | 08a2843 | 2014-10-13 15:51:55 -0700 | [diff] [blame] | 4657 | my $has_flow_statement = 0; |
| 4658 | my $has_arg_concat = 0; |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4659 | ($dstat, $dcond, $ln, $cnt, $off) = |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 4660 | ctx_statement_block($linenr, $realcnt, 0); |
| 4661 | $ctx = $dstat; |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4662 | #print "dstat<$dstat> dcond<$dcond> cnt<$cnt> off<$off>\n"; |
Andy Whitcroft | a3bb97a | 2008-07-23 21:29:00 -0700 | [diff] [blame] | 4663 | #print "LINE<$lines[$ln-1]> len<" . length($lines[$ln-1]) . "\n"; |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4664 | |
Joe Perches | 08a2843 | 2014-10-13 15:51:55 -0700 | [diff] [blame] | 4665 | $has_flow_statement = 1 if ($ctx =~ /\b(goto|return)\b/); |
Joe Perches | 62e15a6 | 2016-01-20 14:59:18 -0800 | [diff] [blame] | 4666 | $has_arg_concat = 1 if ($ctx =~ /\#\#/ && $ctx !~ /\#\#\s*(?:__VA_ARGS__|args)\b/); |
Joe Perches | 08a2843 | 2014-10-13 15:51:55 -0700 | [diff] [blame] | 4667 | |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 4668 | $dstat =~ s/^.\s*\#\s*define\s+$Ident(?:\([^\)]*\))?\s*//; |
Andy Whitcroft | 292f1a9 | 2008-07-23 21:29:11 -0700 | [diff] [blame] | 4669 | $dstat =~ s/$;//g; |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4670 | $dstat =~ s/\\\n.//g; |
| 4671 | $dstat =~ s/^\s*//s; |
| 4672 | $dstat =~ s/\s*$//s; |
| 4673 | |
| 4674 | # Flatten any parentheses and braces |
Andy Whitcroft | bf30d6e | 2008-10-15 22:02:33 -0700 | [diff] [blame] | 4675 | while ($dstat =~ s/\([^\(\)]*\)/1/ || |
| 4676 | $dstat =~ s/\{[^\{\}]*\}/1/ || |
Vladimir Zapolskiy | 6b10df4 | 2016-01-20 14:59:21 -0800 | [diff] [blame] | 4677 | $dstat =~ s/.\[[^\[\]]*\]/1/) |
Andy Whitcroft | bf30d6e | 2008-10-15 22:02:33 -0700 | [diff] [blame] | 4678 | { |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 4679 | } |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 4680 | |
Andy Whitcroft | e45bab8 | 2012-03-23 15:02:18 -0700 | [diff] [blame] | 4681 | # Flatten any obvious string concatentation. |
Joe Perches | 33acb54 | 2015-06-25 15:02:54 -0700 | [diff] [blame] | 4682 | while ($dstat =~ s/($String)\s*$Ident/$1/ || |
| 4683 | $dstat =~ s/$Ident\s*($String)/$1/) |
Andy Whitcroft | e45bab8 | 2012-03-23 15:02:18 -0700 | [diff] [blame] | 4684 | { |
| 4685 | } |
| 4686 | |
Joe Perches | 42e1529 | 2016-03-15 14:58:01 -0700 | [diff] [blame] | 4687 | # Make asm volatile uses seem like a generic function |
| 4688 | $dstat =~ s/\b_*asm_*\s+_*volatile_*\b/asm_volatile/g; |
| 4689 | |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4690 | my $exceptions = qr{ |
| 4691 | $Declare| |
| 4692 | module_param_named| |
Kees Cook | a0a0a7a | 2012-10-04 17:13:38 -0700 | [diff] [blame] | 4693 | MODULE_PARM_DESC| |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4694 | DECLARE_PER_CPU| |
| 4695 | DEFINE_PER_CPU| |
Andy Whitcroft | 383099f | 2009-01-06 14:41:18 -0800 | [diff] [blame] | 4696 | __typeof__\(| |
Stefani Seibold | 22fd2d3 | 2010-03-05 13:43:52 -0800 | [diff] [blame] | 4697 | union| |
| 4698 | struct| |
Andy Whitcroft | ea71a0a | 2009-09-21 17:04:38 -0700 | [diff] [blame] | 4699 | \.$Ident\s*=\s*| |
Vladimir Zapolskiy | 6b10df4 | 2016-01-20 14:59:21 -0800 | [diff] [blame] | 4700 | ^\"|\"$| |
| 4701 | ^\[ |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4702 | }x; |
Andy Whitcroft | 5eaa20b | 2010-10-26 14:23:18 -0700 | [diff] [blame] | 4703 | #print "REST<$rest> dstat<$dstat> ctx<$ctx>\n"; |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 4704 | if ($dstat ne '' && |
| 4705 | $dstat !~ /^(?:$Ident|-?$Constant),$/ && # 10, // foo(), |
| 4706 | $dstat !~ /^(?:$Ident|-?$Constant);$/ && # foo(); |
Joe Perches | 3cc4b1c | 2013-07-03 15:05:27 -0700 | [diff] [blame] | 4707 | $dstat !~ /^[!~-]?(?:$Lval|$Constant)$/ && # 10 // foo() // !foo // ~foo // -foo // foo->bar // foo.bar->baz |
Joe Perches | 356fd39 | 2014-08-06 16:10:31 -0700 | [diff] [blame] | 4708 | $dstat !~ /^'X'$/ && $dstat !~ /^'XX'$/ && # character constants |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 4709 | $dstat !~ /$exceptions/ && |
| 4710 | $dstat !~ /^\.$Ident\s*=/ && # .foo = |
Joe Perches | e942e2c | 2013-04-17 15:58:26 -0700 | [diff] [blame] | 4711 | $dstat !~ /^(?:\#\s*$Ident|\#\s*$Constant)\s*$/ && # stringification #foo |
Andy Whitcroft | 72f115f | 2012-01-10 15:10:06 -0800 | [diff] [blame] | 4712 | $dstat !~ /^do\s*$Constant\s*while\s*$Constant;?$/ && # do {...} while (...); // do {...} while (...) |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 4713 | $dstat !~ /^for\s*$Constant$/ && # for (...) |
| 4714 | $dstat !~ /^for\s*$Constant\s+(?:$Ident|-?$Constant)$/ && # for (...) bar() |
| 4715 | $dstat !~ /^do\s*{/ && # do {... |
Eddie Kovsky | 4e5d56b | 2015-09-09 15:37:52 -0700 | [diff] [blame] | 4716 | $dstat !~ /^\(\{/ && # ({... |
Joe Perches | f95a7e6 | 2013-09-11 14:24:00 -0700 | [diff] [blame] | 4717 | $ctx !~ /^.\s*#\s*define\s+TRACE_(?:SYSTEM|INCLUDE_FILE|INCLUDE_PATH)\b/) |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 4718 | { |
| 4719 | $ctx =~ s/\n*$//; |
| 4720 | my $herectx = $here . "\n"; |
| 4721 | my $cnt = statement_rawlines($ctx); |
| 4722 | |
| 4723 | for (my $n = 0; $n < $cnt; $n++) { |
| 4724 | $herectx .= raw_line($linenr, $n) . "\n"; |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 4725 | } |
| 4726 | |
Andy Whitcroft | f74bd19 | 2012-01-10 15:09:54 -0800 | [diff] [blame] | 4727 | if ($dstat =~ /;/) { |
| 4728 | ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE", |
| 4729 | "Macros with multiple statements should be enclosed in a do - while loop\n" . "$herectx"); |
| 4730 | } else { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4731 | ERROR("COMPLEX_MACRO", |
Andrew Morton | 388982b | 2014-10-13 15:51:40 -0700 | [diff] [blame] | 4732 | "Macros with complex values should be enclosed in parentheses\n" . "$herectx"); |
Andy Whitcroft | d8aaf12 | 2007-06-23 17:16:44 -0700 | [diff] [blame] | 4733 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 4734 | } |
Joe Perches | 5023d34 | 2012-12-17 16:01:47 -0800 | [diff] [blame] | 4735 | |
Joe Perches | 08a2843 | 2014-10-13 15:51:55 -0700 | [diff] [blame] | 4736 | # check for macros with flow control, but without ## concatenation |
| 4737 | # ## concatenation is commonly a macro that defines a function so ignore those |
| 4738 | if ($has_flow_statement && !$has_arg_concat) { |
| 4739 | my $herectx = $here . "\n"; |
| 4740 | my $cnt = statement_rawlines($ctx); |
| 4741 | |
| 4742 | for (my $n = 0; $n < $cnt; $n++) { |
| 4743 | $herectx .= raw_line($linenr, $n) . "\n"; |
| 4744 | } |
| 4745 | WARN("MACRO_WITH_FLOW_CONTROL", |
| 4746 | "Macros with flow control statements should be avoided\n" . "$herectx"); |
| 4747 | } |
| 4748 | |
Joe Perches | 481eb48 | 2012-12-17 16:01:56 -0800 | [diff] [blame] | 4749 | # check for line continuations outside of #defines, preprocessor #, and asm |
Joe Perches | 5023d34 | 2012-12-17 16:01:47 -0800 | [diff] [blame] | 4750 | |
| 4751 | } else { |
| 4752 | if ($prevline !~ /^..*\\$/ && |
Joe Perches | 481eb48 | 2012-12-17 16:01:56 -0800 | [diff] [blame] | 4753 | $line !~ /^\+\s*\#.*\\$/ && # preprocessor |
| 4754 | $line !~ /^\+.*\b(__asm__|asm)\b.*\\$/ && # asm |
Joe Perches | 5023d34 | 2012-12-17 16:01:47 -0800 | [diff] [blame] | 4755 | $line =~ /^\+.*\\$/) { |
| 4756 | WARN("LINE_CONTINUATIONS", |
| 4757 | "Avoid unnecessary line continuations\n" . $herecurr); |
| 4758 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 4759 | } |
| 4760 | |
Joe Perches | b13edf7 | 2012-07-30 14:41:24 -0700 | [diff] [blame] | 4761 | # do {} while (0) macro tests: |
| 4762 | # single-statement macros do not need to be enclosed in do while (0) loop, |
| 4763 | # macro should not end with a semicolon |
| 4764 | if ($^V && $^V ge 5.10.0 && |
| 4765 | $realfile !~ m@/vmlinux.lds.h$@ && |
| 4766 | $line =~ /^.\s*\#\s*define\s+$Ident(\()?/) { |
| 4767 | my $ln = $linenr; |
| 4768 | my $cnt = $realcnt; |
| 4769 | my ($off, $dstat, $dcond, $rest); |
| 4770 | my $ctx = ''; |
| 4771 | ($dstat, $dcond, $ln, $cnt, $off) = |
| 4772 | ctx_statement_block($linenr, $realcnt, 0); |
| 4773 | $ctx = $dstat; |
| 4774 | |
| 4775 | $dstat =~ s/\\\n.//g; |
Joe Perches | 1b36b20 | 2015-02-13 14:38:32 -0800 | [diff] [blame] | 4776 | $dstat =~ s/$;/ /g; |
Joe Perches | b13edf7 | 2012-07-30 14:41:24 -0700 | [diff] [blame] | 4777 | |
| 4778 | if ($dstat =~ /^\+\s*#\s*define\s+$Ident\s*${balanced_parens}\s*do\s*{(.*)\s*}\s*while\s*\(\s*0\s*\)\s*([;\s]*)\s*$/) { |
| 4779 | my $stmts = $2; |
| 4780 | my $semis = $3; |
| 4781 | |
| 4782 | $ctx =~ s/\n*$//; |
| 4783 | my $cnt = statement_rawlines($ctx); |
| 4784 | my $herectx = $here . "\n"; |
| 4785 | |
| 4786 | for (my $n = 0; $n < $cnt; $n++) { |
| 4787 | $herectx .= raw_line($linenr, $n) . "\n"; |
| 4788 | } |
| 4789 | |
Joe Perches | ac8e97f | 2012-08-21 16:15:53 -0700 | [diff] [blame] | 4790 | if (($stmts =~ tr/;/;/) == 1 && |
| 4791 | $stmts !~ /^\s*(if|while|for|switch)\b/) { |
Joe Perches | b13edf7 | 2012-07-30 14:41:24 -0700 | [diff] [blame] | 4792 | WARN("SINGLE_STATEMENT_DO_WHILE_MACRO", |
| 4793 | "Single statement macros should not use a do {} while (0) loop\n" . "$herectx"); |
| 4794 | } |
| 4795 | if (defined $semis && $semis ne "") { |
| 4796 | WARN("DO_WHILE_MACRO_WITH_TRAILING_SEMICOLON", |
| 4797 | "do {} while (0) macros should not be semicolon terminated\n" . "$herectx"); |
| 4798 | } |
Joe Perches | f5ef95b | 2014-06-04 16:12:06 -0700 | [diff] [blame] | 4799 | } elsif ($dstat =~ /^\+\s*#\s*define\s+$Ident.*;\s*$/) { |
| 4800 | $ctx =~ s/\n*$//; |
| 4801 | my $cnt = statement_rawlines($ctx); |
| 4802 | my $herectx = $here . "\n"; |
| 4803 | |
| 4804 | for (my $n = 0; $n < $cnt; $n++) { |
| 4805 | $herectx .= raw_line($linenr, $n) . "\n"; |
| 4806 | } |
| 4807 | |
| 4808 | WARN("TRAILING_SEMICOLON", |
| 4809 | "macros should not use a trailing semicolon\n" . "$herectx"); |
Joe Perches | b13edf7 | 2012-07-30 14:41:24 -0700 | [diff] [blame] | 4810 | } |
| 4811 | } |
| 4812 | |
Mike Frysinger | 080ba92 | 2009-01-06 14:41:25 -0800 | [diff] [blame] | 4813 | # make sure symbols are always wrapped with VMLINUX_SYMBOL() ... |
| 4814 | # all assignments may have only one of the following with an assignment: |
| 4815 | # . |
| 4816 | # ALIGN(...) |
| 4817 | # VMLINUX_SYMBOL(...) |
| 4818 | if ($realfile eq 'vmlinux.lds.h' && $line =~ /(?:(?:^|\s)$Ident\s*=|=\s*$Ident(?:\s|$))/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4819 | WARN("MISSING_VMLINUX_SYMBOL", |
| 4820 | "vmlinux.lds.h needs VMLINUX_SYMBOL() around C-visible symbols\n" . $herecurr); |
Mike Frysinger | 080ba92 | 2009-01-06 14:41:25 -0800 | [diff] [blame] | 4821 | } |
| 4822 | |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 4823 | # check for redundant bracing round if etc |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4824 | if ($line =~ /(^.*)\bif\b/ && $1 !~ /else\s*$/) { |
| 4825 | my ($level, $endln, @chunks) = |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4826 | ctx_statement_full($linenr, $realcnt, 1); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4827 | #print "chunks<$#chunks> linenr<$linenr> endln<$endln> level<$level>\n"; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4828 | #print "APW: <<$chunks[1][0]>><<$chunks[1][1]>>\n"; |
| 4829 | if ($#chunks > 0 && $level == 0) { |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4830 | my @allowed = (); |
| 4831 | my $allow = 0; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4832 | my $seen = 0; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4833 | my $herectx = $here . "\n"; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4834 | my $ln = $linenr - 1; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4835 | for my $chunk (@chunks) { |
| 4836 | my ($cond, $block) = @{$chunk}; |
| 4837 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4838 | # If the condition carries leading newlines, then count those as offsets. |
| 4839 | my ($whitespace) = ($cond =~ /^((?:\s*\n[+-])*\s*)/s); |
| 4840 | my $offset = statement_rawlines($whitespace) - 1; |
| 4841 | |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4842 | $allowed[$allow] = 0; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4843 | #print "COND<$cond> whitespace<$whitespace> offset<$offset>\n"; |
| 4844 | |
| 4845 | # We have looked at and allowed this specific line. |
| 4846 | $suppress_ifbraces{$ln + $offset} = 1; |
| 4847 | |
| 4848 | $herectx .= "$rawlines[$ln + $offset]\n[...]\n"; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4849 | $ln += statement_rawlines($block) - 1; |
| 4850 | |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4851 | substr($block, 0, length($cond), ''); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4852 | |
| 4853 | $seen++ if ($block =~ /^\s*{/); |
| 4854 | |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4855 | #print "cond<$cond> block<$block> allowed<$allowed[$allow]>\n"; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4856 | if (statement_lines($cond) > 1) { |
| 4857 | #print "APW: ALLOWED: cond<$cond>\n"; |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4858 | $allowed[$allow] = 1; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4859 | } |
| 4860 | if ($block =~/\b(?:if|for|while)\b/) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4861 | #print "APW: ALLOWED: block<$block>\n"; |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4862 | $allowed[$allow] = 1; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4863 | } |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4864 | if (statement_block_size($block) > 1) { |
| 4865 | #print "APW: ALLOWED: lines block<$block>\n"; |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4866 | $allowed[$allow] = 1; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4867 | } |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4868 | $allow++; |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4869 | } |
Joe Perches | aad4f61 | 2012-03-23 15:02:19 -0700 | [diff] [blame] | 4870 | if ($seen) { |
| 4871 | my $sum_allowed = 0; |
| 4872 | foreach (@allowed) { |
| 4873 | $sum_allowed += $_; |
| 4874 | } |
| 4875 | if ($sum_allowed == 0) { |
| 4876 | WARN("BRACES", |
| 4877 | "braces {} are not necessary for any arm of this statement\n" . $herectx); |
| 4878 | } elsif ($sum_allowed != $allow && |
| 4879 | $seen != $allow) { |
| 4880 | CHK("BRACES", |
| 4881 | "braces {} should be used on all arms of this statement\n" . $herectx); |
| 4882 | } |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4883 | } |
| 4884 | } |
| 4885 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4886 | if (!defined $suppress_ifbraces{$linenr - 1} && |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 4887 | $line =~ /\b(if|while|for|else)\b/) { |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4888 | my $allowed = 0; |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 4889 | |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4890 | # Check the pre-context. |
| 4891 | if (substr($line, 0, $-[0]) =~ /(\}\s*)$/) { |
| 4892 | #print "APW: ALLOWED: pre<$1>\n"; |
| 4893 | $allowed = 1; |
| 4894 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4895 | |
| 4896 | my ($level, $endln, @chunks) = |
| 4897 | ctx_statement_full($linenr, $realcnt, $-[0]); |
| 4898 | |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4899 | # Check the condition. |
| 4900 | my ($cond, $block) = @{$chunks[0]}; |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4901 | #print "CHECKING<$linenr> cond<$cond> block<$block>\n"; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4902 | if (defined $cond) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4903 | substr($block, 0, length($cond), ''); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4904 | } |
| 4905 | if (statement_lines($cond) > 1) { |
| 4906 | #print "APW: ALLOWED: cond<$cond>\n"; |
| 4907 | $allowed = 1; |
| 4908 | } |
| 4909 | if ($block =~/\b(?:if|for|while)\b/) { |
| 4910 | #print "APW: ALLOWED: block<$block>\n"; |
| 4911 | $allowed = 1; |
| 4912 | } |
| 4913 | if (statement_block_size($block) > 1) { |
| 4914 | #print "APW: ALLOWED: lines block<$block>\n"; |
| 4915 | $allowed = 1; |
| 4916 | } |
| 4917 | # Check the post-context. |
| 4918 | if (defined $chunks[1]) { |
| 4919 | my ($cond, $block) = @{$chunks[1]}; |
| 4920 | if (defined $cond) { |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 4921 | substr($block, 0, length($cond), ''); |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 4922 | } |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4923 | if ($block =~ /^\s*\{/) { |
| 4924 | #print "APW: ALLOWED: chunk-1 block<$block>\n"; |
| 4925 | $allowed = 1; |
| 4926 | } |
| 4927 | } |
| 4928 | if ($level == 0 && $block =~ /^\s*\{/ && !$allowed) { |
Justin P. Mattock | 6993248 | 2011-07-26 23:06:29 -0700 | [diff] [blame] | 4929 | my $herectx = $here . "\n"; |
Andy Whitcroft | f055663 | 2008-10-15 22:02:23 -0700 | [diff] [blame] | 4930 | my $cnt = statement_rawlines($block); |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4931 | |
Andy Whitcroft | f055663 | 2008-10-15 22:02:23 -0700 | [diff] [blame] | 4932 | for (my $n = 0; $n < $cnt; $n++) { |
Justin P. Mattock | 6993248 | 2011-07-26 23:06:29 -0700 | [diff] [blame] | 4933 | $herectx .= raw_line($linenr, $n) . "\n"; |
Andy Whitcroft | cf65504 | 2008-03-04 14:28:20 -0800 | [diff] [blame] | 4934 | } |
| 4935 | |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4936 | WARN("BRACES", |
| 4937 | "braces {} are not necessary for single statement blocks\n" . $herectx); |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 4938 | } |
| 4939 | } |
| 4940 | |
Joe Perches | 0979ae6 | 2012-12-17 16:01:59 -0800 | [diff] [blame] | 4941 | # check for unnecessary blank lines around braces |
Joe Perches | 77b9a53 | 2013-07-03 15:05:29 -0700 | [diff] [blame] | 4942 | if (($line =~ /^.\s*}\s*$/ && $prevrawline =~ /^.\s*$/)) { |
Joe Perches | f8e5821 | 2015-02-13 14:38:46 -0800 | [diff] [blame] | 4943 | if (CHK("BRACES", |
| 4944 | "Blank lines aren't necessary before a close brace '}'\n" . $hereprev) && |
| 4945 | $fix && $prevrawline =~ /^\+/) { |
| 4946 | fix_delete_line($fixlinenr - 1, $prevrawline); |
| 4947 | } |
Joe Perches | 0979ae6 | 2012-12-17 16:01:59 -0800 | [diff] [blame] | 4948 | } |
Joe Perches | 77b9a53 | 2013-07-03 15:05:29 -0700 | [diff] [blame] | 4949 | if (($rawline =~ /^.\s*$/ && $prevline =~ /^..*{\s*$/)) { |
Joe Perches | f8e5821 | 2015-02-13 14:38:46 -0800 | [diff] [blame] | 4950 | if (CHK("BRACES", |
| 4951 | "Blank lines aren't necessary after an open brace '{'\n" . $hereprev) && |
| 4952 | $fix) { |
| 4953 | fix_delete_line($fixlinenr, $rawline); |
| 4954 | } |
Joe Perches | 0979ae6 | 2012-12-17 16:01:59 -0800 | [diff] [blame] | 4955 | } |
| 4956 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 4957 | # no volatiles please |
Andy Whitcroft | 6c72ffa | 2007-10-18 03:05:08 -0700 | [diff] [blame] | 4958 | my $asm_volatile = qr{\b(__asm__|asm)\s+(__volatile__|volatile)\b}; |
| 4959 | if ($line =~ /\bvolatile\b/ && $line !~ /$asm_volatile/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 4960 | WARN("VOLATILE", |
| 4961 | "Use of volatile is usually wrong: see Documentation/volatile-considered-harmful.txt\n" . $herecurr); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 4962 | } |
| 4963 | |
Joe Perches | 5e4f6ba | 2014-12-10 15:52:05 -0800 | [diff] [blame] | 4964 | # Check for user-visible strings broken across lines, which breaks the ability |
| 4965 | # to grep for the string. Make exceptions when the previous string ends in a |
| 4966 | # newline (multiple lines in one string constant) or '\t', '\r', ';', or '{' |
| 4967 | # (common in inline assembly) or is a octal \123 or hexadecimal \xaf value |
Joe Perches | 33acb54 | 2015-06-25 15:02:54 -0700 | [diff] [blame] | 4968 | if ($line =~ /^\+\s*$String/ && |
Joe Perches | 5e4f6ba | 2014-12-10 15:52:05 -0800 | [diff] [blame] | 4969 | $prevline =~ /"\s*$/ && |
| 4970 | $prevrawline !~ /(?:\\(?:[ntr]|[0-7]{1,3}|x[0-9a-fA-F]{1,2})|;\s*|\{\s*)"\s*$/) { |
| 4971 | if (WARN("SPLIT_STRING", |
| 4972 | "quoted string split across lines\n" . $hereprev) && |
| 4973 | $fix && |
| 4974 | $prevrawline =~ /^\+.*"\s*$/ && |
| 4975 | $last_coalesced_string_linenr != $linenr - 1) { |
| 4976 | my $extracted_string = get_quoted_string($line, $rawline); |
| 4977 | my $comma_close = ""; |
| 4978 | if ($rawline =~ /\Q$extracted_string\E(\s*\)\s*;\s*$|\s*,\s*)/) { |
| 4979 | $comma_close = $1; |
| 4980 | } |
| 4981 | |
| 4982 | fix_delete_line($fixlinenr - 1, $prevrawline); |
| 4983 | fix_delete_line($fixlinenr, $rawline); |
| 4984 | my $fixedline = $prevrawline; |
| 4985 | $fixedline =~ s/"\s*$//; |
| 4986 | $fixedline .= substr($extracted_string, 1) . trim($comma_close); |
| 4987 | fix_insert_line($fixlinenr - 1, $fixedline); |
| 4988 | $fixedline = $rawline; |
| 4989 | $fixedline =~ s/\Q$extracted_string\E\Q$comma_close\E//; |
| 4990 | if ($fixedline !~ /\+\s*$/) { |
| 4991 | fix_insert_line($fixlinenr, $fixedline); |
| 4992 | } |
| 4993 | $last_coalesced_string_linenr = $linenr; |
| 4994 | } |
| 4995 | } |
| 4996 | |
| 4997 | # check for missing a space in a string concatenation |
| 4998 | if ($prevrawline =~ /[^\\]\w"$/ && $rawline =~ /^\+[\t ]+"\w/) { |
| 4999 | WARN('MISSING_SPACE', |
| 5000 | "break quoted strings at a space character\n" . $hereprev); |
| 5001 | } |
| 5002 | |
| 5003 | # check for spaces before a quoted newline |
| 5004 | if ($rawline =~ /^.*\".*\s\\n/) { |
| 5005 | if (WARN("QUOTED_WHITESPACE_BEFORE_NEWLINE", |
| 5006 | "unnecessary whitespace before a quoted newline\n" . $herecurr) && |
| 5007 | $fix) { |
| 5008 | $fixed[$fixlinenr] =~ s/^(\+.*\".*)\s+\\n/$1\\n/; |
| 5009 | } |
| 5010 | |
| 5011 | } |
| 5012 | |
Joe Perches | f17dba4 | 2014-10-13 15:51:51 -0700 | [diff] [blame] | 5013 | # concatenated string without spaces between elements |
Joe Perches | 33acb54 | 2015-06-25 15:02:54 -0700 | [diff] [blame] | 5014 | if ($line =~ /$String[A-Z_]/ || $line =~ /[A-Za-z0-9_]$String/) { |
Joe Perches | f17dba4 | 2014-10-13 15:51:51 -0700 | [diff] [blame] | 5015 | CHK("CONCATENATED_STRING", |
| 5016 | "Concatenated strings should use spaces between elements\n" . $herecurr); |
| 5017 | } |
| 5018 | |
Joe Perches | 90ad30e | 2014-12-10 15:51:59 -0800 | [diff] [blame] | 5019 | # uncoalesced string fragments |
Joe Perches | 33acb54 | 2015-06-25 15:02:54 -0700 | [diff] [blame] | 5020 | if ($line =~ /$String\s*"/) { |
Joe Perches | 90ad30e | 2014-12-10 15:51:59 -0800 | [diff] [blame] | 5021 | WARN("STRING_FRAGMENTS", |
| 5022 | "Consecutive strings are generally better as a single string\n" . $herecurr); |
| 5023 | } |
| 5024 | |
Joe Perches | 6e30075 | 2015-09-09 15:37:47 -0700 | [diff] [blame] | 5025 | # check for %L{u,d,i} and 0x%[udi] in strings |
Joe Perches | 5e4f6ba | 2014-12-10 15:52:05 -0800 | [diff] [blame] | 5026 | my $string; |
| 5027 | while ($line =~ /(?:^|")([X\t]*)(?:"|$)/g) { |
| 5028 | $string = substr($rawline, $-[1], $+[1] - $-[1]); |
| 5029 | $string =~ s/%%/__/g; |
Joe Perches | 6e30075 | 2015-09-09 15:37:47 -0700 | [diff] [blame] | 5030 | if ($string =~ /(?<!%)%[\*\d\.\$]*L[udi]/) { |
Joe Perches | 5e4f6ba | 2014-12-10 15:52:05 -0800 | [diff] [blame] | 5031 | WARN("PRINTF_L", |
| 5032 | "\%Ld/%Lu are not-standard C, use %lld/%llu\n" . $herecurr); |
| 5033 | last; |
| 5034 | } |
Joe Perches | 6e30075 | 2015-09-09 15:37:47 -0700 | [diff] [blame] | 5035 | if ($string =~ /0x%[\*\d\.\$\Llzth]*[udi]/) { |
| 5036 | ERROR("PRINTF_0xDECIMAL", |
| 5037 | "Prefixing 0x with decimal output is defective\n" . $herecurr); |
| 5038 | } |
Joe Perches | 5e4f6ba | 2014-12-10 15:52:05 -0800 | [diff] [blame] | 5039 | } |
| 5040 | |
| 5041 | # check for line continuations in quoted strings with odd counts of " |
| 5042 | if ($rawline =~ /\\$/ && $rawline =~ tr/"/"/ % 2) { |
| 5043 | WARN("LINE_CONTINUATIONS", |
| 5044 | "Avoid line continuations in quoted strings\n" . $herecurr); |
| 5045 | } |
| 5046 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 5047 | # warn about #if 0 |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5048 | if ($line =~ /^.\s*\#\s*if\s+0\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5049 | CHK("REDUNDANT_CODE", |
| 5050 | "if this code is redundant consider removing it\n" . |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 5051 | $herecurr); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 5052 | } |
| 5053 | |
Andy Whitcroft | 03df4b5 | 2012-12-17 16:01:52 -0800 | [diff] [blame] | 5054 | # check for needless "if (<foo>) fn(<foo>)" uses |
| 5055 | if ($prevline =~ /\bif\s*\(\s*($Lval)\s*\)/) { |
Joe Perches | 100425d | 2015-09-09 15:37:36 -0700 | [diff] [blame] | 5056 | my $tested = quotemeta($1); |
| 5057 | my $expr = '\s*\(\s*' . $tested . '\s*\)\s*;'; |
| 5058 | if ($line =~ /\b(kfree|usb_free_urb|debugfs_remove(?:_recursive)?|(?:kmem_cache|mempool|dma_pool)_destroy)$expr/) { |
| 5059 | my $func = $1; |
| 5060 | if (WARN('NEEDLESS_IF', |
| 5061 | "$func(NULL) is safe and this check is probably not required\n" . $hereprev) && |
| 5062 | $fix) { |
| 5063 | my $do_fix = 1; |
| 5064 | my $leading_tabs = ""; |
| 5065 | my $new_leading_tabs = ""; |
| 5066 | if ($lines[$linenr - 2] =~ /^\+(\t*)if\s*\(\s*$tested\s*\)\s*$/) { |
| 5067 | $leading_tabs = $1; |
| 5068 | } else { |
| 5069 | $do_fix = 0; |
| 5070 | } |
| 5071 | if ($lines[$linenr - 1] =~ /^\+(\t+)$func\s*\(\s*$tested\s*\)\s*;\s*$/) { |
| 5072 | $new_leading_tabs = $1; |
| 5073 | if (length($leading_tabs) + 1 ne length($new_leading_tabs)) { |
| 5074 | $do_fix = 0; |
| 5075 | } |
| 5076 | } else { |
| 5077 | $do_fix = 0; |
| 5078 | } |
| 5079 | if ($do_fix) { |
| 5080 | fix_delete_line($fixlinenr - 1, $prevrawline); |
| 5081 | $fixed[$fixlinenr] =~ s/^\+$new_leading_tabs/\+$leading_tabs/; |
| 5082 | } |
| 5083 | } |
Greg Kroah-Hartman | 4c432a8 | 2008-07-23 21:29:04 -0700 | [diff] [blame] | 5084 | } |
| 5085 | } |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 5086 | |
Joe Perches | ebfdc40 | 2014-08-06 16:10:27 -0700 | [diff] [blame] | 5087 | # check for unnecessary "Out of Memory" messages |
| 5088 | if ($line =~ /^\+.*\b$logFunctions\s*\(/ && |
| 5089 | $prevline =~ /^[ \+]\s*if\s*\(\s*(\!\s*|NULL\s*==\s*)?($Lval)(\s*==\s*NULL\s*)?\s*\)/ && |
| 5090 | (defined $1 || defined $3) && |
| 5091 | $linenr > 3) { |
| 5092 | my $testval = $2; |
| 5093 | my $testline = $lines[$linenr - 3]; |
| 5094 | |
| 5095 | my ($s, $c) = ctx_statement_block($linenr - 3, $realcnt, 0); |
| 5096 | # print("line: <$line>\nprevline: <$prevline>\ns: <$s>\nc: <$c>\n\n\n"); |
| 5097 | |
| 5098 | if ($c =~ /(?:^|\n)[ \+]\s*(?:$Type\s*)?\Q$testval\E\s*=\s*(?:\([^\)]*\)\s*)?\s*(?:devm_)?(?:[kv][czm]alloc(?:_node|_array)?\b|kstrdup|(?:dev_)?alloc_skb)/) { |
| 5099 | WARN("OOM_MESSAGE", |
| 5100 | "Possible unnecessary 'out of memory' message\n" . $hereprev); |
| 5101 | } |
| 5102 | } |
| 5103 | |
Joe Perches | f78d98f | 2014-10-13 15:52:01 -0700 | [diff] [blame] | 5104 | # check for logging functions with KERN_<LEVEL> |
Paolo Bonzini | dcaf112 | 2015-02-13 14:38:26 -0800 | [diff] [blame] | 5105 | if ($line !~ /printk(?:_ratelimited|_once)?\s*\(/ && |
Joe Perches | f78d98f | 2014-10-13 15:52:01 -0700 | [diff] [blame] | 5106 | $line =~ /\b$logFunctions\s*\(.*\b(KERN_[A-Z]+)\b/) { |
| 5107 | my $level = $1; |
| 5108 | if (WARN("UNNECESSARY_KERN_LEVEL", |
| 5109 | "Possible unnecessary $level\n" . $herecurr) && |
| 5110 | $fix) { |
| 5111 | $fixed[$fixlinenr] =~ s/\s*$level\s*//; |
| 5112 | } |
| 5113 | } |
| 5114 | |
Joe Perches | abb08a5 | 2014-12-10 15:51:46 -0800 | [diff] [blame] | 5115 | # check for mask then right shift without a parentheses |
| 5116 | if ($^V && $^V ge 5.10.0 && |
| 5117 | $line =~ /$LvalOrFunc\s*\&\s*($LvalOrFunc)\s*>>/ && |
| 5118 | $4 !~ /^\&/) { # $LvalOrFunc may be &foo, ignore if so |
| 5119 | WARN("MASK_THEN_SHIFT", |
| 5120 | "Possible precedence defect with mask then right shift - may need parentheses\n" . $herecurr); |
| 5121 | } |
| 5122 | |
Joe Perches | b75ac61 | 2014-12-10 15:52:02 -0800 | [diff] [blame] | 5123 | # check for pointer comparisons to NULL |
| 5124 | if ($^V && $^V ge 5.10.0) { |
| 5125 | while ($line =~ /\b$LvalOrFunc\s*(==|\!=)\s*NULL\b/g) { |
| 5126 | my $val = $1; |
| 5127 | my $equal = "!"; |
| 5128 | $equal = "" if ($4 eq "!="); |
| 5129 | if (CHK("COMPARISON_TO_NULL", |
| 5130 | "Comparison to NULL could be written \"${equal}${val}\"\n" . $herecurr) && |
| 5131 | $fix) { |
| 5132 | $fixed[$fixlinenr] =~ s/\b\Q$val\E\s*(?:==|\!=)\s*NULL\b/$equal$val/; |
| 5133 | } |
| 5134 | } |
| 5135 | } |
| 5136 | |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 5137 | # check for bad placement of section $InitAttribute (e.g.: __initdata) |
| 5138 | if ($line =~ /(\b$InitAttribute\b)/) { |
| 5139 | my $attr = $1; |
| 5140 | if ($line =~ /^\+\s*static\s+(?:const\s+)?(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*[=;]/) { |
| 5141 | my $ptr = $1; |
| 5142 | my $var = $2; |
| 5143 | if ((($ptr =~ /\b(union|struct)\s+$attr\b/ && |
| 5144 | ERROR("MISPLACED_INIT", |
| 5145 | "$attr should be placed after $var\n" . $herecurr)) || |
| 5146 | ($ptr !~ /\b(union|struct)\s+$attr\b/ && |
| 5147 | WARN("MISPLACED_INIT", |
| 5148 | "$attr should be placed after $var\n" . $herecurr))) && |
| 5149 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5150 | $fixed[$fixlinenr] =~ s/(\bstatic\s+(?:const\s+)?)(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*([=;])\s*/"$1" . trim(string_find_replace($2, "\\s*$attr\\s*", " ")) . " " . trim(string_find_replace($3, "\\s*$attr\\s*", "")) . " $attr" . ("$4" eq ";" ? ";" : " = ")/e; |
Joe Perches | 8716de3 | 2013-09-11 14:24:05 -0700 | [diff] [blame] | 5151 | } |
| 5152 | } |
| 5153 | } |
| 5154 | |
Joe Perches | e970b884 | 2013-11-12 15:10:10 -0800 | [diff] [blame] | 5155 | # check for $InitAttributeData (ie: __initdata) with const |
| 5156 | if ($line =~ /\bconst\b/ && $line =~ /($InitAttributeData)/) { |
| 5157 | my $attr = $1; |
| 5158 | $attr =~ /($InitAttributePrefix)(.*)/; |
| 5159 | my $attr_prefix = $1; |
| 5160 | my $attr_type = $2; |
| 5161 | if (ERROR("INIT_ATTRIBUTE", |
| 5162 | "Use of const init definition must use ${attr_prefix}initconst\n" . $herecurr) && |
| 5163 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5164 | $fixed[$fixlinenr] =~ |
Joe Perches | e970b884 | 2013-11-12 15:10:10 -0800 | [diff] [blame] | 5165 | s/$InitAttributeData/${attr_prefix}initconst/; |
| 5166 | } |
| 5167 | } |
| 5168 | |
| 5169 | # check for $InitAttributeConst (ie: __initconst) without const |
| 5170 | if ($line !~ /\bconst\b/ && $line =~ /($InitAttributeConst)/) { |
| 5171 | my $attr = $1; |
| 5172 | if (ERROR("INIT_ATTRIBUTE", |
| 5173 | "Use of $attr requires a separate use of const\n" . $herecurr) && |
| 5174 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5175 | my $lead = $fixed[$fixlinenr] =~ |
Joe Perches | e970b884 | 2013-11-12 15:10:10 -0800 | [diff] [blame] | 5176 | /(^\+\s*(?:static\s+))/; |
| 5177 | $lead = rtrim($1); |
| 5178 | $lead = "$lead " if ($lead !~ /^\+$/); |
| 5179 | $lead = "${lead}const "; |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5180 | $fixed[$fixlinenr] =~ s/(^\+\s*(?:static\s+))/$lead/; |
Joe Perches | e970b884 | 2013-11-12 15:10:10 -0800 | [diff] [blame] | 5181 | } |
| 5182 | } |
| 5183 | |
Joe Perches | c17893c | 2015-04-16 12:44:42 -0700 | [diff] [blame] | 5184 | # check for __read_mostly with const non-pointer (should just be const) |
| 5185 | if ($line =~ /\b__read_mostly\b/ && |
| 5186 | $line =~ /($Type)\s*$Ident/ && $1 !~ /\*\s*$/ && $1 =~ /\bconst\b/) { |
| 5187 | if (ERROR("CONST_READ_MOSTLY", |
| 5188 | "Invalid use of __read_mostly with const type\n" . $herecurr) && |
| 5189 | $fix) { |
| 5190 | $fixed[$fixlinenr] =~ s/\s+__read_mostly\b//; |
| 5191 | } |
| 5192 | } |
| 5193 | |
Joe Perches | fbdb813 | 2014-04-03 14:49:14 -0700 | [diff] [blame] | 5194 | # don't use __constant_<foo> functions outside of include/uapi/ |
| 5195 | if ($realfile !~ m@^include/uapi/@ && |
| 5196 | $line =~ /(__constant_(?:htons|ntohs|[bl]e(?:16|32|64)_to_cpu|cpu_to_[bl]e(?:16|32|64)))\s*\(/) { |
| 5197 | my $constant_func = $1; |
| 5198 | my $func = $constant_func; |
| 5199 | $func =~ s/^__constant_//; |
| 5200 | if (WARN("CONSTANT_CONVERSION", |
| 5201 | "$constant_func should be $func\n" . $herecurr) && |
| 5202 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5203 | $fixed[$fixlinenr] =~ s/\b$constant_func\b/$func/g; |
Joe Perches | fbdb813 | 2014-04-03 14:49:14 -0700 | [diff] [blame] | 5204 | } |
| 5205 | } |
| 5206 | |
Patrick Pannuto | 1a15a25 | 2010-08-09 17:21:01 -0700 | [diff] [blame] | 5207 | # prefer usleep_range over udelay |
Bruce Allan | 37581c2 | 2013-02-21 16:44:19 -0800 | [diff] [blame] | 5208 | if ($line =~ /\budelay\s*\(\s*(\d+)\s*\)/) { |
Joe Perches | 43c1d77 | 2014-04-03 14:49:11 -0700 | [diff] [blame] | 5209 | my $delay = $1; |
Patrick Pannuto | 1a15a25 | 2010-08-09 17:21:01 -0700 | [diff] [blame] | 5210 | # ignore udelay's < 10, however |
Joe Perches | 43c1d77 | 2014-04-03 14:49:11 -0700 | [diff] [blame] | 5211 | if (! ($delay < 10) ) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5212 | CHK("USLEEP_RANGE", |
Joe Perches | 43c1d77 | 2014-04-03 14:49:11 -0700 | [diff] [blame] | 5213 | "usleep_range is preferred over udelay; see Documentation/timers/timers-howto.txt\n" . $herecurr); |
| 5214 | } |
| 5215 | if ($delay > 2000) { |
| 5216 | WARN("LONG_UDELAY", |
| 5217 | "long udelay - prefer mdelay; see arch/arm/include/asm/delay.h\n" . $herecurr); |
Patrick Pannuto | 1a15a25 | 2010-08-09 17:21:01 -0700 | [diff] [blame] | 5218 | } |
| 5219 | } |
| 5220 | |
Patrick Pannuto | 09ef872 | 2010-08-09 17:21:02 -0700 | [diff] [blame] | 5221 | # warn about unexpectedly long msleep's |
| 5222 | if ($line =~ /\bmsleep\s*\((\d+)\);/) { |
| 5223 | if ($1 < 20) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5224 | WARN("MSLEEP", |
Joe Perches | 43c1d77 | 2014-04-03 14:49:11 -0700 | [diff] [blame] | 5225 | "msleep < 20ms can sleep for up to 20ms; see Documentation/timers/timers-howto.txt\n" . $herecurr); |
Patrick Pannuto | 09ef872 | 2010-08-09 17:21:02 -0700 | [diff] [blame] | 5226 | } |
| 5227 | } |
| 5228 | |
Joe Perches | 36ec193 | 2013-07-03 15:05:25 -0700 | [diff] [blame] | 5229 | # check for comparisons of jiffies |
| 5230 | if ($line =~ /\bjiffies\s*$Compare|$Compare\s*jiffies\b/) { |
| 5231 | WARN("JIFFIES_COMPARISON", |
| 5232 | "Comparing jiffies is almost always wrong; prefer time_after, time_before and friends\n" . $herecurr); |
| 5233 | } |
| 5234 | |
Joe Perches | 9d7a34a | 2013-07-03 15:05:26 -0700 | [diff] [blame] | 5235 | # check for comparisons of get_jiffies_64() |
| 5236 | if ($line =~ /\bget_jiffies_64\s*\(\s*\)\s*$Compare|$Compare\s*get_jiffies_64\s*\(\s*\)/) { |
| 5237 | WARN("JIFFIES_COMPARISON", |
| 5238 | "Comparing get_jiffies_64() is almost always wrong; prefer time_after64, time_before64 and friends\n" . $herecurr); |
| 5239 | } |
| 5240 | |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 5241 | # warn about #ifdefs in C files |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5242 | # if ($line =~ /^.\s*\#\s*if(|n)def/ && ($realfile =~ /\.c$/)) { |
Andy Whitcroft | 00df344 | 2007-06-08 13:47:06 -0700 | [diff] [blame] | 5243 | # print "#ifdef in C files should be avoided\n"; |
| 5244 | # print "$herecurr"; |
| 5245 | # $clean = 0; |
| 5246 | # } |
| 5247 | |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 5248 | # warn about spacing in #ifdefs |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5249 | if ($line =~ /^.\s*\#\s*(ifdef|ifndef|elif)\s\s+/) { |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 5250 | if (ERROR("SPACING", |
| 5251 | "exactly one space required after that #$1\n" . $herecurr) && |
| 5252 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5253 | $fixed[$fixlinenr] =~ |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 5254 | s/^(.\s*\#\s*(ifdef|ifndef|elif))\s{2,}/$1 /; |
| 5255 | } |
| 5256 | |
Andy Whitcroft | 22f2a2e | 2007-08-10 13:01:03 -0700 | [diff] [blame] | 5257 | } |
| 5258 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 5259 | # check for spinlock_t definitions without a comment. |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 5260 | if ($line =~ /^.\s*(struct\s+mutex|spinlock_t)\s+\S+;/ || |
| 5261 | $line =~ /^.\s*(DEFINE_MUTEX)\s*\(/) { |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 5262 | my $which = $1; |
| 5263 | if (!ctx_has_comment($first_line, $linenr)) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5264 | CHK("UNCOMMENTED_DEFINITION", |
| 5265 | "$1 definition without comment\n" . $herecurr); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 5266 | } |
| 5267 | } |
| 5268 | # check for memory barriers without a comment. |
Michael S. Tsirkin | 402c255 | 2016-01-04 09:39:01 +0200 | [diff] [blame] | 5269 | |
| 5270 | my $barriers = qr{ |
| 5271 | mb| |
| 5272 | rmb| |
| 5273 | wmb| |
| 5274 | read_barrier_depends |
| 5275 | }x; |
| 5276 | my $barrier_stems = qr{ |
| 5277 | mb__before_atomic| |
| 5278 | mb__after_atomic| |
| 5279 | store_release| |
| 5280 | load_acquire| |
| 5281 | store_mb| |
| 5282 | (?:$barriers) |
| 5283 | }x; |
| 5284 | my $all_barriers = qr{ |
| 5285 | (?:$barriers)| |
Michael S. Tsirkin | 43e361f | 2016-01-04 10:00:10 +0200 | [diff] [blame] | 5286 | smp_(?:$barrier_stems)| |
| 5287 | virt_(?:$barrier_stems) |
Michael S. Tsirkin | 402c255 | 2016-01-04 09:39:01 +0200 | [diff] [blame] | 5288 | }x; |
| 5289 | |
| 5290 | if ($line =~ /\b(?:$all_barriers)\s*\(/) { |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 5291 | if (!ctx_has_comment($first_line, $linenr)) { |
Joe Perches | c1fd7bb | 2013-11-12 15:10:11 -0800 | [diff] [blame] | 5292 | WARN("MEMORY_BARRIER", |
| 5293 | "memory barrier without comment\n" . $herecurr); |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 5294 | } |
| 5295 | } |
Paul E. McKenney | 3ad8177 | 2015-07-02 11:55:40 -0700 | [diff] [blame] | 5296 | |
Michael S. Tsirkin | f4073b0 | 2016-01-04 09:54:51 +0200 | [diff] [blame] | 5297 | my $underscore_smp_barriers = qr{__smp_(?:$barrier_stems)}x; |
| 5298 | |
| 5299 | if ($realfile !~ m@^include/asm-generic/@ && |
| 5300 | $realfile !~ m@/barrier\.h$@ && |
| 5301 | $line =~ m/\b(?:$underscore_smp_barriers)\s*\(/ && |
| 5302 | $line !~ m/^.\s*\#\s*define\s+(?:$underscore_smp_barriers)\s*\(/) { |
| 5303 | WARN("MEMORY_BARRIER", |
| 5304 | "__smp memory barriers shouldn't be used outside barrier.h and asm-generic\n" . $herecurr); |
| 5305 | } |
| 5306 | |
Joe Perches | cb426e9 | 2015-06-25 15:02:46 -0700 | [diff] [blame] | 5307 | # check for waitqueue_active without a comment. |
| 5308 | if ($line =~ /\bwaitqueue_active\s*\(/) { |
| 5309 | if (!ctx_has_comment($first_line, $linenr)) { |
| 5310 | WARN("WAITQUEUE_ACTIVE", |
| 5311 | "waitqueue_active without comment\n" . $herecurr); |
| 5312 | } |
| 5313 | } |
Paul E. McKenney | 3ad8177 | 2015-07-02 11:55:40 -0700 | [diff] [blame] | 5314 | |
| 5315 | # Check for expedited grace periods that interrupt non-idle non-nohz |
| 5316 | # online CPUs. These expedited can therefore degrade real-time response |
| 5317 | # if used carelessly, and should be avoided where not absolutely |
| 5318 | # needed. It is always OK to use synchronize_rcu_expedited() and |
| 5319 | # synchronize_sched_expedited() at boot time (before real-time applications |
| 5320 | # start) and in error situations where real-time response is compromised in |
| 5321 | # any case. Note that synchronize_srcu_expedited() does -not- interrupt |
| 5322 | # other CPUs, so don't warn on uses of synchronize_srcu_expedited(). |
| 5323 | # Of course, nothing comes for free, and srcu_read_lock() and |
| 5324 | # srcu_read_unlock() do contain full memory barriers in payment for |
| 5325 | # synchronize_srcu_expedited() non-interruption properties. |
| 5326 | if ($line =~ /\b(synchronize_rcu_expedited|synchronize_sched_expedited)\(/) { |
| 5327 | WARN("EXPEDITED_RCU_GRACE_PERIOD", |
| 5328 | "expedited RCU grace periods should be avoided where they can degrade real-time response\n" . $herecurr); |
| 5329 | |
| 5330 | } |
| 5331 | |
Andy Whitcroft | 4a0df2e | 2007-06-08 13:46:39 -0700 | [diff] [blame] | 5332 | # check of hardware specific defines |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5333 | if ($line =~ m@^.\s*\#\s*if.*\b(__i386__|__powerpc64__|__sun__|__s390x__)\b@ && $realfile !~ m@include/asm-@) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5334 | CHK("ARCH_DEFINES", |
| 5335 | "architecture specific defines should be avoided\n" . $herecurr); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 5336 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 5337 | |
Tobias Klauser | d4977c7 | 2010-05-24 14:33:30 -0700 | [diff] [blame] | 5338 | # Check that the storage class is at the beginning of a declaration |
| 5339 | if ($line =~ /\b$Storage\b/ && $line !~ /^.\s*$Storage\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5340 | WARN("STORAGE_CLASS", |
| 5341 | "storage class should be at the beginning of the declaration\n" . $herecurr) |
Tobias Klauser | d4977c7 | 2010-05-24 14:33:30 -0700 | [diff] [blame] | 5342 | } |
| 5343 | |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 5344 | # check the location of the inline attribute, that it is between |
| 5345 | # storage class and type. |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 5346 | if ($line =~ /\b$Type\s+$Inline\b/ || |
| 5347 | $line =~ /\b$Inline\s+$Storage\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5348 | ERROR("INLINE_LOCATION", |
| 5349 | "inline keyword should sit between storage class and type\n" . $herecurr); |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 5350 | } |
| 5351 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 5352 | # Check for __inline__ and __inline, prefer inline |
Joe Perches | 2b7ab45 | 2013-11-12 15:10:14 -0800 | [diff] [blame] | 5353 | if ($realfile !~ m@\binclude/uapi/@ && |
| 5354 | $line =~ /\b(__inline__|__inline)\b/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5355 | if (WARN("INLINE", |
| 5356 | "plain inline is preferred over $1\n" . $herecurr) && |
| 5357 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5358 | $fixed[$fixlinenr] =~ s/\b(__inline__|__inline)\b/inline/; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5359 | |
| 5360 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 5361 | } |
| 5362 | |
Joe Perches | 3d130fd | 2011-01-12 17:00:00 -0800 | [diff] [blame] | 5363 | # Check for __attribute__ packed, prefer __packed |
Joe Perches | 2b7ab45 | 2013-11-12 15:10:14 -0800 | [diff] [blame] | 5364 | if ($realfile !~ m@\binclude/uapi/@ && |
| 5365 | $line =~ /\b__attribute__\s*\(\s*\(.*\bpacked\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5366 | WARN("PREFER_PACKED", |
| 5367 | "__packed is preferred over __attribute__((packed))\n" . $herecurr); |
Joe Perches | 3d130fd | 2011-01-12 17:00:00 -0800 | [diff] [blame] | 5368 | } |
| 5369 | |
Joe Perches | 39b7e28 | 2011-07-25 17:13:24 -0700 | [diff] [blame] | 5370 | # Check for __attribute__ aligned, prefer __aligned |
Joe Perches | 2b7ab45 | 2013-11-12 15:10:14 -0800 | [diff] [blame] | 5371 | if ($realfile !~ m@\binclude/uapi/@ && |
| 5372 | $line =~ /\b__attribute__\s*\(\s*\(.*aligned/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5373 | WARN("PREFER_ALIGNED", |
| 5374 | "__aligned(size) is preferred over __attribute__((aligned(size)))\n" . $herecurr); |
Joe Perches | 39b7e28 | 2011-07-25 17:13:24 -0700 | [diff] [blame] | 5375 | } |
| 5376 | |
Joe Perches | 5f14d3b | 2012-01-10 15:09:52 -0800 | [diff] [blame] | 5377 | # Check for __attribute__ format(printf, prefer __printf |
Joe Perches | 2b7ab45 | 2013-11-12 15:10:14 -0800 | [diff] [blame] | 5378 | if ($realfile !~ m@\binclude/uapi/@ && |
| 5379 | $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5380 | if (WARN("PREFER_PRINTF", |
| 5381 | "__printf(string-index, first-to-check) is preferred over __attribute__((format(printf, string-index, first-to-check)))\n" . $herecurr) && |
| 5382 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5383 | $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf\s*,\s*(.*)\)\s*\)\s*\)/"__printf(" . trim($1) . ")"/ex; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5384 | |
| 5385 | } |
Joe Perches | 5f14d3b | 2012-01-10 15:09:52 -0800 | [diff] [blame] | 5386 | } |
| 5387 | |
Joe Perches | 6061d94 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5388 | # Check for __attribute__ format(scanf, prefer __scanf |
Joe Perches | 2b7ab45 | 2013-11-12 15:10:14 -0800 | [diff] [blame] | 5389 | if ($realfile !~ m@\binclude/uapi/@ && |
| 5390 | $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\b/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5391 | if (WARN("PREFER_SCANF", |
| 5392 | "__scanf(string-index, first-to-check) is preferred over __attribute__((format(scanf, string-index, first-to-check)))\n" . $herecurr) && |
| 5393 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5394 | $fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\s*,\s*(.*)\)\s*\)\s*\)/"__scanf(" . trim($1) . ")"/ex; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5395 | } |
Joe Perches | 6061d94 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5396 | } |
| 5397 | |
Joe Perches | 619a908 | 2014-12-10 15:51:35 -0800 | [diff] [blame] | 5398 | # Check for __attribute__ weak, or __weak declarations (may have link issues) |
| 5399 | if ($^V && $^V ge 5.10.0 && |
| 5400 | $line =~ /(?:$Declare|$DeclareMisordered)\s*$Ident\s*$balanced_parens\s*(?:$Attribute)?\s*;/ && |
| 5401 | ($line =~ /\b__attribute__\s*\(\s*\(.*\bweak\b/ || |
| 5402 | $line =~ /\b__weak\b/)) { |
| 5403 | ERROR("WEAK_DECLARATION", |
| 5404 | "Using weak declarations can have unintended link defects\n" . $herecurr); |
| 5405 | } |
| 5406 | |
Joe Perches | e6176fa | 2015-06-25 15:02:49 -0700 | [diff] [blame] | 5407 | # check for c99 types like uint8_t used outside of uapi/ |
| 5408 | if ($realfile !~ m@\binclude/uapi/@ && |
| 5409 | $line =~ /\b($Declare)\s*$Ident\s*[=;,\[]/) { |
| 5410 | my $type = $1; |
| 5411 | if ($type =~ /\b($typeC99Typedefs)\b/) { |
| 5412 | $type = $1; |
| 5413 | my $kernel_type = 'u'; |
| 5414 | $kernel_type = 's' if ($type =~ /^_*[si]/); |
| 5415 | $type =~ /(\d+)/; |
| 5416 | $kernel_type .= $1; |
| 5417 | if (CHK("PREFER_KERNEL_TYPES", |
| 5418 | "Prefer kernel type '$kernel_type' over '$type'\n" . $herecurr) && |
| 5419 | $fix) { |
| 5420 | $fixed[$fixlinenr] =~ s/\b$type\b/$kernel_type/; |
| 5421 | } |
| 5422 | } |
| 5423 | } |
| 5424 | |
Joe Perches | 938224b | 2016-01-20 14:59:15 -0800 | [diff] [blame] | 5425 | # check for cast of C90 native int or longer types constants |
| 5426 | if ($line =~ /(\(\s*$C90_int_types\s*\)\s*)($Constant)\b/) { |
| 5427 | my $cast = $1; |
| 5428 | my $const = $2; |
| 5429 | if (WARN("TYPECAST_INT_CONSTANT", |
| 5430 | "Unnecessary typecast of c90 int constant\n" . $herecurr) && |
| 5431 | $fix) { |
| 5432 | my $suffix = ""; |
| 5433 | my $newconst = $const; |
| 5434 | $newconst =~ s/${Int_type}$//; |
| 5435 | $suffix .= 'U' if ($cast =~ /\bunsigned\b/); |
| 5436 | if ($cast =~ /\blong\s+long\b/) { |
| 5437 | $suffix .= 'LL'; |
| 5438 | } elsif ($cast =~ /\blong\b/) { |
| 5439 | $suffix .= 'L'; |
| 5440 | } |
| 5441 | $fixed[$fixlinenr] =~ s/\Q$cast\E$const\b/$newconst$suffix/; |
| 5442 | } |
| 5443 | } |
| 5444 | |
Joe Perches | 8f53a9b | 2010-03-05 13:43:48 -0800 | [diff] [blame] | 5445 | # check for sizeof(&) |
| 5446 | if ($line =~ /\bsizeof\s*\(\s*\&/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5447 | WARN("SIZEOF_ADDRESS", |
| 5448 | "sizeof(& should be avoided\n" . $herecurr); |
Joe Perches | 8f53a9b | 2010-03-05 13:43:48 -0800 | [diff] [blame] | 5449 | } |
| 5450 | |
Joe Perches | 66c80b6 | 2012-07-30 14:41:22 -0700 | [diff] [blame] | 5451 | # check for sizeof without parenthesis |
| 5452 | if ($line =~ /\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5453 | if (WARN("SIZEOF_PARENTHESIS", |
| 5454 | "sizeof $1 should be sizeof($1)\n" . $herecurr) && |
| 5455 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5456 | $fixed[$fixlinenr] =~ s/\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/"sizeof(" . trim($1) . ")"/ex; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5457 | } |
Joe Perches | 66c80b6 | 2012-07-30 14:41:22 -0700 | [diff] [blame] | 5458 | } |
| 5459 | |
Joe Perches | 88982fe | 2012-12-17 16:02:00 -0800 | [diff] [blame] | 5460 | # check for struct spinlock declarations |
| 5461 | if ($line =~ /^.\s*\bstruct\s+spinlock\s+\w+\s*;/) { |
| 5462 | WARN("USE_SPINLOCK_T", |
| 5463 | "struct spinlock should be spinlock_t\n" . $herecurr); |
| 5464 | } |
| 5465 | |
Joe Perches | a6962d7 | 2013-04-29 16:18:13 -0700 | [diff] [blame] | 5466 | # check for seq_printf uses that could be seq_puts |
Joe Perches | 0666872 | 2013-11-12 15:10:07 -0800 | [diff] [blame] | 5467 | if ($sline =~ /\bseq_printf\s*\(.*"\s*\)\s*;\s*$/) { |
Joe Perches | a6962d7 | 2013-04-29 16:18:13 -0700 | [diff] [blame] | 5468 | my $fmt = get_quoted_string($line, $rawline); |
Heba Aamer | caac1d5 | 2015-02-13 14:38:49 -0800 | [diff] [blame] | 5469 | $fmt =~ s/%%//g; |
| 5470 | if ($fmt !~ /%/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5471 | if (WARN("PREFER_SEQ_PUTS", |
| 5472 | "Prefer seq_puts to seq_printf\n" . $herecurr) && |
| 5473 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5474 | $fixed[$fixlinenr] =~ s/\bseq_printf\b/seq_puts/; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5475 | } |
Joe Perches | a6962d7 | 2013-04-29 16:18:13 -0700 | [diff] [blame] | 5476 | } |
| 5477 | } |
| 5478 | |
Andy Whitcroft | 554e165 | 2012-01-10 15:09:57 -0800 | [diff] [blame] | 5479 | # Check for misused memsets |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5480 | if ($^V && $^V ge 5.10.0 && |
| 5481 | defined $stat && |
Mateusz Kulikowski | 9e20a85 | 2015-06-25 15:03:16 -0700 | [diff] [blame] | 5482 | $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*$FuncArg\s*\)/) { |
Andy Whitcroft | 554e165 | 2012-01-10 15:09:57 -0800 | [diff] [blame] | 5483 | |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5484 | my $ms_addr = $2; |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5485 | my $ms_val = $7; |
| 5486 | my $ms_size = $12; |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5487 | |
Andy Whitcroft | 554e165 | 2012-01-10 15:09:57 -0800 | [diff] [blame] | 5488 | if ($ms_size =~ /^(0x|)0$/i) { |
| 5489 | ERROR("MEMSET", |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5490 | "memset to 0's uses 0 as the 2nd argument, not the 3rd\n" . "$here\n$stat\n"); |
Andy Whitcroft | 554e165 | 2012-01-10 15:09:57 -0800 | [diff] [blame] | 5491 | } elsif ($ms_size =~ /^(0x|)1$/i) { |
| 5492 | WARN("MEMSET", |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5493 | "single byte memset is suspicious. Swapped 2nd/3rd argument?\n" . "$here\n$stat\n"); |
| 5494 | } |
| 5495 | } |
| 5496 | |
Joe Perches | 98a9bba | 2014-01-23 15:54:52 -0800 | [diff] [blame] | 5497 | # Check for memcpy(foo, bar, ETH_ALEN) that could be ether_addr_copy(foo, bar) |
| 5498 | if ($^V && $^V ge 5.10.0 && |
Mateusz Kulikowski | 10895d2 | 2015-06-25 15:03:21 -0700 | [diff] [blame] | 5499 | defined $stat && |
| 5500 | $stat =~ /^\+(?:.*?)\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { |
Joe Perches | 98a9bba | 2014-01-23 15:54:52 -0800 | [diff] [blame] | 5501 | if (WARN("PREFER_ETHER_ADDR_COPY", |
Mateusz Kulikowski | 10895d2 | 2015-06-25 15:03:21 -0700 | [diff] [blame] | 5502 | "Prefer ether_addr_copy() over memcpy() if the Ethernet addresses are __aligned(2)\n" . "$here\n$stat\n") && |
Joe Perches | 98a9bba | 2014-01-23 15:54:52 -0800 | [diff] [blame] | 5503 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5504 | $fixed[$fixlinenr] =~ s/\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/ether_addr_copy($2, $7)/; |
Joe Perches | 98a9bba | 2014-01-23 15:54:52 -0800 | [diff] [blame] | 5505 | } |
| 5506 | } |
| 5507 | |
Mateusz Kulikowski | b6117d1 | 2015-06-25 15:03:13 -0700 | [diff] [blame] | 5508 | # Check for memcmp(foo, bar, ETH_ALEN) that could be ether_addr_equal*(foo, bar) |
| 5509 | if ($^V && $^V ge 5.10.0 && |
| 5510 | defined $stat && |
| 5511 | $stat =~ /^\+(?:.*?)\bmemcmp\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { |
| 5512 | WARN("PREFER_ETHER_ADDR_EQUAL", |
| 5513 | "Prefer ether_addr_equal() or ether_addr_equal_unaligned() over memcmp()\n" . "$here\n$stat\n") |
| 5514 | } |
| 5515 | |
Mateusz Kulikowski | 8617cd0 | 2015-06-25 15:03:19 -0700 | [diff] [blame] | 5516 | # check for memset(foo, 0x0, ETH_ALEN) that could be eth_zero_addr |
| 5517 | # check for memset(foo, 0xFF, ETH_ALEN) that could be eth_broadcast_addr |
| 5518 | if ($^V && $^V ge 5.10.0 && |
| 5519 | defined $stat && |
| 5520 | $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) { |
| 5521 | |
| 5522 | my $ms_val = $7; |
| 5523 | |
| 5524 | if ($ms_val =~ /^(?:0x|)0+$/i) { |
| 5525 | if (WARN("PREFER_ETH_ZERO_ADDR", |
| 5526 | "Prefer eth_zero_addr over memset()\n" . "$here\n$stat\n") && |
| 5527 | $fix) { |
| 5528 | $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_zero_addr($2)/; |
| 5529 | } |
| 5530 | } elsif ($ms_val =~ /^(?:0xff|255)$/i) { |
| 5531 | if (WARN("PREFER_ETH_BROADCAST_ADDR", |
| 5532 | "Prefer eth_broadcast_addr() over memset()\n" . "$here\n$stat\n") && |
| 5533 | $fix) { |
| 5534 | $fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_broadcast_addr($2)/; |
| 5535 | } |
| 5536 | } |
| 5537 | } |
| 5538 | |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5539 | # typecasts on min/max could be min_t/max_t |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5540 | if ($^V && $^V ge 5.10.0 && |
| 5541 | defined $stat && |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5542 | $stat =~ /^\+(?:.*?)\b(min|max)\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\)/) { |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5543 | if (defined $2 || defined $7) { |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5544 | my $call = $1; |
| 5545 | my $cast1 = deparenthesize($2); |
| 5546 | my $arg1 = $3; |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5547 | my $cast2 = deparenthesize($7); |
| 5548 | my $arg2 = $8; |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5549 | my $cast; |
| 5550 | |
Joe Perches | d1fe9c0 | 2012-03-23 15:02:16 -0700 | [diff] [blame] | 5551 | if ($cast1 ne "" && $cast2 ne "" && $cast1 ne $cast2) { |
Joe Perches | d7c76ba | 2012-01-10 15:09:58 -0800 | [diff] [blame] | 5552 | $cast = "$cast1 or $cast2"; |
| 5553 | } elsif ($cast1 ne "") { |
| 5554 | $cast = $cast1; |
| 5555 | } else { |
| 5556 | $cast = $cast2; |
| 5557 | } |
| 5558 | WARN("MINMAX", |
| 5559 | "$call() should probably be ${call}_t($cast, $arg1, $arg2)\n" . "$here\n$stat\n"); |
Andy Whitcroft | 554e165 | 2012-01-10 15:09:57 -0800 | [diff] [blame] | 5560 | } |
| 5561 | } |
| 5562 | |
Joe Perches | 4a27319 | 2012-07-30 14:41:20 -0700 | [diff] [blame] | 5563 | # check usleep_range arguments |
| 5564 | if ($^V && $^V ge 5.10.0 && |
| 5565 | defined $stat && |
| 5566 | $stat =~ /^\+(?:.*?)\busleep_range\s*\(\s*($FuncArg)\s*,\s*($FuncArg)\s*\)/) { |
| 5567 | my $min = $1; |
| 5568 | my $max = $7; |
| 5569 | if ($min eq $max) { |
| 5570 | WARN("USLEEP_RANGE", |
| 5571 | "usleep_range should not use min == max args; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); |
| 5572 | } elsif ($min =~ /^\d+$/ && $max =~ /^\d+$/ && |
| 5573 | $min > $max) { |
| 5574 | WARN("USLEEP_RANGE", |
| 5575 | "usleep_range args reversed, use min then max; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n"); |
| 5576 | } |
| 5577 | } |
| 5578 | |
Joe Perches | 823b794 | 2013-11-12 15:10:15 -0800 | [diff] [blame] | 5579 | # check for naked sscanf |
| 5580 | if ($^V && $^V ge 5.10.0 && |
| 5581 | defined $stat && |
Joe Perches | 6c8bd70 | 2014-04-03 14:49:16 -0700 | [diff] [blame] | 5582 | $line =~ /\bsscanf\b/ && |
Joe Perches | 823b794 | 2013-11-12 15:10:15 -0800 | [diff] [blame] | 5583 | ($stat !~ /$Ident\s*=\s*sscanf\s*$balanced_parens/ && |
| 5584 | $stat !~ /\bsscanf\s*$balanced_parens\s*(?:$Compare)/ && |
| 5585 | $stat !~ /(?:$Compare)\s*\bsscanf\s*$balanced_parens/)) { |
| 5586 | my $lc = $stat =~ tr@\n@@; |
| 5587 | $lc = $lc + $linenr; |
| 5588 | my $stat_real = raw_line($linenr, 0); |
| 5589 | for (my $count = $linenr + 1; $count <= $lc; $count++) { |
| 5590 | $stat_real = $stat_real . "\n" . raw_line($count, 0); |
| 5591 | } |
| 5592 | WARN("NAKED_SSCANF", |
| 5593 | "unchecked sscanf return value\n" . "$here\n$stat_real\n"); |
| 5594 | } |
| 5595 | |
Joe Perches | afc819a | 2014-06-04 16:12:08 -0700 | [diff] [blame] | 5596 | # check for simple sscanf that should be kstrto<foo> |
| 5597 | if ($^V && $^V ge 5.10.0 && |
| 5598 | defined $stat && |
| 5599 | $line =~ /\bsscanf\b/) { |
| 5600 | my $lc = $stat =~ tr@\n@@; |
| 5601 | $lc = $lc + $linenr; |
| 5602 | my $stat_real = raw_line($linenr, 0); |
| 5603 | for (my $count = $linenr + 1; $count <= $lc; $count++) { |
| 5604 | $stat_real = $stat_real . "\n" . raw_line($count, 0); |
| 5605 | } |
| 5606 | if ($stat_real =~ /\bsscanf\b\s*\(\s*$FuncArg\s*,\s*("[^"]+")/) { |
| 5607 | my $format = $6; |
| 5608 | my $count = $format =~ tr@%@%@; |
| 5609 | if ($count == 1 && |
| 5610 | $format =~ /^"\%(?i:ll[udxi]|[udxi]ll|ll|[hl]h?[udxi]|[udxi][hl]h?|[hl]h?|[udxi])"$/) { |
| 5611 | WARN("SSCANF_TO_KSTRTO", |
| 5612 | "Prefer kstrto<type> to single variable sscanf\n" . "$here\n$stat_real\n"); |
| 5613 | } |
| 5614 | } |
| 5615 | } |
| 5616 | |
Joe Perches | 70dc8a4 | 2013-09-11 14:23:58 -0700 | [diff] [blame] | 5617 | # check for new externs in .h files. |
| 5618 | if ($realfile =~ /\.h$/ && |
| 5619 | $line =~ /^\+\s*(extern\s+)$Type\s*$Ident\s*\(/s) { |
Joe Perches | d1d8578 | 2013-09-24 15:27:46 -0700 | [diff] [blame] | 5620 | if (CHK("AVOID_EXTERNS", |
| 5621 | "extern prototypes should be avoided in .h files\n" . $herecurr) && |
Joe Perches | 70dc8a4 | 2013-09-11 14:23:58 -0700 | [diff] [blame] | 5622 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5623 | $fixed[$fixlinenr] =~ s/(.*)\bextern\b\s*(.*)/$1$2/; |
Joe Perches | 70dc8a4 | 2013-09-11 14:23:58 -0700 | [diff] [blame] | 5624 | } |
| 5625 | } |
| 5626 | |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 5627 | # check for new externs in .c files. |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 5628 | if ($realfile =~ /\.c$/ && defined $stat && |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5629 | $stat =~ /^.\s*(?:extern\s+)?$Type\s+($Ident)(\s*)\(/s) |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 5630 | { |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5631 | my $function_name = $1; |
| 5632 | my $paren_space = $2; |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 5633 | |
| 5634 | my $s = $stat; |
| 5635 | if (defined $cond) { |
| 5636 | substr($s, 0, length($cond), ''); |
| 5637 | } |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5638 | if ($s =~ /^\s*;/ && |
| 5639 | $function_name ne 'uninitialized_var') |
| 5640 | { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5641 | WARN("AVOID_EXTERNS", |
| 5642 | "externs should be avoided in .c files\n" . $herecurr); |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 5643 | } |
| 5644 | |
| 5645 | if ($paren_space =~ /\n/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5646 | WARN("FUNCTION_ARGUMENTS", |
| 5647 | "arguments for function declarations should follow identifier\n" . $herecurr); |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 5648 | } |
Andy Whitcroft | 9c9ba34 | 2008-04-29 00:59:33 -0700 | [diff] [blame] | 5649 | |
| 5650 | } elsif ($realfile =~ /\.c$/ && defined $stat && |
| 5651 | $stat =~ /^.\s*extern\s+/) |
| 5652 | { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5653 | WARN("AVOID_EXTERNS", |
| 5654 | "externs should be avoided in .c files\n" . $herecurr); |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 5655 | } |
| 5656 | |
| 5657 | # checks for new __setup's |
| 5658 | if ($rawline =~ /\b__setup\("([^"]*)"/) { |
| 5659 | my $name = $1; |
| 5660 | |
| 5661 | if (!grep(/$name/, @setup_docs)) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5662 | CHK("UNDOCUMENTED_SETUP", |
| 5663 | "__setup appears un-documented -- check Documentation/kernel-parameters.txt\n" . $herecurr); |
Andy Whitcroft | de7d4f0 | 2007-07-15 23:37:22 -0700 | [diff] [blame] | 5664 | } |
Andy Whitcroft | 653d487 | 2007-06-23 17:16:34 -0700 | [diff] [blame] | 5665 | } |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 5666 | |
| 5667 | # check for pointless casting of kmalloc return |
Joe Perches | caf2a54 | 2011-01-12 16:59:56 -0800 | [diff] [blame] | 5668 | if ($line =~ /\*\s*\)\s*[kv][czm]alloc(_node){0,1}\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5669 | WARN("UNNECESSARY_CASTS", |
| 5670 | "unnecessary cast may hide bugs, see http://c-faq.com/malloc/mallocnocast.html\n" . $herecurr); |
Andy Whitcroft | 9c0ca6f | 2007-10-16 23:29:38 -0700 | [diff] [blame] | 5671 | } |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 5672 | |
Joe Perches | a640d25 | 2013-07-03 15:05:21 -0700 | [diff] [blame] | 5673 | # alloc style |
| 5674 | # p = alloc(sizeof(struct foo), ...) should be p = alloc(sizeof(*p), ...) |
| 5675 | if ($^V && $^V ge 5.10.0 && |
| 5676 | $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*([kv][mz]alloc(?:_node)?)\s*\(\s*(sizeof\s*\(\s*struct\s+$Lval\s*\))/) { |
| 5677 | CHK("ALLOC_SIZEOF_STRUCT", |
| 5678 | "Prefer $3(sizeof(*$1)...) over $3($4...)\n" . $herecurr); |
| 5679 | } |
| 5680 | |
Joe Perches | 60a5536 | 2014-06-04 16:12:07 -0700 | [diff] [blame] | 5681 | # check for k[mz]alloc with multiplies that could be kmalloc_array/kcalloc |
| 5682 | if ($^V && $^V ge 5.10.0 && |
Joe Perches | e367455 | 2014-08-06 16:10:55 -0700 | [diff] [blame] | 5683 | $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)\s*,/) { |
Joe Perches | 60a5536 | 2014-06-04 16:12:07 -0700 | [diff] [blame] | 5684 | my $oldfunc = $3; |
| 5685 | my $a1 = $4; |
| 5686 | my $a2 = $10; |
| 5687 | my $newfunc = "kmalloc_array"; |
| 5688 | $newfunc = "kcalloc" if ($oldfunc eq "kzalloc"); |
Joe Perches | e367455 | 2014-08-06 16:10:55 -0700 | [diff] [blame] | 5689 | my $r1 = $a1; |
| 5690 | my $r2 = $a2; |
| 5691 | if ($a1 =~ /^sizeof\s*\S/) { |
| 5692 | $r1 = $a2; |
| 5693 | $r2 = $a1; |
| 5694 | } |
| 5695 | if ($r1 !~ /^sizeof\b/ && $r2 =~ /^sizeof\s*\S/ && |
| 5696 | !($r1 =~ /^$Constant$/ || $r1 =~ /^[A-Z_][A-Z0-9_]*$/)) { |
Joe Perches | 60a5536 | 2014-06-04 16:12:07 -0700 | [diff] [blame] | 5697 | if (WARN("ALLOC_WITH_MULTIPLY", |
| 5698 | "Prefer $newfunc over $oldfunc with multiply\n" . $herecurr) && |
| 5699 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5700 | $fixed[$fixlinenr] =~ s/\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)/$1 . ' = ' . "$newfunc(" . trim($r1) . ', ' . trim($r2)/e; |
Joe Perches | 60a5536 | 2014-06-04 16:12:07 -0700 | [diff] [blame] | 5701 | |
| 5702 | } |
| 5703 | } |
| 5704 | } |
| 5705 | |
Joe Perches | 972fdea | 2013-04-29 16:18:12 -0700 | [diff] [blame] | 5706 | # check for krealloc arg reuse |
| 5707 | if ($^V && $^V ge 5.10.0 && |
| 5708 | $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*krealloc\s*\(\s*\1\s*,/) { |
| 5709 | WARN("KREALLOC_ARG_REUSE", |
| 5710 | "Reusing the krealloc arg is almost always a bug\n" . $herecurr); |
| 5711 | } |
| 5712 | |
Joe Perches | 5ce59ae | 2013-02-21 16:44:18 -0800 | [diff] [blame] | 5713 | # check for alloc argument mismatch |
| 5714 | if ($line =~ /\b(kcalloc|kmalloc_array)\s*\(\s*sizeof\b/) { |
| 5715 | WARN("ALLOC_ARRAY_ARGS", |
| 5716 | "$1 uses number as first arg, sizeof is generally wrong\n" . $herecurr); |
| 5717 | } |
| 5718 | |
Joe Perches | caf2a54 | 2011-01-12 16:59:56 -0800 | [diff] [blame] | 5719 | # check for multiple semicolons |
| 5720 | if ($line =~ /;\s*;\s*$/) { |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5721 | if (WARN("ONE_SEMICOLON", |
| 5722 | "Statements terminations use 1 semicolon\n" . $herecurr) && |
| 5723 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5724 | $fixed[$fixlinenr] =~ s/(\s*;\s*){2,}$/;/g; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5725 | } |
Joe Perches | d1e2ad0 | 2012-12-17 16:02:01 -0800 | [diff] [blame] | 5726 | } |
| 5727 | |
Tomas Winkler | cec3aaa | 2016-08-02 14:04:39 -0700 | [diff] [blame] | 5728 | # check for #defines like: 1 << <digit> that could be BIT(digit), it is not exported to uapi |
| 5729 | if ($realfile !~ m@^include/uapi/@ && |
| 5730 | $line =~ /#\s*define\s+\w+\s+\(?\s*1\s*([ulUL]*)\s*\<\<\s*(?:\d+|$Ident)\s*\)?/) { |
Joe Perches | 0ab9019 | 2014-12-10 15:51:57 -0800 | [diff] [blame] | 5731 | my $ull = ""; |
| 5732 | $ull = "_ULL" if (defined($1) && $1 =~ /ll/i); |
| 5733 | if (CHK("BIT_MACRO", |
| 5734 | "Prefer using the BIT$ull macro\n" . $herecurr) && |
| 5735 | $fix) { |
| 5736 | $fixed[$fixlinenr] =~ s/\(?\s*1\s*[ulUL]*\s*<<\s*(\d+|$Ident)\s*\)?/BIT${ull}($1)/; |
| 5737 | } |
| 5738 | } |
| 5739 | |
Joe Perches | 2d63274 | 2016-05-20 17:04:00 -0700 | [diff] [blame] | 5740 | # check for #if defined CONFIG_<FOO> || defined CONFIG_<FOO>_MODULE |
| 5741 | if ($line =~ /^\+\s*#\s*if\s+defined(?:\s*\(?\s*|\s+)(CONFIG_[A-Z_]+)\s*\)?\s*\|\|\s*defined(?:\s*\(?\s*|\s+)\1_MODULE\s*\)?\s*$/) { |
| 5742 | my $config = $1; |
| 5743 | if (WARN("PREFER_IS_ENABLED", |
| 5744 | "Prefer IS_ENABLED(<FOO>) to CONFIG_<FOO> || CONFIG_<FOO>_MODULE\n" . $herecurr) && |
| 5745 | $fix) { |
| 5746 | $fixed[$fixlinenr] = "\+#if IS_ENABLED($config)"; |
| 5747 | } |
| 5748 | } |
| 5749 | |
Joe Perches | e81f239 | 2014-08-06 16:11:25 -0700 | [diff] [blame] | 5750 | # check for case / default statements not preceded by break/fallthrough/switch |
Joe Perches | c34c09a | 2014-01-23 15:54:43 -0800 | [diff] [blame] | 5751 | if ($line =~ /^.\s*(?:case\s+(?:$Ident|$Constant)\s*|default):/) { |
| 5752 | my $has_break = 0; |
| 5753 | my $has_statement = 0; |
| 5754 | my $count = 0; |
| 5755 | my $prevline = $linenr; |
Joe Perches | e81f239 | 2014-08-06 16:11:25 -0700 | [diff] [blame] | 5756 | while ($prevline > 1 && ($file || $count < 3) && !$has_break) { |
Joe Perches | c34c09a | 2014-01-23 15:54:43 -0800 | [diff] [blame] | 5757 | $prevline--; |
| 5758 | my $rline = $rawlines[$prevline - 1]; |
| 5759 | my $fline = $lines[$prevline - 1]; |
| 5760 | last if ($fline =~ /^\@\@/); |
| 5761 | next if ($fline =~ /^\-/); |
| 5762 | next if ($fline =~ /^.(?:\s*(?:case\s+(?:$Ident|$Constant)[\s$;]*|default):[\s$;]*)*$/); |
| 5763 | $has_break = 1 if ($rline =~ /fall[\s_-]*(through|thru)/i); |
| 5764 | next if ($fline =~ /^.[\s$;]*$/); |
| 5765 | $has_statement = 1; |
| 5766 | $count++; |
| 5767 | $has_break = 1 if ($fline =~ /\bswitch\b|\b(?:break\s*;[\s$;]*$|return\b|goto\b|continue\b)/); |
| 5768 | } |
| 5769 | if (!$has_break && $has_statement) { |
| 5770 | WARN("MISSING_BREAK", |
| 5771 | "Possible switch case/default not preceeded by break or fallthrough comment\n" . $herecurr); |
| 5772 | } |
| 5773 | } |
| 5774 | |
Joe Perches | d1e2ad0 | 2012-12-17 16:02:01 -0800 | [diff] [blame] | 5775 | # check for switch/default statements without a break; |
| 5776 | if ($^V && $^V ge 5.10.0 && |
| 5777 | defined $stat && |
| 5778 | $stat =~ /^\+[$;\s]*(?:case[$;\s]+\w+[$;\s]*:[$;\s]*|)*[$;\s]*\bdefault[$;\s]*:[$;\s]*;/g) { |
| 5779 | my $ctx = ''; |
| 5780 | my $herectx = $here . "\n"; |
| 5781 | my $cnt = statement_rawlines($stat); |
| 5782 | for (my $n = 0; $n < $cnt; $n++) { |
| 5783 | $herectx .= raw_line($linenr, $n) . "\n"; |
| 5784 | } |
| 5785 | WARN("DEFAULT_NO_BREAK", |
| 5786 | "switch default: should use break\n" . $herectx); |
Joe Perches | caf2a54 | 2011-01-12 16:59:56 -0800 | [diff] [blame] | 5787 | } |
| 5788 | |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 5789 | # check for gcc specific __FUNCTION__ |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5790 | if ($line =~ /\b__FUNCTION__\b/) { |
| 5791 | if (WARN("USE_FUNC", |
| 5792 | "__func__ should be used instead of gcc specific __FUNCTION__\n" . $herecurr) && |
| 5793 | $fix) { |
Joe Perches | 194f66f | 2014-08-06 16:11:03 -0700 | [diff] [blame] | 5794 | $fixed[$fixlinenr] =~ s/\b__FUNCTION__\b/__func__/g; |
Joe Perches | d5e616f | 2013-09-11 14:23:54 -0700 | [diff] [blame] | 5795 | } |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 5796 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 5797 | |
Joe Perches | 62ec818 | 2015-02-13 14:38:18 -0800 | [diff] [blame] | 5798 | # check for uses of __DATE__, __TIME__, __TIMESTAMP__ |
| 5799 | while ($line =~ /\b(__(?:DATE|TIME|TIMESTAMP)__)\b/g) { |
| 5800 | ERROR("DATE_TIME", |
| 5801 | "Use of the '$1' macro makes the build non-deterministic\n" . $herecurr); |
| 5802 | } |
| 5803 | |
Joe Perches | 2c92488 | 2012-03-23 15:02:20 -0700 | [diff] [blame] | 5804 | # check for use of yield() |
| 5805 | if ($line =~ /\byield\s*\(\s*\)/) { |
| 5806 | WARN("YIELD", |
| 5807 | "Using yield() is generally wrong. See yield() kernel-doc (sched/core.c)\n" . $herecurr); |
| 5808 | } |
| 5809 | |
Joe Perches | 179f8f4 | 2013-07-03 15:05:30 -0700 | [diff] [blame] | 5810 | # check for comparisons against true and false |
| 5811 | if ($line =~ /\+\s*(.*?)\b(true|false|$Lval)\s*(==|\!=)\s*(true|false|$Lval)\b(.*)$/i) { |
| 5812 | my $lead = $1; |
| 5813 | my $arg = $2; |
| 5814 | my $test = $3; |
| 5815 | my $otype = $4; |
| 5816 | my $trail = $5; |
| 5817 | my $op = "!"; |
| 5818 | |
| 5819 | ($arg, $otype) = ($otype, $arg) if ($arg =~ /^(?:true|false)$/i); |
| 5820 | |
| 5821 | my $type = lc($otype); |
| 5822 | if ($type =~ /^(?:true|false)$/) { |
| 5823 | if (("$test" eq "==" && "$type" eq "true") || |
| 5824 | ("$test" eq "!=" && "$type" eq "false")) { |
| 5825 | $op = ""; |
| 5826 | } |
| 5827 | |
| 5828 | CHK("BOOL_COMPARISON", |
| 5829 | "Using comparison to $otype is error prone\n" . $herecurr); |
| 5830 | |
| 5831 | ## maybe suggesting a correct construct would better |
| 5832 | ## "Using comparison to $otype is error prone. Perhaps use '${lead}${op}${arg}${trail}'\n" . $herecurr); |
| 5833 | |
| 5834 | } |
| 5835 | } |
| 5836 | |
Thomas Gleixner | 4882720b | 2010-09-07 14:34:01 +0000 | [diff] [blame] | 5837 | # check for semaphores initialized locked |
| 5838 | if ($line =~ /^.\s*sema_init.+,\W?0\W?\)/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5839 | WARN("CONSIDER_COMPLETION", |
| 5840 | "consider using a completion\n" . $herecurr); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 5841 | } |
Joe Perches | 6712d85 | 2012-03-23 15:02:20 -0700 | [diff] [blame] | 5842 | |
Joe Perches | 67d0a07 | 2011-10-31 17:13:10 -0700 | [diff] [blame] | 5843 | # recommend kstrto* over simple_strto* and strict_strto* |
| 5844 | if ($line =~ /\b((simple|strict)_(strto(l|ll|ul|ull)))\s*\(/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5845 | WARN("CONSIDER_KSTRTO", |
Joe Perches | 67d0a07 | 2011-10-31 17:13:10 -0700 | [diff] [blame] | 5846 | "$1 is obsolete, use k$3 instead\n" . $herecurr); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 5847 | } |
Joe Perches | 6712d85 | 2012-03-23 15:02:20 -0700 | [diff] [blame] | 5848 | |
Fabian Frederick | ae3ccc4 | 2014-06-04 16:12:10 -0700 | [diff] [blame] | 5849 | # check for __initcall(), use device_initcall() explicitly or more appropriate function please |
Michael Ellerman | f3db663 | 2008-07-23 21:28:57 -0700 | [diff] [blame] | 5850 | if ($line =~ /^.\s*__initcall\s*\(/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5851 | WARN("USE_DEVICE_INITCALL", |
Fabian Frederick | ae3ccc4 | 2014-06-04 16:12:10 -0700 | [diff] [blame] | 5852 | "please use device_initcall() or more appropriate function instead of __initcall() (see include/linux/init.h)\n" . $herecurr); |
Michael Ellerman | f3db663 | 2008-07-23 21:28:57 -0700 | [diff] [blame] | 5853 | } |
Joe Perches | 6712d85 | 2012-03-23 15:02:20 -0700 | [diff] [blame] | 5854 | |
Joe Perches | 0f3c5aa | 2015-02-13 14:39:05 -0800 | [diff] [blame] | 5855 | # check for various structs that are normally const (ops, kgdb, device_tree) |
| 5856 | my $const_structs = qr{ |
| 5857 | acpi_dock_ops| |
Emese Revfy | 7940484 | 2010-03-05 13:43:53 -0800 | [diff] [blame] | 5858 | address_space_operations| |
| 5859 | backlight_ops| |
| 5860 | block_device_operations| |
| 5861 | dentry_operations| |
| 5862 | dev_pm_ops| |
| 5863 | dma_map_ops| |
| 5864 | extent_io_ops| |
| 5865 | file_lock_operations| |
| 5866 | file_operations| |
| 5867 | hv_ops| |
| 5868 | ide_dma_ops| |
| 5869 | intel_dvo_dev_ops| |
| 5870 | item_operations| |
| 5871 | iwl_ops| |
| 5872 | kgdb_arch| |
| 5873 | kgdb_io| |
| 5874 | kset_uevent_ops| |
| 5875 | lock_manager_operations| |
| 5876 | microcode_ops| |
| 5877 | mtrr_ops| |
| 5878 | neigh_ops| |
| 5879 | nlmsvc_binding| |
Joe Perches | 0f3c5aa | 2015-02-13 14:39:05 -0800 | [diff] [blame] | 5880 | of_device_id| |
Emese Revfy | 7940484 | 2010-03-05 13:43:53 -0800 | [diff] [blame] | 5881 | pci_raw_ops| |
| 5882 | pipe_buf_operations| |
| 5883 | platform_hibernation_ops| |
| 5884 | platform_suspend_ops| |
| 5885 | proto_ops| |
| 5886 | rpc_pipe_ops| |
| 5887 | seq_operations| |
| 5888 | snd_ac97_build_ops| |
| 5889 | soc_pcmcia_socket_ops| |
| 5890 | stacktrace_ops| |
| 5891 | sysfs_ops| |
| 5892 | tty_operations| |
Joe Perches | 6d07d01b | 2015-04-16 12:44:33 -0700 | [diff] [blame] | 5893 | uart_ops| |
Emese Revfy | 7940484 | 2010-03-05 13:43:53 -0800 | [diff] [blame] | 5894 | usb_mon_operations| |
| 5895 | wd_ops}x; |
Andy Whitcroft | 6903ffb | 2009-01-15 13:51:07 -0800 | [diff] [blame] | 5896 | if ($line !~ /\bconst\b/ && |
Joe Perches | 0f3c5aa | 2015-02-13 14:39:05 -0800 | [diff] [blame] | 5897 | $line =~ /\bstruct\s+($const_structs)\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5898 | WARN("CONST_STRUCT", |
| 5899 | "struct $1 should normally be const\n" . |
Andy Whitcroft | 6903ffb | 2009-01-15 13:51:07 -0800 | [diff] [blame] | 5900 | $herecurr); |
Andy Whitcroft | 2b6db5c | 2009-01-06 14:41:29 -0800 | [diff] [blame] | 5901 | } |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 5902 | |
| 5903 | # use of NR_CPUS is usually wrong |
| 5904 | # ignore definitions of NR_CPUS and usage to define arrays as likely right |
| 5905 | if ($line =~ /\bNR_CPUS\b/ && |
Andy Whitcroft | c45dcab | 2008-06-05 22:46:01 -0700 | [diff] [blame] | 5906 | $line !~ /^.\s*\s*#\s*if\b.*\bNR_CPUS\b/ && |
| 5907 | $line !~ /^.\s*\s*#\s*define\b.*\bNR_CPUS\b/ && |
Andy Whitcroft | 171ae1a | 2008-04-29 00:59:32 -0700 | [diff] [blame] | 5908 | $line !~ /^.\s*$Declare\s.*\[[^\]]*NR_CPUS[^\]]*\]/ && |
| 5909 | $line !~ /\[[^\]]*\.\.\.[^\]]*NR_CPUS[^\]]*\]/ && |
| 5910 | $line !~ /\[[^\]]*NR_CPUS[^\]]*\.\.\.[^\]]*\]/) |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 5911 | { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5912 | WARN("NR_CPUS", |
| 5913 | "usage of NR_CPUS is often wrong - consider using cpu_possible(), num_possible_cpus(), for_each_possible_cpu(), etc\n" . $herecurr); |
Andy Whitcroft | 773647a | 2008-03-28 14:15:58 -0700 | [diff] [blame] | 5914 | } |
Andy Whitcroft | 9c9ba34 | 2008-04-29 00:59:33 -0700 | [diff] [blame] | 5915 | |
Joe Perches | 52ea850 | 2013-11-12 15:10:09 -0800 | [diff] [blame] | 5916 | # Use of __ARCH_HAS_<FOO> or ARCH_HAVE_<BAR> is wrong. |
| 5917 | if ($line =~ /\+\s*#\s*define\s+((?:__)?ARCH_(?:HAS|HAVE)\w*)\b/) { |
| 5918 | ERROR("DEFINE_ARCH_HAS", |
| 5919 | "#define of '$1' is wrong - use Kconfig variables or standard guards instead\n" . $herecurr); |
| 5920 | } |
| 5921 | |
Joe Perches | acd9362 | 2015-02-13 14:38:38 -0800 | [diff] [blame] | 5922 | # likely/unlikely comparisons similar to "(likely(foo) > 0)" |
| 5923 | if ($^V && $^V ge 5.10.0 && |
| 5924 | $line =~ /\b((?:un)?likely)\s*\(\s*$FuncArg\s*\)\s*$Compare/) { |
| 5925 | WARN("LIKELY_MISUSE", |
| 5926 | "Using $1 should generally have parentheses around the comparison\n" . $herecurr); |
| 5927 | } |
| 5928 | |
Andy Whitcroft | 691d77b | 2009-01-06 14:41:16 -0800 | [diff] [blame] | 5929 | # whine mightly about in_atomic |
| 5930 | if ($line =~ /\bin_atomic\s*\(/) { |
| 5931 | if ($realfile =~ m@^drivers/@) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5932 | ERROR("IN_ATOMIC", |
| 5933 | "do not use in_atomic in drivers\n" . $herecurr); |
Andy Whitcroft | f4a8773 | 2009-02-27 14:03:05 -0800 | [diff] [blame] | 5934 | } elsif ($realfile !~ m@^kernel/@) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5935 | WARN("IN_ATOMIC", |
| 5936 | "use of in_atomic() is incorrect outside core kernel code\n" . $herecurr); |
Andy Whitcroft | 691d77b | 2009-01-06 14:41:16 -0800 | [diff] [blame] | 5937 | } |
| 5938 | } |
Peter Zijlstra | 1704f47 | 2010-03-19 01:37:42 +0100 | [diff] [blame] | 5939 | |
Joe Perches | 481aea5 | 2016-05-20 17:04:08 -0700 | [diff] [blame] | 5940 | # whine about ACCESS_ONCE |
| 5941 | if ($^V && $^V ge 5.10.0 && |
| 5942 | $line =~ /\bACCESS_ONCE\s*$balanced_parens\s*(=(?!=))?\s*($FuncArg)?/) { |
| 5943 | my $par = $1; |
| 5944 | my $eq = $2; |
| 5945 | my $fun = $3; |
| 5946 | $par =~ s/^\(\s*(.*)\s*\)$/$1/; |
| 5947 | if (defined($eq)) { |
| 5948 | if (WARN("PREFER_WRITE_ONCE", |
| 5949 | "Prefer WRITE_ONCE(<FOO>, <BAR>) over ACCESS_ONCE(<FOO>) = <BAR>\n" . $herecurr) && |
| 5950 | $fix) { |
| 5951 | $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)\s*$eq\s*\Q$fun\E/WRITE_ONCE($par, $fun)/; |
| 5952 | } |
| 5953 | } else { |
| 5954 | if (WARN("PREFER_READ_ONCE", |
| 5955 | "Prefer READ_ONCE(<FOO>) over ACCESS_ONCE(<FOO>)\n" . $herecurr) && |
| 5956 | $fix) { |
| 5957 | $fixed[$fixlinenr] =~ s/\bACCESS_ONCE\s*\(\s*\Q$par\E\s*\)/READ_ONCE($par)/; |
| 5958 | } |
| 5959 | } |
| 5960 | } |
| 5961 | |
Peter Zijlstra | 1704f47 | 2010-03-19 01:37:42 +0100 | [diff] [blame] | 5962 | # check for lockdep_set_novalidate_class |
| 5963 | if ($line =~ /^.\s*lockdep_set_novalidate_class\s*\(/ || |
| 5964 | $line =~ /__lockdep_no_validate__\s*\)/ ) { |
| 5965 | if ($realfile !~ m@^kernel/lockdep@ && |
| 5966 | $realfile !~ m@^include/linux/lockdep@ && |
| 5967 | $realfile !~ m@^drivers/base/core@) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5968 | ERROR("LOCKDEP", |
| 5969 | "lockdep_no_validate class is reserved for device->mutex.\n" . $herecurr); |
Peter Zijlstra | 1704f47 | 2010-03-19 01:37:42 +0100 | [diff] [blame] | 5970 | } |
| 5971 | } |
Dave Jones | 88f8831 | 2011-01-12 16:59:59 -0800 | [diff] [blame] | 5972 | |
Joe Perches | b392c64 | 2015-04-16 12:44:16 -0700 | [diff] [blame] | 5973 | if ($line =~ /debugfs_create_\w+.*\b$mode_perms_world_writable\b/ || |
| 5974 | $line =~ /DEVICE_ATTR.*\b$mode_perms_world_writable\b/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 5975 | WARN("EXPORTED_WORLD_WRITABLE", |
| 5976 | "Exporting world writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr); |
Dave Jones | 88f8831 | 2011-01-12 16:59:59 -0800 | [diff] [blame] | 5977 | } |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 5978 | |
Joe Perches | 515a235 | 2014-04-03 14:49:24 -0700 | [diff] [blame] | 5979 | # Mode permission misuses where it seems decimal should be octal |
| 5980 | # This uses a shortcut match to avoid unnecessary uses of a slow foreach loop |
| 5981 | if ($^V && $^V ge 5.10.0 && |
| 5982 | $line =~ /$mode_perms_search/) { |
| 5983 | foreach my $entry (@mode_permission_funcs) { |
| 5984 | my $func = $entry->[0]; |
| 5985 | my $arg_pos = $entry->[1]; |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 5986 | |
Joe Perches | 515a235 | 2014-04-03 14:49:24 -0700 | [diff] [blame] | 5987 | my $skip_args = ""; |
| 5988 | if ($arg_pos > 1) { |
| 5989 | $arg_pos--; |
| 5990 | $skip_args = "(?:\\s*$FuncArg\\s*,\\s*){$arg_pos,$arg_pos}"; |
| 5991 | } |
| 5992 | my $test = "\\b$func\\s*\\(${skip_args}([\\d]+)\\s*[,\\)]"; |
| 5993 | if ($line =~ /$test/) { |
| 5994 | my $val = $1; |
| 5995 | $val = $6 if ($skip_args ne ""); |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 5996 | |
Joe Perches | 515a235 | 2014-04-03 14:49:24 -0700 | [diff] [blame] | 5997 | if ($val !~ /^0$/ && |
| 5998 | (($val =~ /^$Int$/ && $val !~ /^$Octal$/) || |
| 5999 | length($val) ne 4)) { |
| 6000 | ERROR("NON_OCTAL_PERMISSIONS", |
| 6001 | "Use 4 digit octal (0777) not decimal permissions\n" . $herecurr); |
Joe Perches | c0a5c89 | 2015-02-13 14:38:21 -0800 | [diff] [blame] | 6002 | } elsif ($val =~ /^$Octal$/ && (oct($val) & 02)) { |
| 6003 | ERROR("EXPORTED_WORLD_WRITABLE", |
| 6004 | "Exporting writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr); |
Joe Perches | 515a235 | 2014-04-03 14:49:24 -0700 | [diff] [blame] | 6005 | } |
Joe Perches | 2435880 | 2014-04-03 14:49:13 -0700 | [diff] [blame] | 6006 | } |
| 6007 | } |
| 6008 | } |
Bjorn Andersson | 5a6d20c | 2015-06-25 15:03:24 -0700 | [diff] [blame] | 6009 | |
| 6010 | # validate content of MODULE_LICENSE against list from include/linux/module.h |
| 6011 | if ($line =~ /\bMODULE_LICENSE\s*\(\s*($String)\s*\)/) { |
| 6012 | my $extracted_string = get_quoted_string($line, $rawline); |
| 6013 | my $valid_licenses = qr{ |
| 6014 | GPL| |
| 6015 | GPL\ v2| |
| 6016 | GPL\ and\ additional\ rights| |
| 6017 | Dual\ BSD/GPL| |
| 6018 | Dual\ MIT/GPL| |
| 6019 | Dual\ MPL/GPL| |
| 6020 | Proprietary |
| 6021 | }x; |
| 6022 | if ($extracted_string !~ /^"(?:$valid_licenses)"$/x) { |
| 6023 | WARN("MODULE_LICENSE", |
| 6024 | "unknown module license " . $extracted_string . "\n" . $herecurr); |
| 6025 | } |
| 6026 | } |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 6027 | } |
| 6028 | |
| 6029 | # If we have no input at all, then there is nothing to report on |
| 6030 | # so just keep quiet. |
| 6031 | if ($#rawlines == -1) { |
| 6032 | exit(0); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 6033 | } |
| 6034 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 6035 | # In mailback mode only produce a report in the negative, for |
| 6036 | # things that appear to be patches. |
| 6037 | if ($mailback && ($clean == 1 || !$is_patch)) { |
| 6038 | exit(0); |
| 6039 | } |
| 6040 | |
| 6041 | # This is not a patch, and we are are in 'no-patch' mode so |
| 6042 | # just keep quiet. |
| 6043 | if (!$chk_patch && !$is_patch) { |
| 6044 | exit(0); |
| 6045 | } |
| 6046 | |
Joe Perches | 06330fc | 2015-06-25 15:03:11 -0700 | [diff] [blame] | 6047 | if (!$is_patch && $file !~ /cover-letter\.patch$/) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 6048 | ERROR("NOT_UNIFIED_DIFF", |
| 6049 | "Does not appear to be a unified-diff format patch\n"); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 6050 | } |
Allen Hubbe | ed43c4e | 2016-08-02 14:04:45 -0700 | [diff] [blame] | 6051 | if ($is_patch && $has_commit_log && $chk_signoff && $signoff == 0) { |
Joe Perches | 000d1cc1 | 2011-07-25 17:13:25 -0700 | [diff] [blame] | 6052 | ERROR("MISSING_SIGN_OFF", |
| 6053 | "Missing Signed-off-by: line(s)\n"); |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 6054 | } |
| 6055 | |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 6056 | print report_dump(); |
Andy Whitcroft | 13214ad | 2008-02-08 04:22:03 -0800 | [diff] [blame] | 6057 | if ($summary && !($clean == 1 && $quiet == 1)) { |
| 6058 | print "$filename " if ($summary_file); |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 6059 | print "total: $cnt_error errors, $cnt_warn warnings, " . |
| 6060 | (($check)? "$cnt_chk checks, " : "") . |
| 6061 | "$cnt_lines lines checked\n"; |
Andy Whitcroft | f0a594c | 2007-07-19 01:48:34 -0700 | [diff] [blame] | 6062 | } |
Andy Whitcroft | 8905a67 | 2007-11-28 16:21:06 -0800 | [diff] [blame] | 6063 | |
Andy Whitcroft | d2c0a23 | 2010-10-26 14:23:12 -0700 | [diff] [blame] | 6064 | if ($quiet == 0) { |
Joe Perches | ef21219 | 2016-05-20 17:04:11 -0700 | [diff] [blame] | 6065 | # If there were any defects found and not already fixing them |
| 6066 | if (!$clean and !$fix) { |
| 6067 | print << "EOM" |
| 6068 | |
| 6069 | NOTE: For some of the reported defects, checkpatch may be able to |
| 6070 | mechanically convert to the typical style using --fix or --fix-inplace. |
| 6071 | EOM |
| 6072 | } |
Andy Whitcroft | d2c0a23 | 2010-10-26 14:23:12 -0700 | [diff] [blame] | 6073 | # If there were whitespace errors which cleanpatch can fix |
| 6074 | # then suggest that. |
| 6075 | if ($rpt_cleaners) { |
Mike Frysinger | b078121 | 2011-03-22 16:34:43 -0700 | [diff] [blame] | 6076 | $rpt_cleaners = 0; |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 6077 | print << "EOM" |
| 6078 | |
| 6079 | NOTE: Whitespace errors detected. |
| 6080 | You may wish to use scripts/cleanpatch or scripts/cleanfile |
| 6081 | EOM |
Andy Whitcroft | d2c0a23 | 2010-10-26 14:23:12 -0700 | [diff] [blame] | 6082 | } |
| 6083 | } |
| 6084 | |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 6085 | if ($clean == 0 && $fix && |
| 6086 | ("@rawlines" ne "@fixed" || |
| 6087 | $#fixed_inserted >= 0 || $#fixed_deleted >= 0)) { |
Joe Perches | 9624b8d | 2014-01-23 15:54:44 -0800 | [diff] [blame] | 6088 | my $newfile = $filename; |
| 6089 | $newfile .= ".EXPERIMENTAL-checkpatch-fixes" if (!$fix_inplace); |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 6090 | my $linecount = 0; |
| 6091 | my $f; |
| 6092 | |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 6093 | @fixed = fix_inserted_deleted_lines(\@fixed, \@fixed_inserted, \@fixed_deleted); |
| 6094 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 6095 | open($f, '>', $newfile) |
| 6096 | or die "$P: Can't open $newfile for write\n"; |
| 6097 | foreach my $fixed_line (@fixed) { |
| 6098 | $linecount++; |
| 6099 | if ($file) { |
| 6100 | if ($linecount > 3) { |
| 6101 | $fixed_line =~ s/^\+//; |
Joe Perches | d752fcc | 2014-08-06 16:11:05 -0700 | [diff] [blame] | 6102 | print $f $fixed_line . "\n"; |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 6103 | } |
| 6104 | } else { |
| 6105 | print $f $fixed_line . "\n"; |
| 6106 | } |
| 6107 | } |
| 6108 | close($f); |
| 6109 | |
| 6110 | if (!$quiet) { |
| 6111 | print << "EOM"; |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 6112 | |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 6113 | Wrote EXPERIMENTAL --fix correction(s) to '$newfile' |
| 6114 | |
| 6115 | Do _NOT_ trust the results written to this file. |
| 6116 | Do _NOT_ submit these changes without inspecting them for correctness. |
| 6117 | |
| 6118 | This EXPERIMENTAL file is simply a convenience to help rewrite patches. |
| 6119 | No warranties, expressed or implied... |
Joe Perches | 3705ce5 | 2013-07-03 15:05:31 -0700 | [diff] [blame] | 6120 | EOM |
| 6121 | } |
| 6122 | } |
| 6123 | |
Joe Perches | d8469f1 | 2015-06-25 15:03:00 -0700 | [diff] [blame] | 6124 | if ($quiet == 0) { |
| 6125 | print "\n"; |
| 6126 | if ($clean == 1) { |
| 6127 | print "$vname has no obvious style problems and is ready for submission.\n"; |
| 6128 | } else { |
| 6129 | print "$vname has style problems, please review.\n"; |
| 6130 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 6131 | } |
Andy Whitcroft | 0a920b5b | 2007-06-01 00:46:48 -0700 | [diff] [blame] | 6132 | return $clean; |
| 6133 | } |