xref: /linux-6.15/scripts/checkpatch.pl (revision fe043ea1)
10a920b5bSAndy Whitcroft#!/usr/bin/perl -w
2dbf004d7SDave Jones# (c) 2001, Dave Jones. (the file handling bit)
300df344fSAndy Whitcroft# (c) 2005, Joel Schopp <[email protected]> (the ugly bit)
42a5a2c25SAndy Whitcroft# (c) 2007,2008, Andy Whitcroft <[email protected]> (new conditions, test suite)
5015830beSAndy Whitcroft# (c) 2008-2010 Andy Whitcroft <[email protected]>
60a920b5bSAndy Whitcroft# Licensed under the terms of the GNU GPL License version 2
70a920b5bSAndy Whitcroft
80a920b5bSAndy Whitcroftuse strict;
9c707a81dSJoe Perchesuse POSIX;
1036061e38SJoe Perchesuse File::Basename;
1136061e38SJoe Perchesuse Cwd 'abs_path';
1257230297SJoe Perchesuse Term::ANSIColor qw(:constants);
130a920b5bSAndy Whitcroft
140a920b5bSAndy Whitcroftmy $P = $0;
1536061e38SJoe Perchesmy $D = dirname(abs_path($P));
160a920b5bSAndy Whitcroft
17000d1cc1SJoe Perchesmy $V = '0.32';
180a920b5bSAndy Whitcroft
190a920b5bSAndy Whitcroftuse Getopt::Long qw(:config no_auto_abbrev);
200a920b5bSAndy Whitcroft
210a920b5bSAndy Whitcroftmy $quiet = 0;
220a920b5bSAndy Whitcroftmy $tree = 1;
230a920b5bSAndy Whitcroftmy $chk_signoff = 1;
240a920b5bSAndy Whitcroftmy $chk_patch = 1;
25773647a0SAndy Whitcroftmy $tst_only;
266c72ffaaSAndy Whitcroftmy $emacs = 0;
278905a67cSAndy Whitcroftmy $terse = 0;
2834d8815fSJoe Perchesmy $showfile = 0;
296c72ffaaSAndy Whitcroftmy $file = 0;
306c72ffaaSAndy Whitcroftmy $check = 0;
312ac73b4fSJoe Perchesmy $check_orig = 0;
328905a67cSAndy Whitcroftmy $summary = 1;
338905a67cSAndy Whitcroftmy $mailback = 0;
3413214adfSAndy Whitcroftmy $summary_file = 0;
35000d1cc1SJoe Perchesmy $show_types = 0;
363705ce5bSJoe Perchesmy $fix = 0;
379624b8d6SJoe Perchesmy $fix_inplace = 0;
386c72ffaaSAndy Whitcroftmy $root;
39c2fdda0dSAndy Whitcroftmy %debug;
403445686aSJoe Perchesmy %camelcase = ();
4191bfe484SJoe Perchesmy %use_type = ();
4291bfe484SJoe Perchesmy @use = ();
4391bfe484SJoe Perchesmy %ignore_type = ();
44000d1cc1SJoe Perchesmy @ignore = ();
4577f5b10aSHannes Edermy $help = 0;
46000d1cc1SJoe Perchesmy $configuration_file = ".checkpatch.conf";
476cd7f386SJoe Perchesmy $max_line_length = 80;
48d62a201fSDave Hansenmy $ignore_perl_version = 0;
49d62a201fSDave Hansenmy $minimum_perl_version = 5.10.0;
5056193274SVadim Bendeburymy $min_conf_desc_length = 4;
5166b47b4aSKees Cookmy $spelling_file = "$D/spelling.txt";
52ebfd7d62SJoe Perchesmy $codespell = 0;
53f1a63678SMaxim Uvarovmy $codespellfile = "/usr/share/codespell/dictionary.txt";
5457230297SJoe Perchesmy $color = 1;
5577f5b10aSHannes Eder
5677f5b10aSHannes Edersub help {
5777f5b10aSHannes Eder	my ($exitcode) = @_;
5877f5b10aSHannes Eder
5977f5b10aSHannes Eder	print << "EOM";
6077f5b10aSHannes EderUsage: $P [OPTION]... [FILE]...
6177f5b10aSHannes EderVersion: $V
6277f5b10aSHannes Eder
6377f5b10aSHannes EderOptions:
6477f5b10aSHannes Eder  -q, --quiet                quiet
6577f5b10aSHannes Eder  --no-tree                  run without a kernel tree
6677f5b10aSHannes Eder  --no-signoff               do not check for 'Signed-off-by' line
6777f5b10aSHannes Eder  --patch                    treat FILE as patchfile (default)
6877f5b10aSHannes Eder  --emacs                    emacs compile window format
6977f5b10aSHannes Eder  --terse                    one line per report
7034d8815fSJoe Perches  --showfile                 emit diffed file position, not input file position
7177f5b10aSHannes Eder  -f, --file                 treat FILE as regular source file
7277f5b10aSHannes Eder  --subjective, --strict     enable more subjective tests
7391bfe484SJoe Perches  --types TYPE(,TYPE2...)    show only these comma separated message types
74000d1cc1SJoe Perches  --ignore TYPE(,TYPE2...)   ignore various comma separated message types
756cd7f386SJoe Perches  --max-line-length=n        set the maximum line length, if exceeded, warn
7656193274SVadim Bendebury  --min-conf-desc-length=n   set the min description length, if shorter, warn
77000d1cc1SJoe Perches  --show-types               show the message "types" in the output
7877f5b10aSHannes Eder  --root=PATH                PATH to the kernel tree root
7977f5b10aSHannes Eder  --no-summary               suppress the per-file summary
8077f5b10aSHannes Eder  --mailback                 only produce a report in case of warnings/errors
8177f5b10aSHannes Eder  --summary-file             include the filename in summary
8277f5b10aSHannes Eder  --debug KEY=[0|1]          turn on/off debugging of KEY, where KEY is one of
8377f5b10aSHannes Eder                             'values', 'possible', 'type', and 'attr' (default
8477f5b10aSHannes Eder                             is all off)
8577f5b10aSHannes Eder  --test-only=WORD           report only warnings/errors containing WORD
8677f5b10aSHannes Eder                             literally
873705ce5bSJoe Perches  --fix                      EXPERIMENTAL - may create horrible results
883705ce5bSJoe Perches                             If correctable single-line errors exist, create
893705ce5bSJoe Perches                             "<inputfile>.EXPERIMENTAL-checkpatch-fixes"
903705ce5bSJoe Perches                             with potential errors corrected to the preferred
913705ce5bSJoe Perches                             checkpatch style
929624b8d6SJoe Perches  --fix-inplace              EXPERIMENTAL - may create horrible results
939624b8d6SJoe Perches                             Is the same as --fix, but overwrites the input
949624b8d6SJoe Perches                             file.  It's your fault if there's no backup or git
95d62a201fSDave Hansen  --ignore-perl-version      override checking of perl version.  expect
96d62a201fSDave Hansen                             runtime errors.
97ebfd7d62SJoe Perches  --codespell                Use the codespell dictionary for spelling/typos
98f1a63678SMaxim Uvarov                             (default:/usr/share/codespell/dictionary.txt)
99ebfd7d62SJoe Perches  --codespellfile            Use this codespell dictionary
10057230297SJoe Perches  --color                    Use colors when output is STDOUT (default: on)
10177f5b10aSHannes Eder  -h, --help, --version      display this help and exit
10277f5b10aSHannes Eder
10377f5b10aSHannes EderWhen FILE is - read standard input.
10477f5b10aSHannes EderEOM
10577f5b10aSHannes Eder
10677f5b10aSHannes Eder	exit($exitcode);
10777f5b10aSHannes Eder}
10877f5b10aSHannes Eder
109000d1cc1SJoe Perchesmy $conf = which_conf($configuration_file);
110000d1cc1SJoe Perchesif (-f $conf) {
111000d1cc1SJoe Perches	my @conf_args;
112000d1cc1SJoe Perches	open(my $conffile, '<', "$conf")
113000d1cc1SJoe Perches	    or warn "$P: Can't find a readable $configuration_file file $!\n";
114000d1cc1SJoe Perches
115000d1cc1SJoe Perches	while (<$conffile>) {
116000d1cc1SJoe Perches		my $line = $_;
117000d1cc1SJoe Perches
118000d1cc1SJoe Perches		$line =~ s/\s*\n?$//g;
119000d1cc1SJoe Perches		$line =~ s/^\s*//g;
120000d1cc1SJoe Perches		$line =~ s/\s+/ /g;
121000d1cc1SJoe Perches
122000d1cc1SJoe Perches		next if ($line =~ m/^\s*#/);
123000d1cc1SJoe Perches		next if ($line =~ m/^\s*$/);
124000d1cc1SJoe Perches
125000d1cc1SJoe Perches		my @words = split(" ", $line);
126000d1cc1SJoe Perches		foreach my $word (@words) {
127000d1cc1SJoe Perches			last if ($word =~ m/^#/);
128000d1cc1SJoe Perches			push (@conf_args, $word);
129000d1cc1SJoe Perches		}
130000d1cc1SJoe Perches	}
131000d1cc1SJoe Perches	close($conffile);
132000d1cc1SJoe Perches	unshift(@ARGV, @conf_args) if @conf_args;
133000d1cc1SJoe Perches}
134000d1cc1SJoe Perches
1350a920b5bSAndy WhitcroftGetOptions(
1366c72ffaaSAndy Whitcroft	'q|quiet+'	=> \$quiet,
1370a920b5bSAndy Whitcroft	'tree!'		=> \$tree,
1380a920b5bSAndy Whitcroft	'signoff!'	=> \$chk_signoff,
1390a920b5bSAndy Whitcroft	'patch!'	=> \$chk_patch,
1406c72ffaaSAndy Whitcroft	'emacs!'	=> \$emacs,
1418905a67cSAndy Whitcroft	'terse!'	=> \$terse,
14234d8815fSJoe Perches	'showfile!'	=> \$showfile,
14377f5b10aSHannes Eder	'f|file!'	=> \$file,
1446c72ffaaSAndy Whitcroft	'subjective!'	=> \$check,
1456c72ffaaSAndy Whitcroft	'strict!'	=> \$check,
146000d1cc1SJoe Perches	'ignore=s'	=> \@ignore,
14791bfe484SJoe Perches	'types=s'	=> \@use,
148000d1cc1SJoe Perches	'show-types!'	=> \$show_types,
1496cd7f386SJoe Perches	'max-line-length=i' => \$max_line_length,
15056193274SVadim Bendebury	'min-conf-desc-length=i' => \$min_conf_desc_length,
1516c72ffaaSAndy Whitcroft	'root=s'	=> \$root,
1528905a67cSAndy Whitcroft	'summary!'	=> \$summary,
1538905a67cSAndy Whitcroft	'mailback!'	=> \$mailback,
15413214adfSAndy Whitcroft	'summary-file!'	=> \$summary_file,
1553705ce5bSJoe Perches	'fix!'		=> \$fix,
1569624b8d6SJoe Perches	'fix-inplace!'	=> \$fix_inplace,
157d62a201fSDave Hansen	'ignore-perl-version!' => \$ignore_perl_version,
158c2fdda0dSAndy Whitcroft	'debug=s'	=> \%debug,
159773647a0SAndy Whitcroft	'test-only=s'	=> \$tst_only,
160ebfd7d62SJoe Perches	'codespell!'	=> \$codespell,
161ebfd7d62SJoe Perches	'codespellfile=s'	=> \$codespellfile,
16257230297SJoe Perches	'color!'	=> \$color,
16377f5b10aSHannes Eder	'h|help'	=> \$help,
16477f5b10aSHannes Eder	'version'	=> \$help
16577f5b10aSHannes Eder) or help(1);
16677f5b10aSHannes Eder
16777f5b10aSHannes Ederhelp(0) if ($help);
1680a920b5bSAndy Whitcroft
1699624b8d6SJoe Perches$fix = 1 if ($fix_inplace);
1702ac73b4fSJoe Perches$check_orig = $check;
1719624b8d6SJoe Perches
1720a920b5bSAndy Whitcroftmy $exit = 0;
1730a920b5bSAndy Whitcroft
174d62a201fSDave Hansenif ($^V && $^V lt $minimum_perl_version) {
175d62a201fSDave Hansen	printf "$P: requires at least perl version %vd\n", $minimum_perl_version;
176d62a201fSDave Hansen	if (!$ignore_perl_version) {
177d62a201fSDave Hansen		exit(1);
178d62a201fSDave Hansen	}
179d62a201fSDave Hansen}
180d62a201fSDave Hansen
1810a920b5bSAndy Whitcroftif ($#ARGV < 0) {
18277f5b10aSHannes Eder	print "$P: no input files\n";
1830a920b5bSAndy Whitcroft	exit(1);
1840a920b5bSAndy Whitcroft}
1850a920b5bSAndy Whitcroft
18691bfe484SJoe Perchessub hash_save_array_words {
18791bfe484SJoe Perches	my ($hashRef, $arrayRef) = @_;
18891bfe484SJoe Perches
18991bfe484SJoe Perches	my @array = split(/,/, join(',', @$arrayRef));
19091bfe484SJoe Perches	foreach my $word (@array) {
191000d1cc1SJoe Perches		$word =~ s/\s*\n?$//g;
192000d1cc1SJoe Perches		$word =~ s/^\s*//g;
193000d1cc1SJoe Perches		$word =~ s/\s+/ /g;
194000d1cc1SJoe Perches		$word =~ tr/[a-z]/[A-Z]/;
195000d1cc1SJoe Perches
196000d1cc1SJoe Perches		next if ($word =~ m/^\s*#/);
197000d1cc1SJoe Perches		next if ($word =~ m/^\s*$/);
198000d1cc1SJoe Perches
19991bfe484SJoe Perches		$hashRef->{$word}++;
200000d1cc1SJoe Perches	}
20191bfe484SJoe Perches}
20291bfe484SJoe Perches
20391bfe484SJoe Perchessub hash_show_words {
20491bfe484SJoe Perches	my ($hashRef, $prefix) = @_;
20591bfe484SJoe Perches
2063c816e49SJoe Perches	if (keys %$hashRef) {
207d8469f16SJoe Perches		print "\nNOTE: $prefix message types:";
20858cb3cf6SJoe Perches		foreach my $word (sort keys %$hashRef) {
20991bfe484SJoe Perches			print " $word";
21091bfe484SJoe Perches		}
211d8469f16SJoe Perches		print "\n";
21291bfe484SJoe Perches	}
21391bfe484SJoe Perches}
21491bfe484SJoe Perches
21591bfe484SJoe Percheshash_save_array_words(\%ignore_type, \@ignore);
21691bfe484SJoe Percheshash_save_array_words(\%use_type, \@use);
217000d1cc1SJoe Perches
218c2fdda0dSAndy Whitcroftmy $dbg_values = 0;
219c2fdda0dSAndy Whitcroftmy $dbg_possible = 0;
2207429c690SAndy Whitcroftmy $dbg_type = 0;
221a1ef277eSAndy Whitcroftmy $dbg_attr = 0;
222c2fdda0dSAndy Whitcroftfor my $key (keys %debug) {
22321caa13cSAndy Whitcroft	## no critic
22421caa13cSAndy Whitcroft	eval "\${dbg_$key} = '$debug{$key}';";
22521caa13cSAndy Whitcroft	die "$@" if ($@);
226c2fdda0dSAndy Whitcroft}
227c2fdda0dSAndy Whitcroft
228d2c0a235SAndy Whitcroftmy $rpt_cleaners = 0;
229d2c0a235SAndy Whitcroft
2308905a67cSAndy Whitcroftif ($terse) {
2318905a67cSAndy Whitcroft	$emacs = 1;
2328905a67cSAndy Whitcroft	$quiet++;
2338905a67cSAndy Whitcroft}
2348905a67cSAndy Whitcroft
2356c72ffaaSAndy Whitcroftif ($tree) {
2366c72ffaaSAndy Whitcroft	if (defined $root) {
2376c72ffaaSAndy Whitcroft		if (!top_of_kernel_tree($root)) {
2386c72ffaaSAndy Whitcroft			die "$P: $root: --root does not point at a valid tree\n";
2396c72ffaaSAndy Whitcroft		}
2406c72ffaaSAndy Whitcroft	} else {
2416c72ffaaSAndy Whitcroft		if (top_of_kernel_tree('.')) {
2426c72ffaaSAndy Whitcroft			$root = '.';
2436c72ffaaSAndy Whitcroft		} elsif ($0 =~ m@(.*)/scripts/[^/]*$@ &&
2446c72ffaaSAndy Whitcroft						top_of_kernel_tree($1)) {
2456c72ffaaSAndy Whitcroft			$root = $1;
2466c72ffaaSAndy Whitcroft		}
2476c72ffaaSAndy Whitcroft	}
2486c72ffaaSAndy Whitcroft
2496c72ffaaSAndy Whitcroft	if (!defined $root) {
2500a920b5bSAndy Whitcroft		print "Must be run from the top-level dir. of a kernel tree\n";
2510a920b5bSAndy Whitcroft		exit(2);
2520a920b5bSAndy Whitcroft	}
2536c72ffaaSAndy Whitcroft}
2546c72ffaaSAndy Whitcroft
2556c72ffaaSAndy Whitcroftmy $emitted_corrupt = 0;
2566c72ffaaSAndy Whitcroft
2572ceb532bSAndy Whitcroftour $Ident	= qr{
2582ceb532bSAndy Whitcroft			[A-Za-z_][A-Za-z\d_]*
2592ceb532bSAndy Whitcroft			(?:\s*\#\#\s*[A-Za-z_][A-Za-z\d_]*)*
2602ceb532bSAndy Whitcroft		}x;
2616c72ffaaSAndy Whitcroftour $Storage	= qr{extern|static|asmlinkage};
2626c72ffaaSAndy Whitcroftour $Sparse	= qr{
2636c72ffaaSAndy Whitcroft			__user|
2646c72ffaaSAndy Whitcroft			__kernel|
2656c72ffaaSAndy Whitcroft			__force|
2666c72ffaaSAndy Whitcroft			__iomem|
2676c72ffaaSAndy Whitcroft			__must_check|
2686c72ffaaSAndy Whitcroft			__init_refok|
269417495edSAndy Whitcroft			__kprobes|
270165e72a6SSven Eckelmann			__ref|
271165e72a6SSven Eckelmann			__rcu
2726c72ffaaSAndy Whitcroft		}x;
273e970b884SJoe Perchesour $InitAttributePrefix = qr{__(?:mem|cpu|dev|net_|)};
274e970b884SJoe Perchesour $InitAttributeData = qr{$InitAttributePrefix(?:initdata\b)};
275e970b884SJoe Perchesour $InitAttributeConst = qr{$InitAttributePrefix(?:initconst\b)};
276e970b884SJoe Perchesour $InitAttributeInit = qr{$InitAttributePrefix(?:init\b)};
277e970b884SJoe Perchesour $InitAttribute = qr{$InitAttributeData|$InitAttributeConst|$InitAttributeInit};
2788716de38SJoe Perches
27952131292SWolfram Sang# Notes to $Attribute:
28052131292SWolfram Sang# We need \b after 'init' otherwise 'initconst' will cause a false positive in a check
2816c72ffaaSAndy Whitcroftour $Attribute	= qr{
2826c72ffaaSAndy Whitcroft			const|
28303f1df7dSJoe Perches			__percpu|
28403f1df7dSJoe Perches			__nocast|
28503f1df7dSJoe Perches			__safe|
28603f1df7dSJoe Perches			__bitwise__|
28703f1df7dSJoe Perches			__packed__|
28803f1df7dSJoe Perches			__packed2__|
28903f1df7dSJoe Perches			__naked|
29003f1df7dSJoe Perches			__maybe_unused|
29103f1df7dSJoe Perches			__always_unused|
29203f1df7dSJoe Perches			__noreturn|
29303f1df7dSJoe Perches			__used|
29403f1df7dSJoe Perches			__cold|
295e23ef1f3SJoe Perches			__pure|
29603f1df7dSJoe Perches			__noclone|
29703f1df7dSJoe Perches			__deprecated|
2986c72ffaaSAndy Whitcroft			__read_mostly|
2996c72ffaaSAndy Whitcroft			__kprobes|
3008716de38SJoe Perches			$InitAttribute|
30124e1d81aSAndy Whitcroft			____cacheline_aligned|
30224e1d81aSAndy Whitcroft			____cacheline_aligned_in_smp|
3035fe3af11SAndy Whitcroft			____cacheline_internodealigned_in_smp|
3045fe3af11SAndy Whitcroft			__weak
3056c72ffaaSAndy Whitcroft		  }x;
306c45dcabdSAndy Whitcroftour $Modifier;
30791cb5195SJoe Perchesour $Inline	= qr{inline|__always_inline|noinline|__inline|__inline__};
3086c72ffaaSAndy Whitcroftour $Member	= qr{->$Ident|\.$Ident|\[[^]]*\]};
3096c72ffaaSAndy Whitcroftour $Lval	= qr{$Ident(?:$Member)*};
3106c72ffaaSAndy Whitcroft
31195e2c602SJoe Perchesour $Int_type	= qr{(?i)llu|ull|ll|lu|ul|l|u};
31295e2c602SJoe Perchesour $Binary	= qr{(?i)0b[01]+$Int_type?};
31395e2c602SJoe Perchesour $Hex	= qr{(?i)0x[0-9a-f]+$Int_type?};
31495e2c602SJoe Perchesour $Int	= qr{[0-9]+$Int_type?};
3152435880fSJoe Perchesour $Octal	= qr{0[0-7]+$Int_type?};
316c0a5c898SJoe Perchesour $String	= qr{"[X\t]*"};
317326b1ffcSJoe Perchesour $Float_hex	= qr{(?i)0x[0-9a-f]+p-?[0-9]+[fl]?};
318326b1ffcSJoe Perchesour $Float_dec	= qr{(?i)(?:[0-9]+\.[0-9]*|[0-9]*\.[0-9]+)(?:e-?[0-9]+)?[fl]?};
319326b1ffcSJoe Perchesour $Float_int	= qr{(?i)[0-9]+e-?[0-9]+[fl]?};
32074349bccSJoe Perchesour $Float	= qr{$Float_hex|$Float_dec|$Float_int};
3212435880fSJoe Perchesour $Constant	= qr{$Float|$Binary|$Octal|$Hex|$Int};
322326b1ffcSJoe Perchesour $Assignment	= qr{\*\=|/=|%=|\+=|-=|<<=|>>=|&=|\^=|\|=|=};
323447432f3SJoe Perchesour $Compare    = qr{<=|>=|==|!=|<|(?<!-)>};
32423f780c9SJoe Perchesour $Arithmetic = qr{\+|-|\*|\/|%};
3256c72ffaaSAndy Whitcroftour $Operators	= qr{
3266c72ffaaSAndy Whitcroft			<=|>=|==|!=|
3276c72ffaaSAndy Whitcroft			=>|->|<<|>>|<|>|!|~|
32823f780c9SJoe Perches			&&|\|\||,|\^|\+\+|--|&|\||$Arithmetic
3296c72ffaaSAndy Whitcroft		  }x;
3306c72ffaaSAndy Whitcroft
33191cb5195SJoe Perchesour $c90_Keywords = qr{do|for|while|if|else|return|goto|continue|switch|default|case|break}x;
33291cb5195SJoe Perches
333ab7e23f3SJoe Perchesour $BasicType;
3348905a67cSAndy Whitcroftour $NonptrType;
3351813087dSJoe Perchesour $NonptrTypeMisordered;
3368716de38SJoe Perchesour $NonptrTypeWithAttr;
3378905a67cSAndy Whitcroftour $Type;
3381813087dSJoe Perchesour $TypeMisordered;
3398905a67cSAndy Whitcroftour $Declare;
3401813087dSJoe Perchesour $DeclareMisordered;
3418905a67cSAndy Whitcroft
34215662b3eSJoe Perchesour $NON_ASCII_UTF8	= qr{
34315662b3eSJoe Perches	[\xC2-\xDF][\x80-\xBF]               # non-overlong 2-byte
344171ae1a4SAndy Whitcroft	|  \xE0[\xA0-\xBF][\x80-\xBF]        # excluding overlongs
345171ae1a4SAndy Whitcroft	| [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2}  # straight 3-byte
346171ae1a4SAndy Whitcroft	|  \xED[\x80-\x9F][\x80-\xBF]        # excluding surrogates
347171ae1a4SAndy Whitcroft	|  \xF0[\x90-\xBF][\x80-\xBF]{2}     # planes 1-3
348171ae1a4SAndy Whitcroft	| [\xF1-\xF3][\x80-\xBF]{3}          # planes 4-15
349171ae1a4SAndy Whitcroft	|  \xF4[\x80-\x8F][\x80-\xBF]{2}     # plane 16
350171ae1a4SAndy Whitcroft}x;
351171ae1a4SAndy Whitcroft
35215662b3eSJoe Perchesour $UTF8	= qr{
35315662b3eSJoe Perches	[\x09\x0A\x0D\x20-\x7E]              # ASCII
35415662b3eSJoe Perches	| $NON_ASCII_UTF8
35515662b3eSJoe Perches}x;
35615662b3eSJoe Perches
357e6176fa4SJoe Perchesour $typeC99Typedefs = qr{(?:__)?(?:[us]_?)?int_?(?:8|16|32|64)_t};
358021158b4SJoe Perchesour $typeOtherOSTypedefs = qr{(?x:
359021158b4SJoe Perches	u_(?:char|short|int|long) |          # bsd
360021158b4SJoe Perches	u(?:nchar|short|int|long)            # sysv
361021158b4SJoe Perches)};
362e6176fa4SJoe Perchesour $typeKernelTypedefs = qr{(?x:
363fb9e9096SAndy Whitcroft	(?:__)?(?:u|s|be|le)(?:8|16|32|64)|
3648ed22cadSAndy Whitcroft	atomic_t
3658ed22cadSAndy Whitcroft)};
366e6176fa4SJoe Perchesour $typeTypedefs = qr{(?x:
367e6176fa4SJoe Perches	$typeC99Typedefs\b|
368e6176fa4SJoe Perches	$typeOtherOSTypedefs\b|
369e6176fa4SJoe Perches	$typeKernelTypedefs\b
370e6176fa4SJoe Perches)};
3718ed22cadSAndy Whitcroft
372691e669bSJoe Perchesour $logFunctions = qr{(?x:
3736e60c02eSJoe Perches	printk(?:_ratelimited|_once|)|
3747d0b6594SJacob Keller	(?:[a-z0-9]+_){1,2}(?:printk|emerg|alert|crit|err|warning|warn|notice|info|debug|dbg|vdbg|devel|cont|WARN)(?:_ratelimited|_once|)|
3756e60c02eSJoe Perches	WARN(?:_RATELIMIT|_ONCE|)|
376b0531722SJoe Perches	panic|
37706668727SJoe Perches	MODULE_[A-Z_]+|
37806668727SJoe Perches	seq_vprintf|seq_printf|seq_puts
379691e669bSJoe Perches)};
380691e669bSJoe Perches
38120112475SJoe Perchesour $signature_tags = qr{(?xi:
38220112475SJoe Perches	Signed-off-by:|
38320112475SJoe Perches	Acked-by:|
38420112475SJoe Perches	Tested-by:|
38520112475SJoe Perches	Reviewed-by:|
38620112475SJoe Perches	Reported-by:|
3878543ae12SMugunthan V N	Suggested-by:|
38820112475SJoe Perches	To:|
38920112475SJoe Perches	Cc:
39020112475SJoe Perches)};
39120112475SJoe Perches
3921813087dSJoe Perchesour @typeListMisordered = (
3931813087dSJoe Perches	qr{char\s+(?:un)?signed},
3941813087dSJoe Perches	qr{int\s+(?:(?:un)?signed\s+)?short\s},
3951813087dSJoe Perches	qr{int\s+short(?:\s+(?:un)?signed)},
3961813087dSJoe Perches	qr{short\s+int(?:\s+(?:un)?signed)},
3971813087dSJoe Perches	qr{(?:un)?signed\s+int\s+short},
3981813087dSJoe Perches	qr{short\s+(?:un)?signed},
3991813087dSJoe Perches	qr{long\s+int\s+(?:un)?signed},
4001813087dSJoe Perches	qr{int\s+long\s+(?:un)?signed},
4011813087dSJoe Perches	qr{long\s+(?:un)?signed\s+int},
4021813087dSJoe Perches	qr{int\s+(?:un)?signed\s+long},
4031813087dSJoe Perches	qr{int\s+(?:un)?signed},
4041813087dSJoe Perches	qr{int\s+long\s+long\s+(?:un)?signed},
4051813087dSJoe Perches	qr{long\s+long\s+int\s+(?:un)?signed},
4061813087dSJoe Perches	qr{long\s+long\s+(?:un)?signed\s+int},
4071813087dSJoe Perches	qr{long\s+long\s+(?:un)?signed},
4081813087dSJoe Perches	qr{long\s+(?:un)?signed},
4091813087dSJoe Perches);
4101813087dSJoe Perches
4118905a67cSAndy Whitcroftour @typeList = (
4128905a67cSAndy Whitcroft	qr{void},
4130c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?char},
4140c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?short\s+int},
4150c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?short},
4160c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?int},
4170c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?long\s+int},
4180c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?long\s+long\s+int},
4190c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?long\s+long},
4200c773d9dSJoe Perches	qr{(?:(?:un)?signed\s+)?long},
4210c773d9dSJoe Perches	qr{(?:un)?signed},
4228905a67cSAndy Whitcroft	qr{float},
4238905a67cSAndy Whitcroft	qr{double},
4248905a67cSAndy Whitcroft	qr{bool},
4258905a67cSAndy Whitcroft	qr{struct\s+$Ident},
4268905a67cSAndy Whitcroft	qr{union\s+$Ident},
4278905a67cSAndy Whitcroft	qr{enum\s+$Ident},
4288905a67cSAndy Whitcroft	qr{${Ident}_t},
4298905a67cSAndy Whitcroft	qr{${Ident}_handler},
4308905a67cSAndy Whitcroft	qr{${Ident}_handler_fn},
4311813087dSJoe Perches	@typeListMisordered,
4328905a67cSAndy Whitcroft);
433485ff23eSAlex Dowadour @typeListFile = ();
4348716de38SJoe Perchesour @typeListWithAttr = (
4358716de38SJoe Perches	@typeList,
4368716de38SJoe Perches	qr{struct\s+$InitAttribute\s+$Ident},
4378716de38SJoe Perches	qr{union\s+$InitAttribute\s+$Ident},
4388716de38SJoe Perches);
4398716de38SJoe Perches
440c45dcabdSAndy Whitcroftour @modifierList = (
441c45dcabdSAndy Whitcroft	qr{fastcall},
442c45dcabdSAndy Whitcroft);
443485ff23eSAlex Dowadour @modifierListFile = ();
4448905a67cSAndy Whitcroft
4452435880fSJoe Perchesour @mode_permission_funcs = (
4462435880fSJoe Perches	["module_param", 3],
4472435880fSJoe Perches	["module_param_(?:array|named|string)", 4],
4482435880fSJoe Perches	["module_param_array_named", 5],
4492435880fSJoe Perches	["debugfs_create_(?:file|u8|u16|u32|u64|x8|x16|x32|x64|size_t|atomic_t|bool|blob|regset32|u32_array)", 2],
4502435880fSJoe Perches	["proc_create(?:_data|)", 2],
4512435880fSJoe Perches	["(?:CLASS|DEVICE|SENSOR)_ATTR", 2],
4522435880fSJoe Perches);
4532435880fSJoe Perches
454515a235eSJoe Perches#Create a search pattern for all these functions to speed up a loop below
455515a235eSJoe Perchesour $mode_perms_search = "";
456515a235eSJoe Perchesforeach my $entry (@mode_permission_funcs) {
457515a235eSJoe Perches	$mode_perms_search .= '|' if ($mode_perms_search ne "");
458515a235eSJoe Perches	$mode_perms_search .= $entry->[0];
459515a235eSJoe Perches}
460515a235eSJoe Perches
461b392c64fSJoe Perchesour $mode_perms_world_writable = qr{
462b392c64fSJoe Perches	S_IWUGO		|
463b392c64fSJoe Perches	S_IWOTH		|
464b392c64fSJoe Perches	S_IRWXUGO	|
465b392c64fSJoe Perches	S_IALLUGO	|
466b392c64fSJoe Perches	0[0-7][0-7][2367]
467b392c64fSJoe Perches}x;
468b392c64fSJoe Perches
4697840a94cSWolfram Sangour $allowed_asm_includes = qr{(?x:
4707840a94cSWolfram Sang	irq|
471cdcee686SSergey Ryazanov	memory|
472cdcee686SSergey Ryazanov	time|
473cdcee686SSergey Ryazanov	reboot
4747840a94cSWolfram Sang)};
4757840a94cSWolfram Sang# memory.h: ARM has a custom one
4767840a94cSWolfram Sang
47766b47b4aSKees Cook# Load common spelling mistakes and build regular expression list.
47866b47b4aSKees Cookmy $misspellings;
47966b47b4aSKees Cookmy %spelling_fix;
48036061e38SJoe Perches
48136061e38SJoe Perchesif (open(my $spelling, '<', $spelling_file)) {
48266b47b4aSKees Cook	while (<$spelling>) {
48366b47b4aSKees Cook		my $line = $_;
48466b47b4aSKees Cook
48566b47b4aSKees Cook		$line =~ s/\s*\n?$//g;
48666b47b4aSKees Cook		$line =~ s/^\s*//g;
48766b47b4aSKees Cook
48866b47b4aSKees Cook		next if ($line =~ m/^\s*#/);
48966b47b4aSKees Cook		next if ($line =~ m/^\s*$/);
49066b47b4aSKees Cook
49166b47b4aSKees Cook		my ($suspect, $fix) = split(/\|\|/, $line);
49266b47b4aSKees Cook
49366b47b4aSKees Cook		$spelling_fix{$suspect} = $fix;
49466b47b4aSKees Cook	}
49566b47b4aSKees Cook	close($spelling);
49636061e38SJoe Perches} else {
49736061e38SJoe Perches	warn "No typos will be found - file '$spelling_file': $!\n";
49836061e38SJoe Perches}
49966b47b4aSKees Cook
500ebfd7d62SJoe Perchesif ($codespell) {
501ebfd7d62SJoe Perches	if (open(my $spelling, '<', $codespellfile)) {
502ebfd7d62SJoe Perches		while (<$spelling>) {
503ebfd7d62SJoe Perches			my $line = $_;
504ebfd7d62SJoe Perches
505ebfd7d62SJoe Perches			$line =~ s/\s*\n?$//g;
506ebfd7d62SJoe Perches			$line =~ s/^\s*//g;
507ebfd7d62SJoe Perches
508ebfd7d62SJoe Perches			next if ($line =~ m/^\s*#/);
509ebfd7d62SJoe Perches			next if ($line =~ m/^\s*$/);
510ebfd7d62SJoe Perches			next if ($line =~ m/, disabled/i);
511ebfd7d62SJoe Perches
512ebfd7d62SJoe Perches			$line =~ s/,.*$//;
513ebfd7d62SJoe Perches
514ebfd7d62SJoe Perches			my ($suspect, $fix) = split(/->/, $line);
515ebfd7d62SJoe Perches
516ebfd7d62SJoe Perches			$spelling_fix{$suspect} = $fix;
517ebfd7d62SJoe Perches		}
518ebfd7d62SJoe Perches		close($spelling);
519ebfd7d62SJoe Perches	} else {
520ebfd7d62SJoe Perches		warn "No codespell typos will be found - file '$codespellfile': $!\n";
521ebfd7d62SJoe Perches	}
522ebfd7d62SJoe Perches}
523ebfd7d62SJoe Perches
524ebfd7d62SJoe Perches$misspellings = join("|", sort keys %spelling_fix) if keys %spelling_fix;
525ebfd7d62SJoe Perches
5268905a67cSAndy Whitcroftsub build_types {
527485ff23eSAlex Dowad	my $mods = "(?x:  \n" . join("|\n  ", (@modifierList, @modifierListFile)) . "\n)";
528485ff23eSAlex Dowad	my $all = "(?x:  \n" . join("|\n  ", (@typeList, @typeListFile)) . "\n)";
5291813087dSJoe Perches	my $Misordered = "(?x:  \n" . join("|\n  ", @typeListMisordered) . "\n)";
5308716de38SJoe Perches	my $allWithAttr = "(?x:  \n" . join("|\n  ", @typeListWithAttr) . "\n)";
531c8cb2ca3SAndy Whitcroft	$Modifier	= qr{(?:$Attribute|$Sparse|$mods)};
532ab7e23f3SJoe Perches	$BasicType	= qr{
533ab7e23f3SJoe Perches				(?:$typeTypedefs\b)|
534ab7e23f3SJoe Perches				(?:${all}\b)
535ab7e23f3SJoe Perches		}x;
5368905a67cSAndy Whitcroft	$NonptrType	= qr{
537d2172eb5SAndy Whitcroft			(?:$Modifier\s+|const\s+)*
538cf655043SAndy Whitcroft			(?:
5396b48db24SAndy Whitcroft				(?:typeof|__typeof__)\s*\([^\)]*\)|
5408ed22cadSAndy Whitcroft				(?:$typeTypedefs\b)|
541c45dcabdSAndy Whitcroft				(?:${all}\b)
542cf655043SAndy Whitcroft			)
543c8cb2ca3SAndy Whitcroft			(?:\s+$Modifier|\s+const)*
5448905a67cSAndy Whitcroft		  }x;
5451813087dSJoe Perches	$NonptrTypeMisordered	= qr{
5461813087dSJoe Perches			(?:$Modifier\s+|const\s+)*
5471813087dSJoe Perches			(?:
5481813087dSJoe Perches				(?:${Misordered}\b)
5491813087dSJoe Perches			)
5501813087dSJoe Perches			(?:\s+$Modifier|\s+const)*
5511813087dSJoe Perches		  }x;
5528716de38SJoe Perches	$NonptrTypeWithAttr	= qr{
5538716de38SJoe Perches			(?:$Modifier\s+|const\s+)*
5548716de38SJoe Perches			(?:
5558716de38SJoe Perches				(?:typeof|__typeof__)\s*\([^\)]*\)|
5568716de38SJoe Perches				(?:$typeTypedefs\b)|
5578716de38SJoe Perches				(?:${allWithAttr}\b)
5588716de38SJoe Perches			)
5598716de38SJoe Perches			(?:\s+$Modifier|\s+const)*
5608716de38SJoe Perches		  }x;
5618905a67cSAndy Whitcroft	$Type	= qr{
562c45dcabdSAndy Whitcroft			$NonptrType
5631574a29fSJoe Perches			(?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)?
564c8cb2ca3SAndy Whitcroft			(?:\s+$Inline|\s+$Modifier)*
5658905a67cSAndy Whitcroft		  }x;
5661813087dSJoe Perches	$TypeMisordered	= qr{
5671813087dSJoe Perches			$NonptrTypeMisordered
5681813087dSJoe Perches			(?:(?:\s|\*|\[\])+\s*const|(?:\s|\*\s*(?:const\s*)?|\[\])+|(?:\s*\[\s*\])+)?
5691813087dSJoe Perches			(?:\s+$Inline|\s+$Modifier)*
5701813087dSJoe Perches		  }x;
57191cb5195SJoe Perches	$Declare	= qr{(?:$Storage\s+(?:$Inline\s+)?)?$Type};
5721813087dSJoe Perches	$DeclareMisordered	= qr{(?:$Storage\s+(?:$Inline\s+)?)?$TypeMisordered};
5738905a67cSAndy Whitcroft}
5748905a67cSAndy Whitcroftbuild_types();
5756c72ffaaSAndy Whitcroft
5767d2367afSJoe Perchesour $Typecast	= qr{\s*(\(\s*$NonptrType\s*\)){0,1}\s*};
577d1fe9c09SJoe Perches
578d1fe9c09SJoe Perches# Using $balanced_parens, $LvalOrFunc, or $FuncArg
579d1fe9c09SJoe Perches# requires at least perl version v5.10.0
580d1fe9c09SJoe Perches# Any use must be runtime checked with $^V
581d1fe9c09SJoe Perches
582d1fe9c09SJoe Perchesour $balanced_parens = qr/(\((?:[^\(\)]++|(?-1))*\))/;
5832435880fSJoe Perchesour $LvalOrFunc	= qr{((?:[\&\*]\s*)?$Lval)\s*($balanced_parens{0,1})\s*};
584c0a5c898SJoe Perchesour $FuncArg = qr{$Typecast{0,1}($LvalOrFunc|$Constant|$String)};
5857d2367afSJoe Perches
586f8422308SJoe Perchesour $declaration_macros = qr{(?x:
587f8422308SJoe Perches	(?:$Storage\s+)?(?:[A-Z_][A-Z0-9]*_){0,2}(?:DEFINE|DECLARE)(?:_[A-Z0-9]+){1,2}\s*\(|
588f8422308SJoe Perches	(?:$Storage\s+)?LIST_HEAD\s*\(|
589f8422308SJoe Perches	(?:$Storage\s+)?${Type}\s+uninitialized_var\s*\(
590f8422308SJoe Perches)};
591f8422308SJoe Perches
5927d2367afSJoe Perchessub deparenthesize {
5937d2367afSJoe Perches	my ($string) = @_;
5947d2367afSJoe Perches	return "" if (!defined($string));
5955b9553abSJoe Perches
5965b9553abSJoe Perches	while ($string =~ /^\s*\(.*\)\s*$/) {
5975b9553abSJoe Perches		$string =~ s@^\s*\(\s*@@;
5985b9553abSJoe Perches		$string =~ s@\s*\)\s*$@@;
5995b9553abSJoe Perches	}
6005b9553abSJoe Perches
6017d2367afSJoe Perches	$string =~ s@\s+@ @g;
6025b9553abSJoe Perches
6037d2367afSJoe Perches	return $string;
6047d2367afSJoe Perches}
6057d2367afSJoe Perches
6063445686aSJoe Perchessub seed_camelcase_file {
6073445686aSJoe Perches	my ($file) = @_;
6083445686aSJoe Perches
6093445686aSJoe Perches	return if (!(-f $file));
6103445686aSJoe Perches
6113445686aSJoe Perches	local $/;
6123445686aSJoe Perches
6133445686aSJoe Perches	open(my $include_file, '<', "$file")
6143445686aSJoe Perches	    or warn "$P: Can't read '$file' $!\n";
6153445686aSJoe Perches	my $text = <$include_file>;
6163445686aSJoe Perches	close($include_file);
6173445686aSJoe Perches
6183445686aSJoe Perches	my @lines = split('\n', $text);
6193445686aSJoe Perches
6203445686aSJoe Perches	foreach my $line (@lines) {
6213445686aSJoe Perches		next if ($line !~ /(?:[A-Z][a-z]|[a-z][A-Z])/);
6223445686aSJoe Perches		if ($line =~ /^[ \t]*(?:#[ \t]*define|typedef\s+$Type)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)/) {
6233445686aSJoe Perches			$camelcase{$1} = 1;
62411ea516aSJoe Perches		} elsif ($line =~ /^\s*$Declare\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[\(\[,;]/) {
62511ea516aSJoe Perches			$camelcase{$1} = 1;
62611ea516aSJoe Perches		} elsif ($line =~ /^\s*(?:union|struct|enum)\s+(\w*(?:[A-Z][a-z]|[a-z][A-Z])\w*)\s*[;\{]/) {
6273445686aSJoe Perches			$camelcase{$1} = 1;
6283445686aSJoe Perches		}
6293445686aSJoe Perches	}
6303445686aSJoe Perches}
6313445686aSJoe Perches
6323445686aSJoe Perchesmy $camelcase_seeded = 0;
6333445686aSJoe Perchessub seed_camelcase_includes {
6343445686aSJoe Perches	return if ($camelcase_seeded);
6353445686aSJoe Perches
6363445686aSJoe Perches	my $files;
637c707a81dSJoe Perches	my $camelcase_cache = "";
638c707a81dSJoe Perches	my @include_files = ();
639c707a81dSJoe Perches
640c707a81dSJoe Perches	$camelcase_seeded = 1;
641351b2a1fSJoe Perches
6423645e328SRichard Genoud	if (-e ".git") {
643351b2a1fSJoe Perches		my $git_last_include_commit = `git log --no-merges --pretty=format:"%h%n" -1 -- include`;
644351b2a1fSJoe Perches		chomp $git_last_include_commit;
645c707a81dSJoe Perches		$camelcase_cache = ".checkpatch-camelcase.git.$git_last_include_commit";
646c707a81dSJoe Perches	} else {
647c707a81dSJoe Perches		my $last_mod_date = 0;
648c707a81dSJoe Perches		$files = `find $root/include -name "*.h"`;
649c707a81dSJoe Perches		@include_files = split('\n', $files);
650c707a81dSJoe Perches		foreach my $file (@include_files) {
651c707a81dSJoe Perches			my $date = POSIX::strftime("%Y%m%d%H%M",
652c707a81dSJoe Perches						   localtime((stat $file)[9]));
653c707a81dSJoe Perches			$last_mod_date = $date if ($last_mod_date < $date);
654c707a81dSJoe Perches		}
655c707a81dSJoe Perches		$camelcase_cache = ".checkpatch-camelcase.date.$last_mod_date";
656c707a81dSJoe Perches	}
657c707a81dSJoe Perches
658c707a81dSJoe Perches	if ($camelcase_cache ne "" && -f $camelcase_cache) {
659c707a81dSJoe Perches		open(my $camelcase_file, '<', "$camelcase_cache")
660c707a81dSJoe Perches		    or warn "$P: Can't read '$camelcase_cache' $!\n";
661351b2a1fSJoe Perches		while (<$camelcase_file>) {
662351b2a1fSJoe Perches			chomp;
663351b2a1fSJoe Perches			$camelcase{$_} = 1;
664351b2a1fSJoe Perches		}
665351b2a1fSJoe Perches		close($camelcase_file);
666351b2a1fSJoe Perches
667351b2a1fSJoe Perches		return;
668351b2a1fSJoe Perches	}
669c707a81dSJoe Perches
6703645e328SRichard Genoud	if (-e ".git") {
671c707a81dSJoe Perches		$files = `git ls-files "include/*.h"`;
672c707a81dSJoe Perches		@include_files = split('\n', $files);
6733445686aSJoe Perches	}
674c707a81dSJoe Perches
6753445686aSJoe Perches	foreach my $file (@include_files) {
6763445686aSJoe Perches		seed_camelcase_file($file);
6773445686aSJoe Perches	}
678351b2a1fSJoe Perches
679c707a81dSJoe Perches	if ($camelcase_cache ne "") {
680351b2a1fSJoe Perches		unlink glob ".checkpatch-camelcase.*";
681c707a81dSJoe Perches		open(my $camelcase_file, '>', "$camelcase_cache")
682c707a81dSJoe Perches		    or warn "$P: Can't write '$camelcase_cache' $!\n";
683351b2a1fSJoe Perches		foreach (sort { lc($a) cmp lc($b) } keys(%camelcase)) {
684351b2a1fSJoe Perches			print $camelcase_file ("$_\n");
685351b2a1fSJoe Perches		}
686351b2a1fSJoe Perches		close($camelcase_file);
687351b2a1fSJoe Perches	}
6883445686aSJoe Perches}
6893445686aSJoe Perches
690d311cd44SJoe Perchessub git_commit_info {
691d311cd44SJoe Perches	my ($commit, $id, $desc) = @_;
692d311cd44SJoe Perches
693d311cd44SJoe Perches	return ($id, $desc) if ((which("git") eq "") || !(-e ".git"));
694d311cd44SJoe Perches
695d311cd44SJoe Perches	my $output = `git log --no-color --format='%H %s' -1 $commit 2>&1`;
696d311cd44SJoe Perches	$output =~ s/^\s*//gm;
697d311cd44SJoe Perches	my @lines = split("\n", $output);
698d311cd44SJoe Perches
6990d7835fcSJoe Perches	return ($id, $desc) if ($#lines < 0);
7000d7835fcSJoe Perches
701d311cd44SJoe Perches	if ($lines[0] =~ /^error: short SHA1 $commit is ambiguous\./) {
702d311cd44SJoe Perches# Maybe one day convert this block of bash into something that returns
703d311cd44SJoe Perches# all matching commit ids, but it's very slow...
704d311cd44SJoe Perches#
705d311cd44SJoe Perches#		echo "checking commits $1..."
706d311cd44SJoe Perches#		git rev-list --remotes | grep -i "^$1" |
707d311cd44SJoe Perches#		while read line ; do
708d311cd44SJoe Perches#		    git log --format='%H %s' -1 $line |
709d311cd44SJoe Perches#		    echo "commit $(cut -c 1-12,41-)"
710d311cd44SJoe Perches#		done
711d311cd44SJoe Perches	} elsif ($lines[0] =~ /^fatal: ambiguous argument '$commit': unknown revision or path not in the working tree\./) {
712d311cd44SJoe Perches	} else {
713d311cd44SJoe Perches		$id = substr($lines[0], 0, 12);
714d311cd44SJoe Perches		$desc = substr($lines[0], 41);
715d311cd44SJoe Perches	}
716d311cd44SJoe Perches
717d311cd44SJoe Perches	return ($id, $desc);
718d311cd44SJoe Perches}
719d311cd44SJoe Perches
7206c72ffaaSAndy Whitcroft$chk_signoff = 0 if ($file);
7210a920b5bSAndy Whitcroft
72200df344fSAndy Whitcroftmy @rawlines = ();
723c2fdda0dSAndy Whitcroftmy @lines = ();
7243705ce5bSJoe Perchesmy @fixed = ();
725d752fcc8SJoe Perchesmy @fixed_inserted = ();
726d752fcc8SJoe Perchesmy @fixed_deleted = ();
727194f66fcSJoe Perchesmy $fixlinenr = -1;
728194f66fcSJoe Perches
729c2fdda0dSAndy Whitcroftmy $vname;
7306c72ffaaSAndy Whitcroftfor my $filename (@ARGV) {
73121caa13cSAndy Whitcroft	my $FILE;
7326c72ffaaSAndy Whitcroft	if ($file) {
73321caa13cSAndy Whitcroft		open($FILE, '-|', "diff -u /dev/null $filename") ||
7346c72ffaaSAndy Whitcroft			die "$P: $filename: diff failed - $!\n";
73521caa13cSAndy Whitcroft	} elsif ($filename eq '-') {
73621caa13cSAndy Whitcroft		open($FILE, '<&STDIN');
7376c72ffaaSAndy Whitcroft	} else {
73821caa13cSAndy Whitcroft		open($FILE, '<', "$filename") ||
7396c72ffaaSAndy Whitcroft			die "$P: $filename: open failed - $!\n";
7406c72ffaaSAndy Whitcroft	}
741c2fdda0dSAndy Whitcroft	if ($filename eq '-') {
742c2fdda0dSAndy Whitcroft		$vname = 'Your patch';
743c2fdda0dSAndy Whitcroft	} else {
744c2fdda0dSAndy Whitcroft		$vname = $filename;
745c2fdda0dSAndy Whitcroft	}
74621caa13cSAndy Whitcroft	while (<$FILE>) {
7470a920b5bSAndy Whitcroft		chomp;
74800df344fSAndy Whitcroft		push(@rawlines, $_);
7496c72ffaaSAndy Whitcroft	}
75021caa13cSAndy Whitcroft	close($FILE);
751d8469f16SJoe Perches
752d8469f16SJoe Perches	if ($#ARGV > 0 && $quiet == 0) {
753d8469f16SJoe Perches		print '-' x length($vname) . "\n";
754d8469f16SJoe Perches		print "$vname\n";
755d8469f16SJoe Perches		print '-' x length($vname) . "\n";
756d8469f16SJoe Perches	}
757d8469f16SJoe Perches
758c2fdda0dSAndy Whitcroft	if (!process($filename)) {
7590a920b5bSAndy Whitcroft		$exit = 1;
7600a920b5bSAndy Whitcroft	}
76100df344fSAndy Whitcroft	@rawlines = ();
76213214adfSAndy Whitcroft	@lines = ();
7633705ce5bSJoe Perches	@fixed = ();
764d752fcc8SJoe Perches	@fixed_inserted = ();
765d752fcc8SJoe Perches	@fixed_deleted = ();
766194f66fcSJoe Perches	$fixlinenr = -1;
767485ff23eSAlex Dowad	@modifierListFile = ();
768485ff23eSAlex Dowad	@typeListFile = ();
769485ff23eSAlex Dowad	build_types();
7700a920b5bSAndy Whitcroft}
7710a920b5bSAndy Whitcroft
772d8469f16SJoe Perchesif (!$quiet) {
7733c816e49SJoe Perches	hash_show_words(\%use_type, "Used");
7743c816e49SJoe Perches	hash_show_words(\%ignore_type, "Ignored");
7753c816e49SJoe Perches
776d8469f16SJoe Perches	if ($^V lt 5.10.0) {
777d8469f16SJoe Perches		print << "EOM"
778d8469f16SJoe Perches
779d8469f16SJoe PerchesNOTE: perl $^V is not modern enough to detect all possible issues.
780d8469f16SJoe Perches      An upgrade to at least perl v5.10.0 is suggested.
781d8469f16SJoe PerchesEOM
782d8469f16SJoe Perches	}
783d8469f16SJoe Perches	if ($exit) {
784d8469f16SJoe Perches		print << "EOM"
785d8469f16SJoe Perches
786d8469f16SJoe PerchesNOTE: If any of the errors are false positives, please report
787d8469f16SJoe Perches      them to the maintainer, see CHECKPATCH in MAINTAINERS.
788d8469f16SJoe PerchesEOM
789d8469f16SJoe Perches	}
790d8469f16SJoe Perches}
791d8469f16SJoe Perches
7920a920b5bSAndy Whitcroftexit($exit);
7930a920b5bSAndy Whitcroft
7940a920b5bSAndy Whitcroftsub top_of_kernel_tree {
7956c72ffaaSAndy Whitcroft	my ($root) = @_;
7966c72ffaaSAndy Whitcroft
7976c72ffaaSAndy Whitcroft	my @tree_check = (
7986c72ffaaSAndy Whitcroft		"COPYING", "CREDITS", "Kbuild", "MAINTAINERS", "Makefile",
7996c72ffaaSAndy Whitcroft		"README", "Documentation", "arch", "include", "drivers",
8006c72ffaaSAndy Whitcroft		"fs", "init", "ipc", "kernel", "lib", "scripts",
8016c72ffaaSAndy Whitcroft	);
8026c72ffaaSAndy Whitcroft
8036c72ffaaSAndy Whitcroft	foreach my $check (@tree_check) {
8046c72ffaaSAndy Whitcroft		if (! -e $root . '/' . $check) {
8050a920b5bSAndy Whitcroft			return 0;
8060a920b5bSAndy Whitcroft		}
8076c72ffaaSAndy Whitcroft	}
8086c72ffaaSAndy Whitcroft	return 1;
8096c72ffaaSAndy Whitcroft}
8100a920b5bSAndy Whitcroft
81120112475SJoe Perchessub parse_email {
81220112475SJoe Perches	my ($formatted_email) = @_;
81320112475SJoe Perches
81420112475SJoe Perches	my $name = "";
81520112475SJoe Perches	my $address = "";
81620112475SJoe Perches	my $comment = "";
81720112475SJoe Perches
81820112475SJoe Perches	if ($formatted_email =~ /^(.*)<(\S+\@\S+)>(.*)$/) {
81920112475SJoe Perches		$name = $1;
82020112475SJoe Perches		$address = $2;
82120112475SJoe Perches		$comment = $3 if defined $3;
82220112475SJoe Perches	} elsif ($formatted_email =~ /^\s*<(\S+\@\S+)>(.*)$/) {
82320112475SJoe Perches		$address = $1;
82420112475SJoe Perches		$comment = $2 if defined $2;
82520112475SJoe Perches	} elsif ($formatted_email =~ /(\S+\@\S+)(.*)$/) {
82620112475SJoe Perches		$address = $1;
82720112475SJoe Perches		$comment = $2 if defined $2;
82820112475SJoe Perches		$formatted_email =~ s/$address.*$//;
82920112475SJoe Perches		$name = $formatted_email;
8303705ce5bSJoe Perches		$name = trim($name);
83120112475SJoe Perches		$name =~ s/^\"|\"$//g;
83220112475SJoe Perches		# If there's a name left after stripping spaces and
83320112475SJoe Perches		# leading quotes, and the address doesn't have both
83420112475SJoe Perches		# leading and trailing angle brackets, the address
83520112475SJoe Perches		# is invalid. ie:
83620112475SJoe Perches		#   "joe smith [email protected]" bad
83720112475SJoe Perches		#   "joe smith <[email protected]" bad
83820112475SJoe Perches		if ($name ne "" && $address !~ /^<[^>]+>$/) {
83920112475SJoe Perches			$name = "";
84020112475SJoe Perches			$address = "";
84120112475SJoe Perches			$comment = "";
84220112475SJoe Perches		}
84320112475SJoe Perches	}
84420112475SJoe Perches
8453705ce5bSJoe Perches	$name = trim($name);
84620112475SJoe Perches	$name =~ s/^\"|\"$//g;
8473705ce5bSJoe Perches	$address = trim($address);
84820112475SJoe Perches	$address =~ s/^\<|\>$//g;
84920112475SJoe Perches
85020112475SJoe Perches	if ($name =~ /[^\w \-]/i) { ##has "must quote" chars
85120112475SJoe Perches		$name =~ s/(?<!\\)"/\\"/g; ##escape quotes
85220112475SJoe Perches		$name = "\"$name\"";
85320112475SJoe Perches	}
85420112475SJoe Perches
85520112475SJoe Perches	return ($name, $address, $comment);
85620112475SJoe Perches}
85720112475SJoe Perches
85820112475SJoe Perchessub format_email {
85920112475SJoe Perches	my ($name, $address) = @_;
86020112475SJoe Perches
86120112475SJoe Perches	my $formatted_email;
86220112475SJoe Perches
8633705ce5bSJoe Perches	$name = trim($name);
86420112475SJoe Perches	$name =~ s/^\"|\"$//g;
8653705ce5bSJoe Perches	$address = trim($address);
86620112475SJoe Perches
86720112475SJoe Perches	if ($name =~ /[^\w \-]/i) { ##has "must quote" chars
86820112475SJoe Perches		$name =~ s/(?<!\\)"/\\"/g; ##escape quotes
86920112475SJoe Perches		$name = "\"$name\"";
87020112475SJoe Perches	}
87120112475SJoe Perches
87220112475SJoe Perches	if ("$name" eq "") {
87320112475SJoe Perches		$formatted_email = "$address";
87420112475SJoe Perches	} else {
87520112475SJoe Perches		$formatted_email = "$name <$address>";
87620112475SJoe Perches	}
87720112475SJoe Perches
87820112475SJoe Perches	return $formatted_email;
87920112475SJoe Perches}
88020112475SJoe Perches
881d311cd44SJoe Perchessub which {
882d311cd44SJoe Perches	my ($bin) = @_;
883d311cd44SJoe Perches
884d311cd44SJoe Perches	foreach my $path (split(/:/, $ENV{PATH})) {
885d311cd44SJoe Perches		if (-e "$path/$bin") {
886d311cd44SJoe Perches			return "$path/$bin";
887d311cd44SJoe Perches		}
888d311cd44SJoe Perches	}
889d311cd44SJoe Perches
890d311cd44SJoe Perches	return "";
891d311cd44SJoe Perches}
892d311cd44SJoe Perches
893000d1cc1SJoe Perchessub which_conf {
894000d1cc1SJoe Perches	my ($conf) = @_;
895000d1cc1SJoe Perches
896000d1cc1SJoe Perches	foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) {
897000d1cc1SJoe Perches		if (-e "$path/$conf") {
898000d1cc1SJoe Perches			return "$path/$conf";
899000d1cc1SJoe Perches		}
900000d1cc1SJoe Perches	}
901000d1cc1SJoe Perches
902000d1cc1SJoe Perches	return "";
903000d1cc1SJoe Perches}
904000d1cc1SJoe Perches
9050a920b5bSAndy Whitcroftsub expand_tabs {
9060a920b5bSAndy Whitcroft	my ($str) = @_;
9070a920b5bSAndy Whitcroft
9080a920b5bSAndy Whitcroft	my $res = '';
9090a920b5bSAndy Whitcroft	my $n = 0;
9100a920b5bSAndy Whitcroft	for my $c (split(//, $str)) {
9110a920b5bSAndy Whitcroft		if ($c eq "\t") {
9120a920b5bSAndy Whitcroft			$res .= ' ';
9130a920b5bSAndy Whitcroft			$n++;
9140a920b5bSAndy Whitcroft			for (; ($n % 8) != 0; $n++) {
9150a920b5bSAndy Whitcroft				$res .= ' ';
9160a920b5bSAndy Whitcroft			}
9170a920b5bSAndy Whitcroft			next;
9180a920b5bSAndy Whitcroft		}
9190a920b5bSAndy Whitcroft		$res .= $c;
9200a920b5bSAndy Whitcroft		$n++;
9210a920b5bSAndy Whitcroft	}
9220a920b5bSAndy Whitcroft
9230a920b5bSAndy Whitcroft	return $res;
9240a920b5bSAndy Whitcroft}
9256c72ffaaSAndy Whitcroftsub copy_spacing {
926773647a0SAndy Whitcroft	(my $res = shift) =~ tr/\t/ /c;
9276c72ffaaSAndy Whitcroft	return $res;
9286c72ffaaSAndy Whitcroft}
9290a920b5bSAndy Whitcroft
9304a0df2efSAndy Whitcroftsub line_stats {
9314a0df2efSAndy Whitcroft	my ($line) = @_;
9324a0df2efSAndy Whitcroft
9334a0df2efSAndy Whitcroft	# Drop the diff line leader and expand tabs
9344a0df2efSAndy Whitcroft	$line =~ s/^.//;
9354a0df2efSAndy Whitcroft	$line = expand_tabs($line);
9364a0df2efSAndy Whitcroft
9374a0df2efSAndy Whitcroft	# Pick the indent from the front of the line.
9384a0df2efSAndy Whitcroft	my ($white) = ($line =~ /^(\s*)/);
9394a0df2efSAndy Whitcroft
9404a0df2efSAndy Whitcroft	return (length($line), length($white));
9414a0df2efSAndy Whitcroft}
9424a0df2efSAndy Whitcroft
943773647a0SAndy Whitcroftmy $sanitise_quote = '';
944773647a0SAndy Whitcroft
945773647a0SAndy Whitcroftsub sanitise_line_reset {
946773647a0SAndy Whitcroft	my ($in_comment) = @_;
947773647a0SAndy Whitcroft
948773647a0SAndy Whitcroft	if ($in_comment) {
949773647a0SAndy Whitcroft		$sanitise_quote = '*/';
950773647a0SAndy Whitcroft	} else {
951773647a0SAndy Whitcroft		$sanitise_quote = '';
952773647a0SAndy Whitcroft	}
953773647a0SAndy Whitcroft}
95400df344fSAndy Whitcroftsub sanitise_line {
95500df344fSAndy Whitcroft	my ($line) = @_;
95600df344fSAndy Whitcroft
95700df344fSAndy Whitcroft	my $res = '';
95800df344fSAndy Whitcroft	my $l = '';
95900df344fSAndy Whitcroft
960c2fdda0dSAndy Whitcroft	my $qlen = 0;
961773647a0SAndy Whitcroft	my $off = 0;
962773647a0SAndy Whitcroft	my $c;
96300df344fSAndy Whitcroft
964773647a0SAndy Whitcroft	# Always copy over the diff marker.
965773647a0SAndy Whitcroft	$res = substr($line, 0, 1);
966773647a0SAndy Whitcroft
967773647a0SAndy Whitcroft	for ($off = 1; $off < length($line); $off++) {
968773647a0SAndy Whitcroft		$c = substr($line, $off, 1);
969773647a0SAndy Whitcroft
970773647a0SAndy Whitcroft		# Comments we are wacking completly including the begin
971773647a0SAndy Whitcroft		# and end, all to $;.
972773647a0SAndy Whitcroft		if ($sanitise_quote eq '' && substr($line, $off, 2) eq '/*') {
973773647a0SAndy Whitcroft			$sanitise_quote = '*/';
974773647a0SAndy Whitcroft
975773647a0SAndy Whitcroft			substr($res, $off, 2, "$;$;");
976773647a0SAndy Whitcroft			$off++;
97700df344fSAndy Whitcroft			next;
978773647a0SAndy Whitcroft		}
97981bc0e02SAndy Whitcroft		if ($sanitise_quote eq '*/' && substr($line, $off, 2) eq '*/') {
980773647a0SAndy Whitcroft			$sanitise_quote = '';
981773647a0SAndy Whitcroft			substr($res, $off, 2, "$;$;");
982773647a0SAndy Whitcroft			$off++;
983773647a0SAndy Whitcroft			next;
984773647a0SAndy Whitcroft		}
985113f04a8SDaniel Walker		if ($sanitise_quote eq '' && substr($line, $off, 2) eq '//') {
986113f04a8SDaniel Walker			$sanitise_quote = '//';
987113f04a8SDaniel Walker
988113f04a8SDaniel Walker			substr($res, $off, 2, $sanitise_quote);
989113f04a8SDaniel Walker			$off++;
990113f04a8SDaniel Walker			next;
991113f04a8SDaniel Walker		}
992773647a0SAndy Whitcroft
993773647a0SAndy Whitcroft		# A \ in a string means ignore the next character.
994773647a0SAndy Whitcroft		if (($sanitise_quote eq "'" || $sanitise_quote eq '"') &&
995773647a0SAndy Whitcroft		    $c eq "\\") {
996773647a0SAndy Whitcroft			substr($res, $off, 2, 'XX');
997773647a0SAndy Whitcroft			$off++;
998773647a0SAndy Whitcroft			next;
999773647a0SAndy Whitcroft		}
1000773647a0SAndy Whitcroft		# Regular quotes.
1001773647a0SAndy Whitcroft		if ($c eq "'" || $c eq '"') {
1002773647a0SAndy Whitcroft			if ($sanitise_quote eq '') {
1003773647a0SAndy Whitcroft				$sanitise_quote = $c;
1004773647a0SAndy Whitcroft
1005773647a0SAndy Whitcroft				substr($res, $off, 1, $c);
1006773647a0SAndy Whitcroft				next;
1007773647a0SAndy Whitcroft			} elsif ($sanitise_quote eq $c) {
1008773647a0SAndy Whitcroft				$sanitise_quote = '';
100900df344fSAndy Whitcroft			}
101000df344fSAndy Whitcroft		}
1011773647a0SAndy Whitcroft
1012fae17daeSAndy Whitcroft		#print "c<$c> SQ<$sanitise_quote>\n";
1013773647a0SAndy Whitcroft		if ($off != 0 && $sanitise_quote eq '*/' && $c ne "\t") {
1014773647a0SAndy Whitcroft			substr($res, $off, 1, $;);
1015113f04a8SDaniel Walker		} elsif ($off != 0 && $sanitise_quote eq '//' && $c ne "\t") {
1016113f04a8SDaniel Walker			substr($res, $off, 1, $;);
1017773647a0SAndy Whitcroft		} elsif ($off != 0 && $sanitise_quote && $c ne "\t") {
1018773647a0SAndy Whitcroft			substr($res, $off, 1, 'X');
101900df344fSAndy Whitcroft		} else {
1020773647a0SAndy Whitcroft			substr($res, $off, 1, $c);
102100df344fSAndy Whitcroft		}
1022c2fdda0dSAndy Whitcroft	}
1023c2fdda0dSAndy Whitcroft
1024113f04a8SDaniel Walker	if ($sanitise_quote eq '//') {
1025113f04a8SDaniel Walker		$sanitise_quote = '';
1026113f04a8SDaniel Walker	}
1027113f04a8SDaniel Walker
1028c2fdda0dSAndy Whitcroft	# The pathname on a #include may be surrounded by '<' and '>'.
1029c45dcabdSAndy Whitcroft	if ($res =~ /^.\s*\#\s*include\s+\<(.*)\>/) {
1030c2fdda0dSAndy Whitcroft		my $clean = 'X' x length($1);
1031c2fdda0dSAndy Whitcroft		$res =~ s@\<.*\>@<$clean>@;
1032c2fdda0dSAndy Whitcroft
1033c2fdda0dSAndy Whitcroft	# The whole of a #error is a string.
1034c45dcabdSAndy Whitcroft	} elsif ($res =~ /^.\s*\#\s*(?:error|warning)\s+(.*)\b/) {
1035c2fdda0dSAndy Whitcroft		my $clean = 'X' x length($1);
1036c45dcabdSAndy Whitcroft		$res =~ s@(\#\s*(?:error|warning)\s+).*@$1$clean@;
1037c2fdda0dSAndy Whitcroft	}
1038c2fdda0dSAndy Whitcroft
103900df344fSAndy Whitcroft	return $res;
104000df344fSAndy Whitcroft}
104100df344fSAndy Whitcroft
1042a6962d72SJoe Perchessub get_quoted_string {
1043a6962d72SJoe Perches	my ($line, $rawline) = @_;
1044a6962d72SJoe Perches
104533acb54aSJoe Perches	return "" if ($line !~ m/($String)/g);
1046a6962d72SJoe Perches	return substr($rawline, $-[0], $+[0] - $-[0]);
1047a6962d72SJoe Perches}
1048a6962d72SJoe Perches
10498905a67cSAndy Whitcroftsub ctx_statement_block {
10508905a67cSAndy Whitcroft	my ($linenr, $remain, $off) = @_;
10518905a67cSAndy Whitcroft	my $line = $linenr - 1;
10528905a67cSAndy Whitcroft	my $blk = '';
10538905a67cSAndy Whitcroft	my $soff = $off;
10548905a67cSAndy Whitcroft	my $coff = $off - 1;
1055773647a0SAndy Whitcroft	my $coff_set = 0;
10568905a67cSAndy Whitcroft
105713214adfSAndy Whitcroft	my $loff = 0;
105813214adfSAndy Whitcroft
10598905a67cSAndy Whitcroft	my $type = '';
10608905a67cSAndy Whitcroft	my $level = 0;
1061a2750645SAndy Whitcroft	my @stack = ();
1062cf655043SAndy Whitcroft	my $p;
10638905a67cSAndy Whitcroft	my $c;
10648905a67cSAndy Whitcroft	my $len = 0;
106513214adfSAndy Whitcroft
106613214adfSAndy Whitcroft	my $remainder;
10678905a67cSAndy Whitcroft	while (1) {
1068a2750645SAndy Whitcroft		@stack = (['', 0]) if ($#stack == -1);
1069a2750645SAndy Whitcroft
1070773647a0SAndy Whitcroft		#warn "CSB: blk<$blk> remain<$remain>\n";
10718905a67cSAndy Whitcroft		# If we are about to drop off the end, pull in more
10728905a67cSAndy Whitcroft		# context.
10738905a67cSAndy Whitcroft		if ($off >= $len) {
10748905a67cSAndy Whitcroft			for (; $remain > 0; $line++) {
1075dea33496SAndy Whitcroft				last if (!defined $lines[$line]);
1076c2fdda0dSAndy Whitcroft				next if ($lines[$line] =~ /^-/);
10778905a67cSAndy Whitcroft				$remain--;
107813214adfSAndy Whitcroft				$loff = $len;
1079c2fdda0dSAndy Whitcroft				$blk .= $lines[$line] . "\n";
10808905a67cSAndy Whitcroft				$len = length($blk);
10818905a67cSAndy Whitcroft				$line++;
10828905a67cSAndy Whitcroft				last;
10838905a67cSAndy Whitcroft			}
10848905a67cSAndy Whitcroft			# Bail if there is no further context.
10858905a67cSAndy Whitcroft			#warn "CSB: blk<$blk> off<$off> len<$len>\n";
108613214adfSAndy Whitcroft			if ($off >= $len) {
10878905a67cSAndy Whitcroft				last;
10888905a67cSAndy Whitcroft			}
1089f74bd194SAndy Whitcroft			if ($level == 0 && substr($blk, $off) =~ /^.\s*#\s*define/) {
1090f74bd194SAndy Whitcroft				$level++;
1091f74bd194SAndy Whitcroft				$type = '#';
1092f74bd194SAndy Whitcroft			}
10938905a67cSAndy Whitcroft		}
1094cf655043SAndy Whitcroft		$p = $c;
10958905a67cSAndy Whitcroft		$c = substr($blk, $off, 1);
109613214adfSAndy Whitcroft		$remainder = substr($blk, $off);
10978905a67cSAndy Whitcroft
1098773647a0SAndy Whitcroft		#warn "CSB: c<$c> type<$type> level<$level> remainder<$remainder> coff_set<$coff_set>\n";
10994635f4fbSAndy Whitcroft
11004635f4fbSAndy Whitcroft		# Handle nested #if/#else.
11014635f4fbSAndy Whitcroft		if ($remainder =~ /^#\s*(?:ifndef|ifdef|if)\s/) {
11024635f4fbSAndy Whitcroft			push(@stack, [ $type, $level ]);
11034635f4fbSAndy Whitcroft		} elsif ($remainder =~ /^#\s*(?:else|elif)\b/) {
11044635f4fbSAndy Whitcroft			($type, $level) = @{$stack[$#stack - 1]};
11054635f4fbSAndy Whitcroft		} elsif ($remainder =~ /^#\s*endif\b/) {
11064635f4fbSAndy Whitcroft			($type, $level) = @{pop(@stack)};
11074635f4fbSAndy Whitcroft		}
11084635f4fbSAndy Whitcroft
11098905a67cSAndy Whitcroft		# Statement ends at the ';' or a close '}' at the
11108905a67cSAndy Whitcroft		# outermost level.
11118905a67cSAndy Whitcroft		if ($level == 0 && $c eq ';') {
11128905a67cSAndy Whitcroft			last;
11138905a67cSAndy Whitcroft		}
11148905a67cSAndy Whitcroft
111513214adfSAndy Whitcroft		# An else is really a conditional as long as its not else if
1116773647a0SAndy Whitcroft		if ($level == 0 && $coff_set == 0 &&
1117773647a0SAndy Whitcroft				(!defined($p) || $p =~ /(?:\s|\}|\+)/) &&
1118773647a0SAndy Whitcroft				$remainder =~ /^(else)(?:\s|{)/ &&
1119773647a0SAndy Whitcroft				$remainder !~ /^else\s+if\b/) {
1120773647a0SAndy Whitcroft			$coff = $off + length($1) - 1;
1121773647a0SAndy Whitcroft			$coff_set = 1;
1122773647a0SAndy Whitcroft			#warn "CSB: mark coff<$coff> soff<$soff> 1<$1>\n";
1123773647a0SAndy Whitcroft			#warn "[" . substr($blk, $soff, $coff - $soff + 1) . "]\n";
112413214adfSAndy Whitcroft		}
112513214adfSAndy Whitcroft
11268905a67cSAndy Whitcroft		if (($type eq '' || $type eq '(') && $c eq '(') {
11278905a67cSAndy Whitcroft			$level++;
11288905a67cSAndy Whitcroft			$type = '(';
11298905a67cSAndy Whitcroft		}
11308905a67cSAndy Whitcroft		if ($type eq '(' && $c eq ')') {
11318905a67cSAndy Whitcroft			$level--;
11328905a67cSAndy Whitcroft			$type = ($level != 0)? '(' : '';
11338905a67cSAndy Whitcroft
11348905a67cSAndy Whitcroft			if ($level == 0 && $coff < $soff) {
11358905a67cSAndy Whitcroft				$coff = $off;
1136773647a0SAndy Whitcroft				$coff_set = 1;
1137773647a0SAndy Whitcroft				#warn "CSB: mark coff<$coff>\n";
11388905a67cSAndy Whitcroft			}
11398905a67cSAndy Whitcroft		}
11408905a67cSAndy Whitcroft		if (($type eq '' || $type eq '{') && $c eq '{') {
11418905a67cSAndy Whitcroft			$level++;
11428905a67cSAndy Whitcroft			$type = '{';
11438905a67cSAndy Whitcroft		}
11448905a67cSAndy Whitcroft		if ($type eq '{' && $c eq '}') {
11458905a67cSAndy Whitcroft			$level--;
11468905a67cSAndy Whitcroft			$type = ($level != 0)? '{' : '';
11478905a67cSAndy Whitcroft
11488905a67cSAndy Whitcroft			if ($level == 0) {
1149b998e001SPatrick Pannuto				if (substr($blk, $off + 1, 1) eq ';') {
1150b998e001SPatrick Pannuto					$off++;
1151b998e001SPatrick Pannuto				}
11528905a67cSAndy Whitcroft				last;
11538905a67cSAndy Whitcroft			}
11548905a67cSAndy Whitcroft		}
1155f74bd194SAndy Whitcroft		# Preprocessor commands end at the newline unless escaped.
1156f74bd194SAndy Whitcroft		if ($type eq '#' && $c eq "\n" && $p ne "\\") {
1157f74bd194SAndy Whitcroft			$level--;
1158f74bd194SAndy Whitcroft			$type = '';
1159f74bd194SAndy Whitcroft			$off++;
1160f74bd194SAndy Whitcroft			last;
1161f74bd194SAndy Whitcroft		}
11628905a67cSAndy Whitcroft		$off++;
11638905a67cSAndy Whitcroft	}
1164a3bb97a7SAndy Whitcroft	# We are truly at the end, so shuffle to the next line.
116513214adfSAndy Whitcroft	if ($off == $len) {
1166a3bb97a7SAndy Whitcroft		$loff = $len + 1;
116713214adfSAndy Whitcroft		$line++;
116813214adfSAndy Whitcroft		$remain--;
116913214adfSAndy Whitcroft	}
11708905a67cSAndy Whitcroft
11718905a67cSAndy Whitcroft	my $statement = substr($blk, $soff, $off - $soff + 1);
11728905a67cSAndy Whitcroft	my $condition = substr($blk, $soff, $coff - $soff + 1);
11738905a67cSAndy Whitcroft
11748905a67cSAndy Whitcroft	#warn "STATEMENT<$statement>\n";
11758905a67cSAndy Whitcroft	#warn "CONDITION<$condition>\n";
11768905a67cSAndy Whitcroft
1177773647a0SAndy Whitcroft	#print "coff<$coff> soff<$off> loff<$loff>\n";
117813214adfSAndy Whitcroft
117913214adfSAndy Whitcroft	return ($statement, $condition,
118013214adfSAndy Whitcroft			$line, $remain + 1, $off - $loff + 1, $level);
118113214adfSAndy Whitcroft}
118213214adfSAndy Whitcroft
1183cf655043SAndy Whitcroftsub statement_lines {
1184cf655043SAndy Whitcroft	my ($stmt) = @_;
1185cf655043SAndy Whitcroft
1186cf655043SAndy Whitcroft	# Strip the diff line prefixes and rip blank lines at start and end.
1187cf655043SAndy Whitcroft	$stmt =~ s/(^|\n)./$1/g;
1188cf655043SAndy Whitcroft	$stmt =~ s/^\s*//;
1189cf655043SAndy Whitcroft	$stmt =~ s/\s*$//;
1190cf655043SAndy Whitcroft
1191cf655043SAndy Whitcroft	my @stmt_lines = ($stmt =~ /\n/g);
1192cf655043SAndy Whitcroft
1193cf655043SAndy Whitcroft	return $#stmt_lines + 2;
1194cf655043SAndy Whitcroft}
1195cf655043SAndy Whitcroft
1196cf655043SAndy Whitcroftsub statement_rawlines {
1197cf655043SAndy Whitcroft	my ($stmt) = @_;
1198cf655043SAndy Whitcroft
1199cf655043SAndy Whitcroft	my @stmt_lines = ($stmt =~ /\n/g);
1200cf655043SAndy Whitcroft
1201cf655043SAndy Whitcroft	return $#stmt_lines + 2;
1202cf655043SAndy Whitcroft}
1203cf655043SAndy Whitcroft
1204cf655043SAndy Whitcroftsub statement_block_size {
1205cf655043SAndy Whitcroft	my ($stmt) = @_;
1206cf655043SAndy Whitcroft
1207cf655043SAndy Whitcroft	$stmt =~ s/(^|\n)./$1/g;
1208cf655043SAndy Whitcroft	$stmt =~ s/^\s*{//;
1209cf655043SAndy Whitcroft	$stmt =~ s/}\s*$//;
1210cf655043SAndy Whitcroft	$stmt =~ s/^\s*//;
1211cf655043SAndy Whitcroft	$stmt =~ s/\s*$//;
1212cf655043SAndy Whitcroft
1213cf655043SAndy Whitcroft	my @stmt_lines = ($stmt =~ /\n/g);
1214cf655043SAndy Whitcroft	my @stmt_statements = ($stmt =~ /;/g);
1215cf655043SAndy Whitcroft
1216cf655043SAndy Whitcroft	my $stmt_lines = $#stmt_lines + 2;
1217cf655043SAndy Whitcroft	my $stmt_statements = $#stmt_statements + 1;
1218cf655043SAndy Whitcroft
1219cf655043SAndy Whitcroft	if ($stmt_lines > $stmt_statements) {
1220cf655043SAndy Whitcroft		return $stmt_lines;
1221cf655043SAndy Whitcroft	} else {
1222cf655043SAndy Whitcroft		return $stmt_statements;
1223cf655043SAndy Whitcroft	}
1224cf655043SAndy Whitcroft}
1225cf655043SAndy Whitcroft
122613214adfSAndy Whitcroftsub ctx_statement_full {
122713214adfSAndy Whitcroft	my ($linenr, $remain, $off) = @_;
122813214adfSAndy Whitcroft	my ($statement, $condition, $level);
122913214adfSAndy Whitcroft
123013214adfSAndy Whitcroft	my (@chunks);
123113214adfSAndy Whitcroft
1232cf655043SAndy Whitcroft	# Grab the first conditional/block pair.
123313214adfSAndy Whitcroft	($statement, $condition, $linenr, $remain, $off, $level) =
123413214adfSAndy Whitcroft				ctx_statement_block($linenr, $remain, $off);
1235773647a0SAndy Whitcroft	#print "F: c<$condition> s<$statement> remain<$remain>\n";
123613214adfSAndy Whitcroft	push(@chunks, [ $condition, $statement ]);
1237cf655043SAndy Whitcroft	if (!($remain > 0 && $condition =~ /^\s*(?:\n[+-])?\s*(?:if|else|do)\b/s)) {
1238cf655043SAndy Whitcroft		return ($level, $linenr, @chunks);
1239cf655043SAndy Whitcroft	}
1240cf655043SAndy Whitcroft
1241cf655043SAndy Whitcroft	# Pull in the following conditional/block pairs and see if they
1242cf655043SAndy Whitcroft	# could continue the statement.
1243cf655043SAndy Whitcroft	for (;;) {
124413214adfSAndy Whitcroft		($statement, $condition, $linenr, $remain, $off, $level) =
124513214adfSAndy Whitcroft				ctx_statement_block($linenr, $remain, $off);
1246cf655043SAndy Whitcroft		#print "C: c<$condition> s<$statement> remain<$remain>\n";
1247773647a0SAndy Whitcroft		last if (!($remain > 0 && $condition =~ /^(?:\s*\n[+-])*\s*(?:else|do)\b/s));
1248cf655043SAndy Whitcroft		#print "C: push\n";
1249cf655043SAndy Whitcroft		push(@chunks, [ $condition, $statement ]);
125013214adfSAndy Whitcroft	}
125113214adfSAndy Whitcroft
125213214adfSAndy Whitcroft	return ($level, $linenr, @chunks);
12538905a67cSAndy Whitcroft}
12548905a67cSAndy Whitcroft
12554a0df2efSAndy Whitcroftsub ctx_block_get {
1256f0a594c1SAndy Whitcroft	my ($linenr, $remain, $outer, $open, $close, $off) = @_;
12574a0df2efSAndy Whitcroft	my $line;
12584a0df2efSAndy Whitcroft	my $start = $linenr - 1;
12594a0df2efSAndy Whitcroft	my $blk = '';
12604a0df2efSAndy Whitcroft	my @o;
12614a0df2efSAndy Whitcroft	my @c;
12624a0df2efSAndy Whitcroft	my @res = ();
12634a0df2efSAndy Whitcroft
1264f0a594c1SAndy Whitcroft	my $level = 0;
12654635f4fbSAndy Whitcroft	my @stack = ($level);
126600df344fSAndy Whitcroft	for ($line = $start; $remain > 0; $line++) {
126700df344fSAndy Whitcroft		next if ($rawlines[$line] =~ /^-/);
126800df344fSAndy Whitcroft		$remain--;
126900df344fSAndy Whitcroft
127000df344fSAndy Whitcroft		$blk .= $rawlines[$line];
12714635f4fbSAndy Whitcroft
12724635f4fbSAndy Whitcroft		# Handle nested #if/#else.
127301464f30SAndy Whitcroft		if ($lines[$line] =~ /^.\s*#\s*(?:ifndef|ifdef|if)\s/) {
12744635f4fbSAndy Whitcroft			push(@stack, $level);
127501464f30SAndy Whitcroft		} elsif ($lines[$line] =~ /^.\s*#\s*(?:else|elif)\b/) {
12764635f4fbSAndy Whitcroft			$level = $stack[$#stack - 1];
127701464f30SAndy Whitcroft		} elsif ($lines[$line] =~ /^.\s*#\s*endif\b/) {
12784635f4fbSAndy Whitcroft			$level = pop(@stack);
12794635f4fbSAndy Whitcroft		}
12804635f4fbSAndy Whitcroft
128101464f30SAndy Whitcroft		foreach my $c (split(//, $lines[$line])) {
1282f0a594c1SAndy Whitcroft			##print "C<$c>L<$level><$open$close>O<$off>\n";
1283f0a594c1SAndy Whitcroft			if ($off > 0) {
1284f0a594c1SAndy Whitcroft				$off--;
1285f0a594c1SAndy Whitcroft				next;
1286f0a594c1SAndy Whitcroft			}
12874a0df2efSAndy Whitcroft
1288f0a594c1SAndy Whitcroft			if ($c eq $close && $level > 0) {
1289f0a594c1SAndy Whitcroft				$level--;
1290f0a594c1SAndy Whitcroft				last if ($level == 0);
1291f0a594c1SAndy Whitcroft			} elsif ($c eq $open) {
1292f0a594c1SAndy Whitcroft				$level++;
1293f0a594c1SAndy Whitcroft			}
1294f0a594c1SAndy Whitcroft		}
12954a0df2efSAndy Whitcroft
1296f0a594c1SAndy Whitcroft		if (!$outer || $level <= 1) {
129700df344fSAndy Whitcroft			push(@res, $rawlines[$line]);
12984a0df2efSAndy Whitcroft		}
12994a0df2efSAndy Whitcroft
1300f0a594c1SAndy Whitcroft		last if ($level == 0);
13014a0df2efSAndy Whitcroft	}
13024a0df2efSAndy Whitcroft
1303f0a594c1SAndy Whitcroft	return ($level, @res);
13044a0df2efSAndy Whitcroft}
13054a0df2efSAndy Whitcroftsub ctx_block_outer {
13064a0df2efSAndy Whitcroft	my ($linenr, $remain) = @_;
13074a0df2efSAndy Whitcroft
1308f0a594c1SAndy Whitcroft	my ($level, @r) = ctx_block_get($linenr, $remain, 1, '{', '}', 0);
1309f0a594c1SAndy Whitcroft	return @r;
13104a0df2efSAndy Whitcroft}
13114a0df2efSAndy Whitcroftsub ctx_block {
13124a0df2efSAndy Whitcroft	my ($linenr, $remain) = @_;
13134a0df2efSAndy Whitcroft
1314f0a594c1SAndy Whitcroft	my ($level, @r) = ctx_block_get($linenr, $remain, 0, '{', '}', 0);
1315f0a594c1SAndy Whitcroft	return @r;
1316653d4876SAndy Whitcroft}
1317653d4876SAndy Whitcroftsub ctx_statement {
1318f0a594c1SAndy Whitcroft	my ($linenr, $remain, $off) = @_;
1319f0a594c1SAndy Whitcroft
1320f0a594c1SAndy Whitcroft	my ($level, @r) = ctx_block_get($linenr, $remain, 0, '(', ')', $off);
1321f0a594c1SAndy Whitcroft	return @r;
1322f0a594c1SAndy Whitcroft}
1323f0a594c1SAndy Whitcroftsub ctx_block_level {
1324653d4876SAndy Whitcroft	my ($linenr, $remain) = @_;
1325653d4876SAndy Whitcroft
1326f0a594c1SAndy Whitcroft	return ctx_block_get($linenr, $remain, 0, '{', '}', 0);
13274a0df2efSAndy Whitcroft}
13289c0ca6f9SAndy Whitcroftsub ctx_statement_level {
13299c0ca6f9SAndy Whitcroft	my ($linenr, $remain, $off) = @_;
13309c0ca6f9SAndy Whitcroft
13319c0ca6f9SAndy Whitcroft	return ctx_block_get($linenr, $remain, 0, '(', ')', $off);
13329c0ca6f9SAndy Whitcroft}
13334a0df2efSAndy Whitcroft
13344a0df2efSAndy Whitcroftsub ctx_locate_comment {
13354a0df2efSAndy Whitcroft	my ($first_line, $end_line) = @_;
13364a0df2efSAndy Whitcroft
13374a0df2efSAndy Whitcroft	# Catch a comment on the end of the line itself.
1338beae6332SAndy Whitcroft	my ($current_comment) = ($rawlines[$end_line - 1] =~ m@.*(/\*.*\*/)\s*(?:\\\s*)?$@);
13394a0df2efSAndy Whitcroft	return $current_comment if (defined $current_comment);
13404a0df2efSAndy Whitcroft
13414a0df2efSAndy Whitcroft	# Look through the context and try and figure out if there is a
13424a0df2efSAndy Whitcroft	# comment.
13434a0df2efSAndy Whitcroft	my $in_comment = 0;
13444a0df2efSAndy Whitcroft	$current_comment = '';
13454a0df2efSAndy Whitcroft	for (my $linenr = $first_line; $linenr < $end_line; $linenr++) {
134600df344fSAndy Whitcroft		my $line = $rawlines[$linenr - 1];
134700df344fSAndy Whitcroft		#warn "           $line\n";
13484a0df2efSAndy Whitcroft		if ($linenr == $first_line and $line =~ m@^.\s*\*@) {
13494a0df2efSAndy Whitcroft			$in_comment = 1;
13504a0df2efSAndy Whitcroft		}
13514a0df2efSAndy Whitcroft		if ($line =~ m@/\*@) {
13524a0df2efSAndy Whitcroft			$in_comment = 1;
13534a0df2efSAndy Whitcroft		}
13544a0df2efSAndy Whitcroft		if (!$in_comment && $current_comment ne '') {
13554a0df2efSAndy Whitcroft			$current_comment = '';
13564a0df2efSAndy Whitcroft		}
13574a0df2efSAndy Whitcroft		$current_comment .= $line . "\n" if ($in_comment);
13584a0df2efSAndy Whitcroft		if ($line =~ m@\*/@) {
13594a0df2efSAndy Whitcroft			$in_comment = 0;
13604a0df2efSAndy Whitcroft		}
13614a0df2efSAndy Whitcroft	}
13624a0df2efSAndy Whitcroft
13634a0df2efSAndy Whitcroft	chomp($current_comment);
13644a0df2efSAndy Whitcroft	return($current_comment);
13654a0df2efSAndy Whitcroft}
13664a0df2efSAndy Whitcroftsub ctx_has_comment {
13674a0df2efSAndy Whitcroft	my ($first_line, $end_line) = @_;
13684a0df2efSAndy Whitcroft	my $cmt = ctx_locate_comment($first_line, $end_line);
13694a0df2efSAndy Whitcroft
137000df344fSAndy Whitcroft	##print "LINE: $rawlines[$end_line - 1 ]\n";
13714a0df2efSAndy Whitcroft	##print "CMMT: $cmt\n";
13724a0df2efSAndy Whitcroft
13734a0df2efSAndy Whitcroft	return ($cmt ne '');
13744a0df2efSAndy Whitcroft}
13754a0df2efSAndy Whitcroft
13764d001e4dSAndy Whitcroftsub raw_line {
13774d001e4dSAndy Whitcroft	my ($linenr, $cnt) = @_;
13784d001e4dSAndy Whitcroft
13794d001e4dSAndy Whitcroft	my $offset = $linenr - 1;
13804d001e4dSAndy Whitcroft	$cnt++;
13814d001e4dSAndy Whitcroft
13824d001e4dSAndy Whitcroft	my $line;
13834d001e4dSAndy Whitcroft	while ($cnt) {
13844d001e4dSAndy Whitcroft		$line = $rawlines[$offset++];
13854d001e4dSAndy Whitcroft		next if (defined($line) && $line =~ /^-/);
13864d001e4dSAndy Whitcroft		$cnt--;
13874d001e4dSAndy Whitcroft	}
13884d001e4dSAndy Whitcroft
13894d001e4dSAndy Whitcroft	return $line;
13904d001e4dSAndy Whitcroft}
13914d001e4dSAndy Whitcroft
13920a920b5bSAndy Whitcroftsub cat_vet {
13930a920b5bSAndy Whitcroft	my ($vet) = @_;
13949c0ca6f9SAndy Whitcroft	my ($res, $coded);
13950a920b5bSAndy Whitcroft
13969c0ca6f9SAndy Whitcroft	$res = '';
13976c72ffaaSAndy Whitcroft	while ($vet =~ /([^[:cntrl:]]*)([[:cntrl:]]|$)/g) {
13986c72ffaaSAndy Whitcroft		$res .= $1;
13996c72ffaaSAndy Whitcroft		if ($2 ne '') {
14009c0ca6f9SAndy Whitcroft			$coded = sprintf("^%c", unpack('C', $2) + 64);
14016c72ffaaSAndy Whitcroft			$res .= $coded;
14026c72ffaaSAndy Whitcroft		}
14039c0ca6f9SAndy Whitcroft	}
14049c0ca6f9SAndy Whitcroft	$res =~ s/$/\$/;
14050a920b5bSAndy Whitcroft
14069c0ca6f9SAndy Whitcroft	return $res;
14070a920b5bSAndy Whitcroft}
14080a920b5bSAndy Whitcroft
1409c2fdda0dSAndy Whitcroftmy $av_preprocessor = 0;
1410cf655043SAndy Whitcroftmy $av_pending;
1411c2fdda0dSAndy Whitcroftmy @av_paren_type;
14121f65f947SAndy Whitcroftmy $av_pend_colon;
1413c2fdda0dSAndy Whitcroft
1414c2fdda0dSAndy Whitcroftsub annotate_reset {
1415c2fdda0dSAndy Whitcroft	$av_preprocessor = 0;
1416cf655043SAndy Whitcroft	$av_pending = '_';
1417cf655043SAndy Whitcroft	@av_paren_type = ('E');
14181f65f947SAndy Whitcroft	$av_pend_colon = 'O';
1419c2fdda0dSAndy Whitcroft}
1420c2fdda0dSAndy Whitcroft
14216c72ffaaSAndy Whitcroftsub annotate_values {
14226c72ffaaSAndy Whitcroft	my ($stream, $type) = @_;
14236c72ffaaSAndy Whitcroft
14246c72ffaaSAndy Whitcroft	my $res;
14251f65f947SAndy Whitcroft	my $var = '_' x length($stream);
14266c72ffaaSAndy Whitcroft	my $cur = $stream;
14276c72ffaaSAndy Whitcroft
1428c2fdda0dSAndy Whitcroft	print "$stream\n" if ($dbg_values > 1);
14296c72ffaaSAndy Whitcroft
14306c72ffaaSAndy Whitcroft	while (length($cur)) {
1431773647a0SAndy Whitcroft		@av_paren_type = ('E') if ($#av_paren_type < 0);
1432cf655043SAndy Whitcroft		print " <" . join('', @av_paren_type) .
1433171ae1a4SAndy Whitcroft				"> <$type> <$av_pending>" if ($dbg_values > 1);
14346c72ffaaSAndy Whitcroft		if ($cur =~ /^(\s+)/o) {
1435c2fdda0dSAndy Whitcroft			print "WS($1)\n" if ($dbg_values > 1);
1436c2fdda0dSAndy Whitcroft			if ($1 =~ /\n/ && $av_preprocessor) {
1437cf655043SAndy Whitcroft				$type = pop(@av_paren_type);
1438c2fdda0dSAndy Whitcroft				$av_preprocessor = 0;
14396c72ffaaSAndy Whitcroft			}
14406c72ffaaSAndy Whitcroft
1441c023e473SFlorian Mickler		} elsif ($cur =~ /^(\(\s*$Type\s*)\)/ && $av_pending eq '_') {
14429446ef56SAndy Whitcroft			print "CAST($1)\n" if ($dbg_values > 1);
14439446ef56SAndy Whitcroft			push(@av_paren_type, $type);
1444addcdceaSAndy Whitcroft			$type = 'c';
14459446ef56SAndy Whitcroft
1446e91b6e26SAndy Whitcroft		} elsif ($cur =~ /^($Type)\s*(?:$Ident|,|\)|\(|\s*$)/) {
1447c2fdda0dSAndy Whitcroft			print "DECLARE($1)\n" if ($dbg_values > 1);
14486c72ffaaSAndy Whitcroft			$type = 'T';
14496c72ffaaSAndy Whitcroft
1450389a2fe5SAndy Whitcroft		} elsif ($cur =~ /^($Modifier)\s*/) {
1451389a2fe5SAndy Whitcroft			print "MODIFIER($1)\n" if ($dbg_values > 1);
1452389a2fe5SAndy Whitcroft			$type = 'T';
1453389a2fe5SAndy Whitcroft
1454c45dcabdSAndy Whitcroft		} elsif ($cur =~ /^(\#\s*define\s*$Ident)(\(?)/o) {
1455171ae1a4SAndy Whitcroft			print "DEFINE($1,$2)\n" if ($dbg_values > 1);
1456c2fdda0dSAndy Whitcroft			$av_preprocessor = 1;
1457171ae1a4SAndy Whitcroft			push(@av_paren_type, $type);
1458171ae1a4SAndy Whitcroft			if ($2 ne '') {
1459cf655043SAndy Whitcroft				$av_pending = 'N';
1460171ae1a4SAndy Whitcroft			}
1461171ae1a4SAndy Whitcroft			$type = 'E';
1462171ae1a4SAndy Whitcroft
1463c45dcabdSAndy Whitcroft		} elsif ($cur =~ /^(\#\s*(?:undef\s*$Ident|include\b))/o) {
1464171ae1a4SAndy Whitcroft			print "UNDEF($1)\n" if ($dbg_values > 1);
1465171ae1a4SAndy Whitcroft			$av_preprocessor = 1;
1466171ae1a4SAndy Whitcroft			push(@av_paren_type, $type);
14676c72ffaaSAndy Whitcroft
1468c45dcabdSAndy Whitcroft		} elsif ($cur =~ /^(\#\s*(?:ifdef|ifndef|if))/o) {
1469cf655043SAndy Whitcroft			print "PRE_START($1)\n" if ($dbg_values > 1);
1470c2fdda0dSAndy Whitcroft			$av_preprocessor = 1;
1471cf655043SAndy Whitcroft
1472cf655043SAndy Whitcroft			push(@av_paren_type, $type);
1473cf655043SAndy Whitcroft			push(@av_paren_type, $type);
1474171ae1a4SAndy Whitcroft			$type = 'E';
1475cf655043SAndy Whitcroft
1476c45dcabdSAndy Whitcroft		} elsif ($cur =~ /^(\#\s*(?:else|elif))/o) {
1477cf655043SAndy Whitcroft			print "PRE_RESTART($1)\n" if ($dbg_values > 1);
1478cf655043SAndy Whitcroft			$av_preprocessor = 1;
1479cf655043SAndy Whitcroft
1480cf655043SAndy Whitcroft			push(@av_paren_type, $av_paren_type[$#av_paren_type]);
1481cf655043SAndy Whitcroft
1482171ae1a4SAndy Whitcroft			$type = 'E';
1483cf655043SAndy Whitcroft
1484c45dcabdSAndy Whitcroft		} elsif ($cur =~ /^(\#\s*(?:endif))/o) {
1485cf655043SAndy Whitcroft			print "PRE_END($1)\n" if ($dbg_values > 1);
1486cf655043SAndy Whitcroft
1487cf655043SAndy Whitcroft			$av_preprocessor = 1;
1488cf655043SAndy Whitcroft
1489cf655043SAndy Whitcroft			# Assume all arms of the conditional end as this
1490cf655043SAndy Whitcroft			# one does, and continue as if the #endif was not here.
1491cf655043SAndy Whitcroft			pop(@av_paren_type);
1492cf655043SAndy Whitcroft			push(@av_paren_type, $type);
1493171ae1a4SAndy Whitcroft			$type = 'E';
14946c72ffaaSAndy Whitcroft
14956c72ffaaSAndy Whitcroft		} elsif ($cur =~ /^(\\\n)/o) {
1496c2fdda0dSAndy Whitcroft			print "PRECONT($1)\n" if ($dbg_values > 1);
14976c72ffaaSAndy Whitcroft
1498171ae1a4SAndy Whitcroft		} elsif ($cur =~ /^(__attribute__)\s*\(?/o) {
1499171ae1a4SAndy Whitcroft			print "ATTR($1)\n" if ($dbg_values > 1);
1500171ae1a4SAndy Whitcroft			$av_pending = $type;
1501171ae1a4SAndy Whitcroft			$type = 'N';
1502171ae1a4SAndy Whitcroft
15036c72ffaaSAndy Whitcroft		} elsif ($cur =~ /^(sizeof)\s*(\()?/o) {
1504c2fdda0dSAndy Whitcroft			print "SIZEOF($1)\n" if ($dbg_values > 1);
15056c72ffaaSAndy Whitcroft			if (defined $2) {
1506cf655043SAndy Whitcroft				$av_pending = 'V';
15076c72ffaaSAndy Whitcroft			}
15086c72ffaaSAndy Whitcroft			$type = 'N';
15096c72ffaaSAndy Whitcroft
151014b111c1SAndy Whitcroft		} elsif ($cur =~ /^(if|while|for)\b/o) {
1511c2fdda0dSAndy Whitcroft			print "COND($1)\n" if ($dbg_values > 1);
151214b111c1SAndy Whitcroft			$av_pending = 'E';
15136c72ffaaSAndy Whitcroft			$type = 'N';
15146c72ffaaSAndy Whitcroft
15151f65f947SAndy Whitcroft		} elsif ($cur =~/^(case)/o) {
15161f65f947SAndy Whitcroft			print "CASE($1)\n" if ($dbg_values > 1);
15171f65f947SAndy Whitcroft			$av_pend_colon = 'C';
15181f65f947SAndy Whitcroft			$type = 'N';
15191f65f947SAndy Whitcroft
152014b111c1SAndy Whitcroft		} elsif ($cur =~/^(return|else|goto|typeof|__typeof__)\b/o) {
1521c2fdda0dSAndy Whitcroft			print "KEYWORD($1)\n" if ($dbg_values > 1);
15226c72ffaaSAndy Whitcroft			$type = 'N';
15236c72ffaaSAndy Whitcroft
15246c72ffaaSAndy Whitcroft		} elsif ($cur =~ /^(\()/o) {
1525c2fdda0dSAndy Whitcroft			print "PAREN('$1')\n" if ($dbg_values > 1);
1526cf655043SAndy Whitcroft			push(@av_paren_type, $av_pending);
1527cf655043SAndy Whitcroft			$av_pending = '_';
15286c72ffaaSAndy Whitcroft			$type = 'N';
15296c72ffaaSAndy Whitcroft
15306c72ffaaSAndy Whitcroft		} elsif ($cur =~ /^(\))/o) {
1531cf655043SAndy Whitcroft			my $new_type = pop(@av_paren_type);
1532cf655043SAndy Whitcroft			if ($new_type ne '_') {
1533cf655043SAndy Whitcroft				$type = $new_type;
1534c2fdda0dSAndy Whitcroft				print "PAREN('$1') -> $type\n"
1535c2fdda0dSAndy Whitcroft							if ($dbg_values > 1);
15366c72ffaaSAndy Whitcroft			} else {
1537c2fdda0dSAndy Whitcroft				print "PAREN('$1')\n" if ($dbg_values > 1);
15386c72ffaaSAndy Whitcroft			}
15396c72ffaaSAndy Whitcroft
1540c8cb2ca3SAndy Whitcroft		} elsif ($cur =~ /^($Ident)\s*\(/o) {
1541c2fdda0dSAndy Whitcroft			print "FUNC($1)\n" if ($dbg_values > 1);
1542c8cb2ca3SAndy Whitcroft			$type = 'V';
1543cf655043SAndy Whitcroft			$av_pending = 'V';
15446c72ffaaSAndy Whitcroft
15458e761b04SAndy Whitcroft		} elsif ($cur =~ /^($Ident\s*):(?:\s*\d+\s*(,|=|;))?/) {
15468e761b04SAndy Whitcroft			if (defined $2 && $type eq 'C' || $type eq 'T') {
15471f65f947SAndy Whitcroft				$av_pend_colon = 'B';
15488e761b04SAndy Whitcroft			} elsif ($type eq 'E') {
15498e761b04SAndy Whitcroft				$av_pend_colon = 'L';
15501f65f947SAndy Whitcroft			}
15511f65f947SAndy Whitcroft			print "IDENT_COLON($1,$type>$av_pend_colon)\n" if ($dbg_values > 1);
15521f65f947SAndy Whitcroft			$type = 'V';
15531f65f947SAndy Whitcroft
15546c72ffaaSAndy Whitcroft		} elsif ($cur =~ /^($Ident|$Constant)/o) {
1555c2fdda0dSAndy Whitcroft			print "IDENT($1)\n" if ($dbg_values > 1);
15566c72ffaaSAndy Whitcroft			$type = 'V';
15576c72ffaaSAndy Whitcroft
15586c72ffaaSAndy Whitcroft		} elsif ($cur =~ /^($Assignment)/o) {
1559c2fdda0dSAndy Whitcroft			print "ASSIGN($1)\n" if ($dbg_values > 1);
15606c72ffaaSAndy Whitcroft			$type = 'N';
15616c72ffaaSAndy Whitcroft
1562cf655043SAndy Whitcroft		} elsif ($cur =~/^(;|{|})/) {
1563c2fdda0dSAndy Whitcroft			print "END($1)\n" if ($dbg_values > 1);
156413214adfSAndy Whitcroft			$type = 'E';
15651f65f947SAndy Whitcroft			$av_pend_colon = 'O';
156613214adfSAndy Whitcroft
15678e761b04SAndy Whitcroft		} elsif ($cur =~/^(,)/) {
15688e761b04SAndy Whitcroft			print "COMMA($1)\n" if ($dbg_values > 1);
15698e761b04SAndy Whitcroft			$type = 'C';
15708e761b04SAndy Whitcroft
15711f65f947SAndy Whitcroft		} elsif ($cur =~ /^(\?)/o) {
15721f65f947SAndy Whitcroft			print "QUESTION($1)\n" if ($dbg_values > 1);
15731f65f947SAndy Whitcroft			$type = 'N';
15741f65f947SAndy Whitcroft
15751f65f947SAndy Whitcroft		} elsif ($cur =~ /^(:)/o) {
15761f65f947SAndy Whitcroft			print "COLON($1,$av_pend_colon)\n" if ($dbg_values > 1);
15771f65f947SAndy Whitcroft
15781f65f947SAndy Whitcroft			substr($var, length($res), 1, $av_pend_colon);
15791f65f947SAndy Whitcroft			if ($av_pend_colon eq 'C' || $av_pend_colon eq 'L') {
15801f65f947SAndy Whitcroft				$type = 'E';
15811f65f947SAndy Whitcroft			} else {
15821f65f947SAndy Whitcroft				$type = 'N';
15831f65f947SAndy Whitcroft			}
15841f65f947SAndy Whitcroft			$av_pend_colon = 'O';
15851f65f947SAndy Whitcroft
15868e761b04SAndy Whitcroft		} elsif ($cur =~ /^(\[)/o) {
158713214adfSAndy Whitcroft			print "CLOSE($1)\n" if ($dbg_values > 1);
15886c72ffaaSAndy Whitcroft			$type = 'N';
15896c72ffaaSAndy Whitcroft
15900d413866SAndy Whitcroft		} elsif ($cur =~ /^(-(?![->])|\+(?!\+)|\*|\&\&|\&)/o) {
159174048ed8SAndy Whitcroft			my $variant;
159274048ed8SAndy Whitcroft
159374048ed8SAndy Whitcroft			print "OPV($1)\n" if ($dbg_values > 1);
159474048ed8SAndy Whitcroft			if ($type eq 'V') {
159574048ed8SAndy Whitcroft				$variant = 'B';
159674048ed8SAndy Whitcroft			} else {
159774048ed8SAndy Whitcroft				$variant = 'U';
159874048ed8SAndy Whitcroft			}
159974048ed8SAndy Whitcroft
160074048ed8SAndy Whitcroft			substr($var, length($res), 1, $variant);
160174048ed8SAndy Whitcroft			$type = 'N';
160274048ed8SAndy Whitcroft
16036c72ffaaSAndy Whitcroft		} elsif ($cur =~ /^($Operators)/o) {
1604c2fdda0dSAndy Whitcroft			print "OP($1)\n" if ($dbg_values > 1);
16056c72ffaaSAndy Whitcroft			if ($1 ne '++' && $1 ne '--') {
16066c72ffaaSAndy Whitcroft				$type = 'N';
16076c72ffaaSAndy Whitcroft			}
16086c72ffaaSAndy Whitcroft
16096c72ffaaSAndy Whitcroft		} elsif ($cur =~ /(^.)/o) {
1610c2fdda0dSAndy Whitcroft			print "C($1)\n" if ($dbg_values > 1);
16116c72ffaaSAndy Whitcroft		}
16126c72ffaaSAndy Whitcroft		if (defined $1) {
16136c72ffaaSAndy Whitcroft			$cur = substr($cur, length($1));
16146c72ffaaSAndy Whitcroft			$res .= $type x length($1);
16156c72ffaaSAndy Whitcroft		}
16166c72ffaaSAndy Whitcroft	}
16176c72ffaaSAndy Whitcroft
16181f65f947SAndy Whitcroft	return ($res, $var);
16196c72ffaaSAndy Whitcroft}
16206c72ffaaSAndy Whitcroft
16218905a67cSAndy Whitcroftsub possible {
162213214adfSAndy Whitcroft	my ($possible, $line) = @_;
16239a974fdbSAndy Whitcroft	my $notPermitted = qr{(?:
16240776e594SAndy Whitcroft		^(?:
16250776e594SAndy Whitcroft			$Modifier|
16260776e594SAndy Whitcroft			$Storage|
16270776e594SAndy Whitcroft			$Type|
16289a974fdbSAndy Whitcroft			DEFINE_\S+
16299a974fdbSAndy Whitcroft		)$|
16309a974fdbSAndy Whitcroft		^(?:
16310776e594SAndy Whitcroft			goto|
16320776e594SAndy Whitcroft			return|
16330776e594SAndy Whitcroft			case|
16340776e594SAndy Whitcroft			else|
16350776e594SAndy Whitcroft			asm|__asm__|
163689a88353SAndy Whitcroft			do|
163789a88353SAndy Whitcroft			\#|
163889a88353SAndy Whitcroft			\#\#|
16399a974fdbSAndy Whitcroft		)(?:\s|$)|
16400776e594SAndy Whitcroft		^(?:typedef|struct|enum)\b
16419a974fdbSAndy Whitcroft	    )}x;
16429a974fdbSAndy Whitcroft	warn "CHECK<$possible> ($line)\n" if ($dbg_possible > 2);
16439a974fdbSAndy Whitcroft	if ($possible !~ $notPermitted) {
1644c45dcabdSAndy Whitcroft		# Check for modifiers.
1645c45dcabdSAndy Whitcroft		$possible =~ s/\s*$Storage\s*//g;
1646c45dcabdSAndy Whitcroft		$possible =~ s/\s*$Sparse\s*//g;
1647c45dcabdSAndy Whitcroft		if ($possible =~ /^\s*$/) {
1648c45dcabdSAndy Whitcroft
1649c45dcabdSAndy Whitcroft		} elsif ($possible =~ /\s/) {
1650c45dcabdSAndy Whitcroft			$possible =~ s/\s*$Type\s*//g;
1651d2506586SAndy Whitcroft			for my $modifier (split(' ', $possible)) {
16529a974fdbSAndy Whitcroft				if ($modifier !~ $notPermitted) {
1653d2506586SAndy Whitcroft					warn "MODIFIER: $modifier ($possible) ($line)\n" if ($dbg_possible);
1654485ff23eSAlex Dowad					push(@modifierListFile, $modifier);
1655d2506586SAndy Whitcroft				}
16569a974fdbSAndy Whitcroft			}
1657c45dcabdSAndy Whitcroft
1658c45dcabdSAndy Whitcroft		} else {
165913214adfSAndy Whitcroft			warn "POSSIBLE: $possible ($line)\n" if ($dbg_possible);
1660485ff23eSAlex Dowad			push(@typeListFile, $possible);
1661c45dcabdSAndy Whitcroft		}
16628905a67cSAndy Whitcroft		build_types();
16630776e594SAndy Whitcroft	} else {
16640776e594SAndy Whitcroft		warn "NOTPOSS: $possible ($line)\n" if ($dbg_possible > 1);
16658905a67cSAndy Whitcroft	}
16668905a67cSAndy Whitcroft}
16678905a67cSAndy Whitcroft
16686c72ffaaSAndy Whitcroftmy $prefix = '';
16696c72ffaaSAndy Whitcroft
1670000d1cc1SJoe Perchessub show_type {
1671cbec18afSJoe Perches	my ($type) = @_;
167291bfe484SJoe Perches
1673cbec18afSJoe Perches	return defined $use_type{$type} if (scalar keys %use_type > 0);
1674cbec18afSJoe Perches
1675cbec18afSJoe Perches	return !defined $ignore_type{$type};
1676000d1cc1SJoe Perches}
1677000d1cc1SJoe Perches
1678f0a594c1SAndy Whitcroftsub report {
1679cbec18afSJoe Perches	my ($level, $type, $msg) = @_;
1680cbec18afSJoe Perches
1681cbec18afSJoe Perches	if (!show_type($type) ||
1682cbec18afSJoe Perches	    (defined $tst_only && $msg !~ /\Q$tst_only\E/)) {
1683773647a0SAndy Whitcroft		return 0;
1684773647a0SAndy Whitcroft	}
168557230297SJoe Perches	my $output = '';
168657230297SJoe Perches	if (-t STDOUT && $color) {
168757230297SJoe Perches		if ($level eq 'ERROR') {
168857230297SJoe Perches			$output .= RED;
168957230297SJoe Perches		} elsif ($level eq 'WARNING') {
169057230297SJoe Perches			$output .= YELLOW;
1691000d1cc1SJoe Perches		} else {
169257230297SJoe Perches			$output .= GREEN;
1693000d1cc1SJoe Perches		}
169457230297SJoe Perches	}
169557230297SJoe Perches	$output .= $prefix . $level . ':';
169657230297SJoe Perches	if ($show_types) {
169757230297SJoe Perches		$output .= BLUE if (-t STDOUT && $color);
169857230297SJoe Perches		$output .= "$type:";
169957230297SJoe Perches	}
170057230297SJoe Perches	$output .= RESET if (-t STDOUT && $color);
170157230297SJoe Perches	$output .= ' ' . $msg . "\n";
170234d8815fSJoe Perches
170334d8815fSJoe Perches	if ($showfile) {
170434d8815fSJoe Perches		my @lines = split("\n", $output, -1);
170534d8815fSJoe Perches		splice(@lines, 1, 1);
170634d8815fSJoe Perches		$output = join("\n", @lines);
170734d8815fSJoe Perches	}
170857230297SJoe Perches	$output = (split('\n', $output))[0] . "\n" if ($terse);
17098905a67cSAndy Whitcroft
171057230297SJoe Perches	push(our @report, $output);
1711773647a0SAndy Whitcroft
1712773647a0SAndy Whitcroft	return 1;
1713f0a594c1SAndy Whitcroft}
1714cbec18afSJoe Perches
1715f0a594c1SAndy Whitcroftsub report_dump {
171613214adfSAndy Whitcroft	our @report;
1717f0a594c1SAndy Whitcroft}
1718000d1cc1SJoe Perches
1719d752fcc8SJoe Perchessub fixup_current_range {
1720d752fcc8SJoe Perches	my ($lineRef, $offset, $length) = @_;
1721d752fcc8SJoe Perches
1722d752fcc8SJoe Perches	if ($$lineRef =~ /^\@\@ -\d+,\d+ \+(\d+),(\d+) \@\@/) {
1723d752fcc8SJoe Perches		my $o = $1;
1724d752fcc8SJoe Perches		my $l = $2;
1725d752fcc8SJoe Perches		my $no = $o + $offset;
1726d752fcc8SJoe Perches		my $nl = $l + $length;
1727d752fcc8SJoe Perches		$$lineRef =~ s/\+$o,$l \@\@/\+$no,$nl \@\@/;
1728d752fcc8SJoe Perches	}
1729d752fcc8SJoe Perches}
1730d752fcc8SJoe Perches
1731d752fcc8SJoe Perchessub fix_inserted_deleted_lines {
1732d752fcc8SJoe Perches	my ($linesRef, $insertedRef, $deletedRef) = @_;
1733d752fcc8SJoe Perches
1734d752fcc8SJoe Perches	my $range_last_linenr = 0;
1735d752fcc8SJoe Perches	my $delta_offset = 0;
1736d752fcc8SJoe Perches
1737d752fcc8SJoe Perches	my $old_linenr = 0;
1738d752fcc8SJoe Perches	my $new_linenr = 0;
1739d752fcc8SJoe Perches
1740d752fcc8SJoe Perches	my $next_insert = 0;
1741d752fcc8SJoe Perches	my $next_delete = 0;
1742d752fcc8SJoe Perches
1743d752fcc8SJoe Perches	my @lines = ();
1744d752fcc8SJoe Perches
1745d752fcc8SJoe Perches	my $inserted = @{$insertedRef}[$next_insert++];
1746d752fcc8SJoe Perches	my $deleted = @{$deletedRef}[$next_delete++];
1747d752fcc8SJoe Perches
1748d752fcc8SJoe Perches	foreach my $old_line (@{$linesRef}) {
1749d752fcc8SJoe Perches		my $save_line = 1;
1750d752fcc8SJoe Perches		my $line = $old_line;	#don't modify the array
1751323b267fSJoe Perches		if ($line =~ /^(?:\+\+\+|\-\-\-)\s+\S+/) {	#new filename
1752d752fcc8SJoe Perches			$delta_offset = 0;
1753d752fcc8SJoe Perches		} elsif ($line =~ /^\@\@ -\d+,\d+ \+\d+,\d+ \@\@/) {	#new hunk
1754d752fcc8SJoe Perches			$range_last_linenr = $new_linenr;
1755d752fcc8SJoe Perches			fixup_current_range(\$line, $delta_offset, 0);
1756d752fcc8SJoe Perches		}
1757d752fcc8SJoe Perches
1758d752fcc8SJoe Perches		while (defined($deleted) && ${$deleted}{'LINENR'} == $old_linenr) {
1759d752fcc8SJoe Perches			$deleted = @{$deletedRef}[$next_delete++];
1760d752fcc8SJoe Perches			$save_line = 0;
1761d752fcc8SJoe Perches			fixup_current_range(\$lines[$range_last_linenr], $delta_offset--, -1);
1762d752fcc8SJoe Perches		}
1763d752fcc8SJoe Perches
1764d752fcc8SJoe Perches		while (defined($inserted) && ${$inserted}{'LINENR'} == $old_linenr) {
1765d752fcc8SJoe Perches			push(@lines, ${$inserted}{'LINE'});
1766d752fcc8SJoe Perches			$inserted = @{$insertedRef}[$next_insert++];
1767d752fcc8SJoe Perches			$new_linenr++;
1768d752fcc8SJoe Perches			fixup_current_range(\$lines[$range_last_linenr], $delta_offset++, 1);
1769d752fcc8SJoe Perches		}
1770d752fcc8SJoe Perches
1771d752fcc8SJoe Perches		if ($save_line) {
1772d752fcc8SJoe Perches			push(@lines, $line);
1773d752fcc8SJoe Perches			$new_linenr++;
1774d752fcc8SJoe Perches		}
1775d752fcc8SJoe Perches
1776d752fcc8SJoe Perches		$old_linenr++;
1777d752fcc8SJoe Perches	}
1778d752fcc8SJoe Perches
1779d752fcc8SJoe Perches	return @lines;
1780d752fcc8SJoe Perches}
1781d752fcc8SJoe Perches
1782f2d7e4d4SJoe Perchessub fix_insert_line {
1783f2d7e4d4SJoe Perches	my ($linenr, $line) = @_;
1784f2d7e4d4SJoe Perches
1785f2d7e4d4SJoe Perches	my $inserted = {
1786f2d7e4d4SJoe Perches		LINENR => $linenr,
1787f2d7e4d4SJoe Perches		LINE => $line,
1788f2d7e4d4SJoe Perches	};
1789f2d7e4d4SJoe Perches	push(@fixed_inserted, $inserted);
1790f2d7e4d4SJoe Perches}
1791f2d7e4d4SJoe Perches
1792f2d7e4d4SJoe Perchessub fix_delete_line {
1793f2d7e4d4SJoe Perches	my ($linenr, $line) = @_;
1794f2d7e4d4SJoe Perches
1795f2d7e4d4SJoe Perches	my $deleted = {
1796f2d7e4d4SJoe Perches		LINENR => $linenr,
1797f2d7e4d4SJoe Perches		LINE => $line,
1798f2d7e4d4SJoe Perches	};
1799f2d7e4d4SJoe Perches
1800f2d7e4d4SJoe Perches	push(@fixed_deleted, $deleted);
1801f2d7e4d4SJoe Perches}
1802f2d7e4d4SJoe Perches
1803de7d4f0eSAndy Whitcroftsub ERROR {
1804cbec18afSJoe Perches	my ($type, $msg) = @_;
1805cbec18afSJoe Perches
1806cbec18afSJoe Perches	if (report("ERROR", $type, $msg)) {
1807de7d4f0eSAndy Whitcroft		our $clean = 0;
18086c72ffaaSAndy Whitcroft		our $cnt_error++;
18093705ce5bSJoe Perches		return 1;
1810de7d4f0eSAndy Whitcroft	}
18113705ce5bSJoe Perches	return 0;
1812773647a0SAndy Whitcroft}
1813de7d4f0eSAndy Whitcroftsub WARN {
1814cbec18afSJoe Perches	my ($type, $msg) = @_;
1815cbec18afSJoe Perches
1816cbec18afSJoe Perches	if (report("WARNING", $type, $msg)) {
1817de7d4f0eSAndy Whitcroft		our $clean = 0;
18186c72ffaaSAndy Whitcroft		our $cnt_warn++;
18193705ce5bSJoe Perches		return 1;
1820de7d4f0eSAndy Whitcroft	}
18213705ce5bSJoe Perches	return 0;
1822773647a0SAndy Whitcroft}
1823de7d4f0eSAndy Whitcroftsub CHK {
1824cbec18afSJoe Perches	my ($type, $msg) = @_;
1825cbec18afSJoe Perches
1826cbec18afSJoe Perches	if ($check && report("CHECK", $type, $msg)) {
1827de7d4f0eSAndy Whitcroft		our $clean = 0;
18286c72ffaaSAndy Whitcroft		our $cnt_chk++;
18293705ce5bSJoe Perches		return 1;
18306c72ffaaSAndy Whitcroft	}
18313705ce5bSJoe Perches	return 0;
1832de7d4f0eSAndy Whitcroft}
1833de7d4f0eSAndy Whitcroft
18346ecd9674SAndy Whitcroftsub check_absolute_file {
18356ecd9674SAndy Whitcroft	my ($absolute, $herecurr) = @_;
18366ecd9674SAndy Whitcroft	my $file = $absolute;
18376ecd9674SAndy Whitcroft
18386ecd9674SAndy Whitcroft	##print "absolute<$absolute>\n";
18396ecd9674SAndy Whitcroft
18406ecd9674SAndy Whitcroft	# See if any suffix of this path is a path within the tree.
18416ecd9674SAndy Whitcroft	while ($file =~ s@^[^/]*/@@) {
18426ecd9674SAndy Whitcroft		if (-f "$root/$file") {
18436ecd9674SAndy Whitcroft			##print "file<$file>\n";
18446ecd9674SAndy Whitcroft			last;
18456ecd9674SAndy Whitcroft		}
18466ecd9674SAndy Whitcroft	}
18476ecd9674SAndy Whitcroft	if (! -f _)  {
18486ecd9674SAndy Whitcroft		return 0;
18496ecd9674SAndy Whitcroft	}
18506ecd9674SAndy Whitcroft
18516ecd9674SAndy Whitcroft	# It is, so see if the prefix is acceptable.
18526ecd9674SAndy Whitcroft	my $prefix = $absolute;
18536ecd9674SAndy Whitcroft	substr($prefix, -length($file)) = '';
18546ecd9674SAndy Whitcroft
18556ecd9674SAndy Whitcroft	##print "prefix<$prefix>\n";
18566ecd9674SAndy Whitcroft	if ($prefix ne ".../") {
1857000d1cc1SJoe Perches		WARN("USE_RELATIVE_PATH",
1858000d1cc1SJoe Perches		     "use relative pathname instead of absolute in changelog text\n" . $herecurr);
18596ecd9674SAndy Whitcroft	}
18606ecd9674SAndy Whitcroft}
18616ecd9674SAndy Whitcroft
18623705ce5bSJoe Perchessub trim {
18633705ce5bSJoe Perches	my ($string) = @_;
18643705ce5bSJoe Perches
1865b34c648bSJoe Perches	$string =~ s/^\s+|\s+$//g;
1866b34c648bSJoe Perches
1867b34c648bSJoe Perches	return $string;
1868b34c648bSJoe Perches}
1869b34c648bSJoe Perches
1870b34c648bSJoe Perchessub ltrim {
1871b34c648bSJoe Perches	my ($string) = @_;
1872b34c648bSJoe Perches
1873b34c648bSJoe Perches	$string =~ s/^\s+//;
1874b34c648bSJoe Perches
1875b34c648bSJoe Perches	return $string;
1876b34c648bSJoe Perches}
1877b34c648bSJoe Perches
1878b34c648bSJoe Perchessub rtrim {
1879b34c648bSJoe Perches	my ($string) = @_;
1880b34c648bSJoe Perches
1881b34c648bSJoe Perches	$string =~ s/\s+$//;
18823705ce5bSJoe Perches
18833705ce5bSJoe Perches	return $string;
18843705ce5bSJoe Perches}
18853705ce5bSJoe Perches
188652ea8506SJoe Perchessub string_find_replace {
188752ea8506SJoe Perches	my ($string, $find, $replace) = @_;
188852ea8506SJoe Perches
188952ea8506SJoe Perches	$string =~ s/$find/$replace/g;
189052ea8506SJoe Perches
189152ea8506SJoe Perches	return $string;
189252ea8506SJoe Perches}
189352ea8506SJoe Perches
18943705ce5bSJoe Perchessub tabify {
18953705ce5bSJoe Perches	my ($leading) = @_;
18963705ce5bSJoe Perches
18973705ce5bSJoe Perches	my $source_indent = 8;
18983705ce5bSJoe Perches	my $max_spaces_before_tab = $source_indent - 1;
18993705ce5bSJoe Perches	my $spaces_to_tab = " " x $source_indent;
19003705ce5bSJoe Perches
19013705ce5bSJoe Perches	#convert leading spaces to tabs
19023705ce5bSJoe Perches	1 while $leading =~ s@^([\t]*)$spaces_to_tab@$1\t@g;
19033705ce5bSJoe Perches	#Remove spaces before a tab
19043705ce5bSJoe Perches	1 while $leading =~ s@^([\t]*)( {1,$max_spaces_before_tab})\t@$1\t@g;
19053705ce5bSJoe Perches
19063705ce5bSJoe Perches	return "$leading";
19073705ce5bSJoe Perches}
19083705ce5bSJoe Perches
1909d1fe9c09SJoe Perchessub pos_last_openparen {
1910d1fe9c09SJoe Perches	my ($line) = @_;
1911d1fe9c09SJoe Perches
1912d1fe9c09SJoe Perches	my $pos = 0;
1913d1fe9c09SJoe Perches
1914d1fe9c09SJoe Perches	my $opens = $line =~ tr/\(/\(/;
1915d1fe9c09SJoe Perches	my $closes = $line =~ tr/\)/\)/;
1916d1fe9c09SJoe Perches
1917d1fe9c09SJoe Perches	my $last_openparen = 0;
1918d1fe9c09SJoe Perches
1919d1fe9c09SJoe Perches	if (($opens == 0) || ($closes >= $opens)) {
1920d1fe9c09SJoe Perches		return -1;
1921d1fe9c09SJoe Perches	}
1922d1fe9c09SJoe Perches
1923d1fe9c09SJoe Perches	my $len = length($line);
1924d1fe9c09SJoe Perches
1925d1fe9c09SJoe Perches	for ($pos = 0; $pos < $len; $pos++) {
1926d1fe9c09SJoe Perches		my $string = substr($line, $pos);
1927d1fe9c09SJoe Perches		if ($string =~ /^($FuncArg|$balanced_parens)/) {
1928d1fe9c09SJoe Perches			$pos += length($1) - 1;
1929d1fe9c09SJoe Perches		} elsif (substr($line, $pos, 1) eq '(') {
1930d1fe9c09SJoe Perches			$last_openparen = $pos;
1931d1fe9c09SJoe Perches		} elsif (index($string, '(') == -1) {
1932d1fe9c09SJoe Perches			last;
1933d1fe9c09SJoe Perches		}
1934d1fe9c09SJoe Perches	}
1935d1fe9c09SJoe Perches
193691cb5195SJoe Perches	return length(expand_tabs(substr($line, 0, $last_openparen))) + 1;
1937d1fe9c09SJoe Perches}
1938d1fe9c09SJoe Perches
19390a920b5bSAndy Whitcroftsub process {
19400a920b5bSAndy Whitcroft	my $filename = shift;
19410a920b5bSAndy Whitcroft
19420a920b5bSAndy Whitcroft	my $linenr=0;
19430a920b5bSAndy Whitcroft	my $prevline="";
1944c2fdda0dSAndy Whitcroft	my $prevrawline="";
19450a920b5bSAndy Whitcroft	my $stashline="";
1946c2fdda0dSAndy Whitcroft	my $stashrawline="";
19470a920b5bSAndy Whitcroft
19484a0df2efSAndy Whitcroft	my $length;
19490a920b5bSAndy Whitcroft	my $indent;
19500a920b5bSAndy Whitcroft	my $previndent=0;
19510a920b5bSAndy Whitcroft	my $stashindent=0;
19520a920b5bSAndy Whitcroft
1953de7d4f0eSAndy Whitcroft	our $clean = 1;
19540a920b5bSAndy Whitcroft	my $signoff = 0;
19550a920b5bSAndy Whitcroft	my $is_patch = 0;
19560a920b5bSAndy Whitcroft
195729ee1b0cSJoe Perches	my $in_header_lines = $file ? 0 : 1;
195815662b3eSJoe Perches	my $in_commit_log = 0;		#Scanning lines before patch
19592a076f40SJoe Perches	my $commit_log_long_line = 0;
1960e518e9a5SJoe Perches	my $commit_log_has_diff = 0;
196113f1937eSJoe Perches	my $reported_maintainer_file = 0;
1962fa64205dSPasi Savanainen	my $non_utf8_charset = 0;
1963fa64205dSPasi Savanainen
1964365dd4eaSJoe Perches	my $last_blank_line = 0;
19655e4f6ba5SJoe Perches	my $last_coalesced_string_linenr = -1;
1966365dd4eaSJoe Perches
196713214adfSAndy Whitcroft	our @report = ();
19686c72ffaaSAndy Whitcroft	our $cnt_lines = 0;
19696c72ffaaSAndy Whitcroft	our $cnt_error = 0;
19706c72ffaaSAndy Whitcroft	our $cnt_warn = 0;
19716c72ffaaSAndy Whitcroft	our $cnt_chk = 0;
19726c72ffaaSAndy Whitcroft
19730a920b5bSAndy Whitcroft	# Trace the real file/line as we go.
19740a920b5bSAndy Whitcroft	my $realfile = '';
19750a920b5bSAndy Whitcroft	my $realline = 0;
19760a920b5bSAndy Whitcroft	my $realcnt = 0;
19770a920b5bSAndy Whitcroft	my $here = '';
19780a920b5bSAndy Whitcroft	my $in_comment = 0;
1979c2fdda0dSAndy Whitcroft	my $comment_edge = 0;
19800a920b5bSAndy Whitcroft	my $first_line = 0;
19811e855726SWolfram Sang	my $p1_prefix = '';
19820a920b5bSAndy Whitcroft
198313214adfSAndy Whitcroft	my $prev_values = 'E';
198413214adfSAndy Whitcroft
198513214adfSAndy Whitcroft	# suppression flags
1986773647a0SAndy Whitcroft	my %suppress_ifbraces;
1987170d3a22SAndy Whitcroft	my %suppress_whiletrailers;
19882b474a1aSAndy Whitcroft	my %suppress_export;
19893e469cdcSAndy Whitcroft	my $suppress_statement = 0;
1990653d4876SAndy Whitcroft
19917e51f197SJoe Perches	my %signatures = ();
1992323c1260SJoe Perches
1993c2fdda0dSAndy Whitcroft	# Pre-scan the patch sanitizing the lines.
1994de7d4f0eSAndy Whitcroft	# Pre-scan the patch looking for any __setup documentation.
1995c2fdda0dSAndy Whitcroft	#
1996de7d4f0eSAndy Whitcroft	my @setup_docs = ();
1997de7d4f0eSAndy Whitcroft	my $setup_docs = 0;
1998773647a0SAndy Whitcroft
1999d8b07710SJoe Perches	my $camelcase_file_seeded = 0;
2000d8b07710SJoe Perches
2001773647a0SAndy Whitcroft	sanitise_line_reset();
2002c2fdda0dSAndy Whitcroft	my $line;
2003c2fdda0dSAndy Whitcroft	foreach my $rawline (@rawlines) {
2004773647a0SAndy Whitcroft		$linenr++;
2005773647a0SAndy Whitcroft		$line = $rawline;
2006c2fdda0dSAndy Whitcroft
20073705ce5bSJoe Perches		push(@fixed, $rawline) if ($fix);
20083705ce5bSJoe Perches
2009773647a0SAndy Whitcroft		if ($rawline=~/^\+\+\+\s+(\S+)/) {
2010de7d4f0eSAndy Whitcroft			$setup_docs = 0;
2011de7d4f0eSAndy Whitcroft			if ($1 =~ m@Documentation/kernel-parameters.txt$@) {
2012de7d4f0eSAndy Whitcroft				$setup_docs = 1;
2013de7d4f0eSAndy Whitcroft			}
2014773647a0SAndy Whitcroft			#next;
2015de7d4f0eSAndy Whitcroft		}
2016773647a0SAndy Whitcroft		if ($rawline=~/^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) {
2017773647a0SAndy Whitcroft			$realline=$1-1;
2018773647a0SAndy Whitcroft			if (defined $2) {
2019773647a0SAndy Whitcroft				$realcnt=$3+1;
2020773647a0SAndy Whitcroft			} else {
2021773647a0SAndy Whitcroft				$realcnt=1+1;
2022773647a0SAndy Whitcroft			}
2023c45dcabdSAndy Whitcroft			$in_comment = 0;
2024773647a0SAndy Whitcroft
2025773647a0SAndy Whitcroft			# Guestimate if this is a continuing comment.  Run
2026773647a0SAndy Whitcroft			# the context looking for a comment "edge".  If this
2027773647a0SAndy Whitcroft			# edge is a close comment then we must be in a comment
2028773647a0SAndy Whitcroft			# at context start.
2029773647a0SAndy Whitcroft			my $edge;
203001fa9147SAndy Whitcroft			my $cnt = $realcnt;
203101fa9147SAndy Whitcroft			for (my $ln = $linenr + 1; $cnt > 0; $ln++) {
203201fa9147SAndy Whitcroft				next if (defined $rawlines[$ln - 1] &&
203301fa9147SAndy Whitcroft					 $rawlines[$ln - 1] =~ /^-/);
203401fa9147SAndy Whitcroft				$cnt--;
203501fa9147SAndy Whitcroft				#print "RAW<$rawlines[$ln - 1]>\n";
2036721c1cb6SAndy Whitcroft				last if (!defined $rawlines[$ln - 1]);
2037fae17daeSAndy Whitcroft				if ($rawlines[$ln - 1] =~ m@(/\*|\*/)@ &&
2038fae17daeSAndy Whitcroft				    $rawlines[$ln - 1] !~ m@"[^"]*(?:/\*|\*/)[^"]*"@) {
2039fae17daeSAndy Whitcroft					($edge) = $1;
2040fae17daeSAndy Whitcroft					last;
2041fae17daeSAndy Whitcroft				}
2042773647a0SAndy Whitcroft			}
2043773647a0SAndy Whitcroft			if (defined $edge && $edge eq '*/') {
2044773647a0SAndy Whitcroft				$in_comment = 1;
2045773647a0SAndy Whitcroft			}
2046773647a0SAndy Whitcroft
2047773647a0SAndy Whitcroft			# Guestimate if this is a continuing comment.  If this
2048773647a0SAndy Whitcroft			# is the start of a diff block and this line starts
2049773647a0SAndy Whitcroft			# ' *' then it is very likely a comment.
2050773647a0SAndy Whitcroft			if (!defined $edge &&
205183242e0cSAndy Whitcroft			    $rawlines[$linenr] =~ m@^.\s*(?:\*\*+| \*)(?:\s|$)@)
2052773647a0SAndy Whitcroft			{
2053773647a0SAndy Whitcroft				$in_comment = 1;
2054773647a0SAndy Whitcroft			}
2055773647a0SAndy Whitcroft
2056773647a0SAndy Whitcroft			##print "COMMENT:$in_comment edge<$edge> $rawline\n";
2057773647a0SAndy Whitcroft			sanitise_line_reset($in_comment);
2058773647a0SAndy Whitcroft
2059171ae1a4SAndy Whitcroft		} elsif ($realcnt && $rawline =~ /^(?:\+| |$)/) {
2060773647a0SAndy Whitcroft			# Standardise the strings and chars within the input to
2061171ae1a4SAndy Whitcroft			# simplify matching -- only bother with positive lines.
2062773647a0SAndy Whitcroft			$line = sanitise_line($rawline);
2063773647a0SAndy Whitcroft		}
2064773647a0SAndy Whitcroft		push(@lines, $line);
2065773647a0SAndy Whitcroft
2066773647a0SAndy Whitcroft		if ($realcnt > 1) {
2067773647a0SAndy Whitcroft			$realcnt-- if ($line =~ /^(?:\+| |$)/);
2068773647a0SAndy Whitcroft		} else {
2069773647a0SAndy Whitcroft			$realcnt = 0;
2070773647a0SAndy Whitcroft		}
2071773647a0SAndy Whitcroft
2072773647a0SAndy Whitcroft		#print "==>$rawline\n";
2073773647a0SAndy Whitcroft		#print "-->$line\n";
2074de7d4f0eSAndy Whitcroft
2075de7d4f0eSAndy Whitcroft		if ($setup_docs && $line =~ /^\+/) {
2076de7d4f0eSAndy Whitcroft			push(@setup_docs, $line);
2077de7d4f0eSAndy Whitcroft		}
2078de7d4f0eSAndy Whitcroft	}
2079de7d4f0eSAndy Whitcroft
20806c72ffaaSAndy Whitcroft	$prefix = '';
20816c72ffaaSAndy Whitcroft
2082773647a0SAndy Whitcroft	$realcnt = 0;
2083773647a0SAndy Whitcroft	$linenr = 0;
2084194f66fcSJoe Perches	$fixlinenr = -1;
20850a920b5bSAndy Whitcroft	foreach my $line (@lines) {
20860a920b5bSAndy Whitcroft		$linenr++;
2087194f66fcSJoe Perches		$fixlinenr++;
20881b5539b1SJoe Perches		my $sline = $line;	#copy of $line
20891b5539b1SJoe Perches		$sline =~ s/$;/ /g;	#with comments as spaces
20900a920b5bSAndy Whitcroft
2091c2fdda0dSAndy Whitcroft		my $rawline = $rawlines[$linenr - 1];
20926c72ffaaSAndy Whitcroft
20930a920b5bSAndy Whitcroft#extract the line range in the file after the patch is applied
2094e518e9a5SJoe Perches		if (!$in_commit_log &&
2095e518e9a5SJoe Perches		    $line =~ /^\@\@ -\d+(?:,\d+)? \+(\d+)(,(\d+))? \@\@/) {
20960a920b5bSAndy Whitcroft			$is_patch = 1;
20974a0df2efSAndy Whitcroft			$first_line = $linenr + 1;
20980a920b5bSAndy Whitcroft			$realline=$1-1;
20990a920b5bSAndy Whitcroft			if (defined $2) {
21000a920b5bSAndy Whitcroft				$realcnt=$3+1;
21010a920b5bSAndy Whitcroft			} else {
21020a920b5bSAndy Whitcroft				$realcnt=1+1;
21030a920b5bSAndy Whitcroft			}
2104c2fdda0dSAndy Whitcroft			annotate_reset();
210513214adfSAndy Whitcroft			$prev_values = 'E';
210613214adfSAndy Whitcroft
2107773647a0SAndy Whitcroft			%suppress_ifbraces = ();
2108170d3a22SAndy Whitcroft			%suppress_whiletrailers = ();
21092b474a1aSAndy Whitcroft			%suppress_export = ();
21103e469cdcSAndy Whitcroft			$suppress_statement = 0;
21110a920b5bSAndy Whitcroft			next;
21120a920b5bSAndy Whitcroft
21134a0df2efSAndy Whitcroft# track the line number as we move through the hunk, note that
21144a0df2efSAndy Whitcroft# new versions of GNU diff omit the leading space on completely
21154a0df2efSAndy Whitcroft# blank context lines so we need to count that too.
2116773647a0SAndy Whitcroft		} elsif ($line =~ /^( |\+|$)/) {
21170a920b5bSAndy Whitcroft			$realline++;
2118d8aaf121SAndy Whitcroft			$realcnt-- if ($realcnt != 0);
21190a920b5bSAndy Whitcroft
21204a0df2efSAndy Whitcroft			# Measure the line length and indent.
2121c2fdda0dSAndy Whitcroft			($length, $indent) = line_stats($rawline);
21220a920b5bSAndy Whitcroft
21230a920b5bSAndy Whitcroft			# Track the previous line.
21240a920b5bSAndy Whitcroft			($prevline, $stashline) = ($stashline, $line);
21250a920b5bSAndy Whitcroft			($previndent, $stashindent) = ($stashindent, $indent);
2126c2fdda0dSAndy Whitcroft			($prevrawline, $stashrawline) = ($stashrawline, $rawline);
2127c2fdda0dSAndy Whitcroft
2128773647a0SAndy Whitcroft			#warn "line<$line>\n";
21296c72ffaaSAndy Whitcroft
2130d8aaf121SAndy Whitcroft		} elsif ($realcnt == 1) {
2131d8aaf121SAndy Whitcroft			$realcnt--;
21320a920b5bSAndy Whitcroft		}
21330a920b5bSAndy Whitcroft
2134cc77cdcaSAndy Whitcroft		my $hunk_line = ($realcnt != 0);
2135cc77cdcaSAndy Whitcroft
21366c72ffaaSAndy Whitcroft		$here = "#$linenr: " if (!$file);
21376c72ffaaSAndy Whitcroft		$here = "#$realline: " if ($file);
2138773647a0SAndy Whitcroft
21392ac73b4fSJoe Perches		my $found_file = 0;
2140773647a0SAndy Whitcroft		# extract the filename as it passes
21413bf9a009SRabin Vincent		if ($line =~ /^diff --git.*?(\S+)$/) {
21423bf9a009SRabin Vincent			$realfile = $1;
21432b7ab453SJoe Perches			$realfile =~ s@^([^/]*)/@@ if (!$file);
2144270c49a0SJoe Perches			$in_commit_log = 0;
21452ac73b4fSJoe Perches			$found_file = 1;
21463bf9a009SRabin Vincent		} elsif ($line =~ /^\+\+\+\s+(\S+)/) {
2147773647a0SAndy Whitcroft			$realfile = $1;
21482b7ab453SJoe Perches			$realfile =~ s@^([^/]*)/@@ if (!$file);
2149270c49a0SJoe Perches			$in_commit_log = 0;
21501e855726SWolfram Sang
21511e855726SWolfram Sang			$p1_prefix = $1;
2152e2f7aa4bSAndy Whitcroft			if (!$file && $tree && $p1_prefix ne '' &&
2153e2f7aa4bSAndy Whitcroft			    -e "$root/$p1_prefix") {
2154000d1cc1SJoe Perches				WARN("PATCH_PREFIX",
2155000d1cc1SJoe Perches				     "patch prefix '$p1_prefix' exists, appears to be a -p0 patch\n");
21561e855726SWolfram Sang			}
2157773647a0SAndy Whitcroft
2158c1ab3326SAndy Whitcroft			if ($realfile =~ m@^include/asm/@) {
2159000d1cc1SJoe Perches				ERROR("MODIFIED_INCLUDE_ASM",
2160000d1cc1SJoe Perches				      "do not modify files in include/asm, change architecture specific files in include/asm-<architecture>\n" . "$here$rawline\n");
2161773647a0SAndy Whitcroft			}
21622ac73b4fSJoe Perches			$found_file = 1;
21632ac73b4fSJoe Perches		}
21642ac73b4fSJoe Perches
216534d8815fSJoe Perches#make up the handle for any error we report on this line
216634d8815fSJoe Perches		if ($showfile) {
216734d8815fSJoe Perches			$prefix = "$realfile:$realline: "
216834d8815fSJoe Perches		} elsif ($emacs) {
216934d8815fSJoe Perches			$prefix = "$filename:$linenr: ";
217034d8815fSJoe Perches		}
217134d8815fSJoe Perches
21722ac73b4fSJoe Perches		if ($found_file) {
21732ac73b4fSJoe Perches			if ($realfile =~ m@^(drivers/net/|net/)@) {
21742ac73b4fSJoe Perches				$check = 1;
21752ac73b4fSJoe Perches			} else {
21762ac73b4fSJoe Perches				$check = $check_orig;
21772ac73b4fSJoe Perches			}
2178773647a0SAndy Whitcroft			next;
2179773647a0SAndy Whitcroft		}
2180773647a0SAndy Whitcroft
2181389834b6SRandy Dunlap		$here .= "FILE: $realfile:$realline:" if ($realcnt != 0);
21820a920b5bSAndy Whitcroft
2183c2fdda0dSAndy Whitcroft		my $hereline = "$here\n$rawline\n";
2184c2fdda0dSAndy Whitcroft		my $herecurr = "$here\n$rawline\n";
2185c2fdda0dSAndy Whitcroft		my $hereprev = "$here\n$prevrawline\n$rawline\n";
21860a920b5bSAndy Whitcroft
21876c72ffaaSAndy Whitcroft		$cnt_lines++ if ($realcnt != 0);
21886c72ffaaSAndy Whitcroft
2189e518e9a5SJoe Perches# Check if the commit log has what seems like a diff which can confuse patch
2190e518e9a5SJoe Perches		if ($in_commit_log && !$commit_log_has_diff &&
2191e518e9a5SJoe Perches		    (($line =~ m@^\s+diff\b.*a/[\w/]+@ &&
2192e518e9a5SJoe Perches		      $line =~ m@^\s+diff\b.*a/([\w/]+)\s+b/$1\b@) ||
2193e518e9a5SJoe Perches		     $line =~ m@^\s*(?:\-\-\-\s+a/|\+\+\+\s+b/)@ ||
2194e518e9a5SJoe Perches		     $line =~ m/^\s*\@\@ \-\d+,\d+ \+\d+,\d+ \@\@/)) {
2195e518e9a5SJoe Perches			ERROR("DIFF_IN_COMMIT_MSG",
2196e518e9a5SJoe Perches			      "Avoid using diff content in the commit message - patch(1) might not work\n" . $herecurr);
2197e518e9a5SJoe Perches			$commit_log_has_diff = 1;
2198e518e9a5SJoe Perches		}
2199e518e9a5SJoe Perches
22003bf9a009SRabin Vincent# Check for incorrect file permissions
22013bf9a009SRabin Vincent		if ($line =~ /^new (file )?mode.*[7531]\d{0,2}$/) {
22023bf9a009SRabin Vincent			my $permhere = $here . "FILE: $realfile\n";
220304db4d25SJoe Perches			if ($realfile !~ m@scripts/@ &&
220404db4d25SJoe Perches			    $realfile !~ /\.(py|pl|awk|sh)$/) {
2205000d1cc1SJoe Perches				ERROR("EXECUTE_PERMISSIONS",
2206000d1cc1SJoe Perches				      "do not set execute permissions for source files\n" . $permhere);
22073bf9a009SRabin Vincent			}
22083bf9a009SRabin Vincent		}
22093bf9a009SRabin Vincent
221020112475SJoe Perches# Check the patch for a signoff:
2211d8aaf121SAndy Whitcroft		if ($line =~ /^\s*signed-off-by:/i) {
22124a0df2efSAndy Whitcroft			$signoff++;
221315662b3eSJoe Perches			$in_commit_log = 0;
22140a920b5bSAndy Whitcroft		}
221520112475SJoe Perches
2216e0d975b1SJoe Perches# Check if MAINTAINERS is being updated.  If so, there's probably no need to
2217e0d975b1SJoe Perches# emit the "does MAINTAINERS need updating?" message on file add/move/delete
2218e0d975b1SJoe Perches		if ($line =~ /^\s*MAINTAINERS\s*\|/) {
2219e0d975b1SJoe Perches			$reported_maintainer_file = 1;
2220e0d975b1SJoe Perches		}
2221e0d975b1SJoe Perches
222220112475SJoe Perches# Check signature styles
2223270c49a0SJoe Perches		if (!$in_header_lines &&
2224ce0338dfSJoe Perches		    $line =~ /^(\s*)([a-z0-9_-]+by:|$signature_tags)(\s*)(.*)/i) {
222520112475SJoe Perches			my $space_before = $1;
222620112475SJoe Perches			my $sign_off = $2;
222720112475SJoe Perches			my $space_after = $3;
222820112475SJoe Perches			my $email = $4;
222920112475SJoe Perches			my $ucfirst_sign_off = ucfirst(lc($sign_off));
223020112475SJoe Perches
2231ce0338dfSJoe Perches			if ($sign_off !~ /$signature_tags/) {
2232ce0338dfSJoe Perches				WARN("BAD_SIGN_OFF",
2233ce0338dfSJoe Perches				     "Non-standard signature: $sign_off\n" . $herecurr);
2234ce0338dfSJoe Perches			}
223520112475SJoe Perches			if (defined $space_before && $space_before ne "") {
22363705ce5bSJoe Perches				if (WARN("BAD_SIGN_OFF",
22373705ce5bSJoe Perches					 "Do not use whitespace before $ucfirst_sign_off\n" . $herecurr) &&
22383705ce5bSJoe Perches				    $fix) {
2239194f66fcSJoe Perches					$fixed[$fixlinenr] =
22403705ce5bSJoe Perches					    "$ucfirst_sign_off $email";
22413705ce5bSJoe Perches				}
224220112475SJoe Perches			}
224320112475SJoe Perches			if ($sign_off =~ /-by:$/i && $sign_off ne $ucfirst_sign_off) {
22443705ce5bSJoe Perches				if (WARN("BAD_SIGN_OFF",
22453705ce5bSJoe Perches					 "'$ucfirst_sign_off' is the preferred signature form\n" . $herecurr) &&
22463705ce5bSJoe Perches				    $fix) {
2247194f66fcSJoe Perches					$fixed[$fixlinenr] =
22483705ce5bSJoe Perches					    "$ucfirst_sign_off $email";
22493705ce5bSJoe Perches				}
22503705ce5bSJoe Perches
225120112475SJoe Perches			}
225220112475SJoe Perches			if (!defined $space_after || $space_after ne " ") {
22533705ce5bSJoe Perches				if (WARN("BAD_SIGN_OFF",
22543705ce5bSJoe Perches					 "Use a single space after $ucfirst_sign_off\n" . $herecurr) &&
22553705ce5bSJoe Perches				    $fix) {
2256194f66fcSJoe Perches					$fixed[$fixlinenr] =
22573705ce5bSJoe Perches					    "$ucfirst_sign_off $email";
22583705ce5bSJoe Perches				}
225920112475SJoe Perches			}
226020112475SJoe Perches
226120112475SJoe Perches			my ($email_name, $email_address, $comment) = parse_email($email);
226220112475SJoe Perches			my $suggested_email = format_email(($email_name, $email_address));
226320112475SJoe Perches			if ($suggested_email eq "") {
2264000d1cc1SJoe Perches				ERROR("BAD_SIGN_OFF",
2265000d1cc1SJoe Perches				      "Unrecognized email address: '$email'\n" . $herecurr);
226620112475SJoe Perches			} else {
226720112475SJoe Perches				my $dequoted = $suggested_email;
226820112475SJoe Perches				$dequoted =~ s/^"//;
226920112475SJoe Perches				$dequoted =~ s/" </ </;
227020112475SJoe Perches				# Don't force email to have quotes
227120112475SJoe Perches				# Allow just an angle bracketed address
227220112475SJoe Perches				if ("$dequoted$comment" ne $email &&
227320112475SJoe Perches				    "<$email_address>$comment" ne $email &&
227420112475SJoe Perches				    "$suggested_email$comment" ne $email) {
2275000d1cc1SJoe Perches					WARN("BAD_SIGN_OFF",
2276000d1cc1SJoe Perches					     "email address '$email' might be better as '$suggested_email$comment'\n" . $herecurr);
227720112475SJoe Perches				}
22780a920b5bSAndy Whitcroft			}
22797e51f197SJoe Perches
22807e51f197SJoe Perches# Check for duplicate signatures
22817e51f197SJoe Perches			my $sig_nospace = $line;
22827e51f197SJoe Perches			$sig_nospace =~ s/\s//g;
22837e51f197SJoe Perches			$sig_nospace = lc($sig_nospace);
22847e51f197SJoe Perches			if (defined $signatures{$sig_nospace}) {
22857e51f197SJoe Perches				WARN("BAD_SIGN_OFF",
22867e51f197SJoe Perches				     "Duplicate signature\n" . $herecurr);
22877e51f197SJoe Perches			} else {
22887e51f197SJoe Perches				$signatures{$sig_nospace} = 1;
22897e51f197SJoe Perches			}
22900a920b5bSAndy Whitcroft		}
22910a920b5bSAndy Whitcroft
2292a2fe16b9SJoe Perches# Check email subject for common tools that don't need to be mentioned
2293a2fe16b9SJoe Perches		if ($in_header_lines &&
2294a2fe16b9SJoe Perches		    $line =~ /^Subject:.*\b(?:checkpatch|sparse|smatch)\b[^:]/i) {
2295a2fe16b9SJoe Perches			WARN("EMAIL_SUBJECT",
2296a2fe16b9SJoe Perches			     "A patch subject line should describe the change not the tool that found it\n" . $herecurr);
2297a2fe16b9SJoe Perches		}
2298a2fe16b9SJoe Perches
22999b3189ebSJoe Perches# Check for old stable address
23009b3189ebSJoe Perches		if ($line =~ /^\s*cc:\s*.*<?\bstable\@kernel\.org\b>?.*$/i) {
23019b3189ebSJoe Perches			ERROR("STABLE_ADDRESS",
23029b3189ebSJoe Perches			      "The 'stable' address should be 'stable\@vger.kernel.org'\n" . $herecurr);
23039b3189ebSJoe Perches		}
23049b3189ebSJoe Perches
23057ebd05efSChristopher Covington# Check for unwanted Gerrit info
23067ebd05efSChristopher Covington		if ($in_commit_log && $line =~ /^\s*change-id:/i) {
23077ebd05efSChristopher Covington			ERROR("GERRIT_CHANGE_ID",
23087ebd05efSChristopher Covington			      "Remove Gerrit Change-Id's before submitting upstream.\n" . $herecurr);
23097ebd05efSChristopher Covington		}
23107ebd05efSChristopher Covington
23112a076f40SJoe Perches# Check for line lengths > 75 in commit log, warn once
23122a076f40SJoe Perches		if ($in_commit_log && !$commit_log_long_line &&
23132a076f40SJoe Perches		    length($line) > 75) {
23142a076f40SJoe Perches			WARN("COMMIT_LOG_LONG_LINE",
23152a076f40SJoe Perches			     "Possible unwrapped commit description (prefer a maximum 75 chars per line)\n" . $herecurr);
23162a076f40SJoe Perches			$commit_log_long_line = 1;
23172a076f40SJoe Perches		}
23182a076f40SJoe Perches
23190d7835fcSJoe Perches# Check for git id commit length and improperly formed commit descriptions
2320*fe043ea1SJoe Perches		if ($in_commit_log &&
2321*fe043ea1SJoe Perches		    ($line =~ /\bcommit\s+[0-9a-f]{5,}\b/i ||
2322*fe043ea1SJoe Perches		     $line =~ /\b[0-9a-f]{12,40}\b/i)) {
2323*fe043ea1SJoe Perches			my $init_char = "c";
2324*fe043ea1SJoe Perches			my $orig_commit = "";
23250d7835fcSJoe Perches			my $short = 1;
23260d7835fcSJoe Perches			my $long = 0;
23270d7835fcSJoe Perches			my $case = 1;
23280d7835fcSJoe Perches			my $space = 1;
23290d7835fcSJoe Perches			my $hasdesc = 0;
233019c146a6SJoe Perches			my $hasparens = 0;
23310d7835fcSJoe Perches			my $id = '0123456789ab';
23320d7835fcSJoe Perches			my $orig_desc = "commit description";
23330d7835fcSJoe Perches			my $description = "";
23340d7835fcSJoe Perches
2335*fe043ea1SJoe Perches			if ($line =~ /\b(c)ommit\s+([0-9a-f]{5,})\b/i) {
2336*fe043ea1SJoe Perches				$init_char = $1;
2337*fe043ea1SJoe Perches				$orig_commit = lc($2);
2338*fe043ea1SJoe Perches			} elsif ($line =~ /\b([0-9a-f]{12,40})\b/i) {
2339*fe043ea1SJoe Perches				$orig_commit = lc($1);
2340*fe043ea1SJoe Perches			}
2341*fe043ea1SJoe Perches
23420d7835fcSJoe Perches			$short = 0 if ($line =~ /\bcommit\s+[0-9a-f]{12,40}/i);
23430d7835fcSJoe Perches			$long = 1 if ($line =~ /\bcommit\s+[0-9a-f]{41,}/i);
23440d7835fcSJoe Perches			$space = 0 if ($line =~ /\bcommit [0-9a-f]/i);
23450d7835fcSJoe Perches			$case = 0 if ($line =~ /\b[Cc]ommit\s+[0-9a-f]{5,40}[^A-F]/);
23460d7835fcSJoe Perches			if ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)"\)/i) {
23470d7835fcSJoe Perches				$orig_desc = $1;
234819c146a6SJoe Perches				$hasparens = 1;
23490d7835fcSJoe Perches			} elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s*$/i &&
23500d7835fcSJoe Perches				 defined $rawlines[$linenr] &&
23510d7835fcSJoe Perches				 $rawlines[$linenr] =~ /^\s*\("([^"]+)"\)/) {
23520d7835fcSJoe Perches				$orig_desc = $1;
235319c146a6SJoe Perches				$hasparens = 1;
2354b671fde0SJoe Perches			} elsif ($line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("[^"]+$/i &&
2355b671fde0SJoe Perches				 defined $rawlines[$linenr] &&
2356b671fde0SJoe Perches				 $rawlines[$linenr] =~ /^\s*[^"]+"\)/) {
2357b671fde0SJoe Perches				$line =~ /\bcommit\s+[0-9a-f]{5,}\s+\("([^"]+)$/i;
2358b671fde0SJoe Perches				$orig_desc = $1;
2359b671fde0SJoe Perches				$rawlines[$linenr] =~ /^\s*([^"]+)"\)/;
2360b671fde0SJoe Perches				$orig_desc .= " " . $1;
236119c146a6SJoe Perches				$hasparens = 1;
23620d7835fcSJoe Perches			}
23630d7835fcSJoe Perches
23640d7835fcSJoe Perches			($id, $description) = git_commit_info($orig_commit,
23650d7835fcSJoe Perches							      $id, $orig_desc);
23660d7835fcSJoe Perches
236719c146a6SJoe Perches			if ($short || $long || $space || $case || ($orig_desc ne $description) || !$hasparens) {
2368d311cd44SJoe Perches				ERROR("GIT_COMMIT_ID",
23690d7835fcSJoe Perches				      "Please use git commit description style 'commit <12+ chars of sha1> (\"<title line>\")' - ie: '${init_char}ommit $id (\"$description\")'\n" . $herecurr);
23700d7835fcSJoe Perches			}
2371d311cd44SJoe Perches		}
2372d311cd44SJoe Perches
237313f1937eSJoe Perches# Check for added, moved or deleted files
237413f1937eSJoe Perches		if (!$reported_maintainer_file && !$in_commit_log &&
237513f1937eSJoe Perches		    ($line =~ /^(?:new|deleted) file mode\s*\d+\s*$/ ||
237613f1937eSJoe Perches		     $line =~ /^rename (?:from|to) [\w\/\.\-]+\s*$/ ||
237713f1937eSJoe Perches		     ($line =~ /\{\s*([\w\/\.\-]*)\s*\=\>\s*([\w\/\.\-]*)\s*\}/ &&
237813f1937eSJoe Perches		      (defined($1) || defined($2))))) {
237913f1937eSJoe Perches			$reported_maintainer_file = 1;
238013f1937eSJoe Perches			WARN("FILE_PATH_CHANGES",
238113f1937eSJoe Perches			     "added, moved or deleted file(s), does MAINTAINERS need updating?\n" . $herecurr);
238213f1937eSJoe Perches		}
238313f1937eSJoe Perches
238400df344fSAndy Whitcroft# Check for wrappage within a valid hunk of the file
23858905a67cSAndy Whitcroft		if ($realcnt != 0 && $line !~ m{^(?:\+|-| |\\ No newline|$)}) {
2386000d1cc1SJoe Perches			ERROR("CORRUPTED_PATCH",
2387000d1cc1SJoe Perches			      "patch seems to be corrupt (line wrapped?)\n" .
23886c72ffaaSAndy Whitcroft				$herecurr) if (!$emitted_corrupt++);
2389de7d4f0eSAndy Whitcroft		}
2390de7d4f0eSAndy Whitcroft
23916ecd9674SAndy Whitcroft# Check for absolute kernel paths.
23926ecd9674SAndy Whitcroft		if ($tree) {
23936ecd9674SAndy Whitcroft			while ($line =~ m{(?:^|\s)(/\S*)}g) {
23946ecd9674SAndy Whitcroft				my $file = $1;
23956ecd9674SAndy Whitcroft
23966ecd9674SAndy Whitcroft				if ($file =~ m{^(.*?)(?::\d+)+:?$} &&
23976ecd9674SAndy Whitcroft				    check_absolute_file($1, $herecurr)) {
23986ecd9674SAndy Whitcroft					#
23996ecd9674SAndy Whitcroft				} else {
24006ecd9674SAndy Whitcroft					check_absolute_file($file, $herecurr);
24016ecd9674SAndy Whitcroft				}
24026ecd9674SAndy Whitcroft			}
24036ecd9674SAndy Whitcroft		}
24046ecd9674SAndy Whitcroft
2405de7d4f0eSAndy Whitcroft# UTF-8 regex found at http://www.w3.org/International/questions/qa-forms-utf-8.en.php
2406de7d4f0eSAndy Whitcroft		if (($realfile =~ /^$/ || $line =~ /^\+/) &&
2407171ae1a4SAndy Whitcroft		    $rawline !~ m/^$UTF8*$/) {
2408171ae1a4SAndy Whitcroft			my ($utf8_prefix) = ($rawline =~ /^($UTF8*)/);
2409171ae1a4SAndy Whitcroft
2410171ae1a4SAndy Whitcroft			my $blank = copy_spacing($rawline);
2411171ae1a4SAndy Whitcroft			my $ptr = substr($blank, 0, length($utf8_prefix)) . "^";
2412171ae1a4SAndy Whitcroft			my $hereptr = "$hereline$ptr\n";
2413171ae1a4SAndy Whitcroft
241434d99219SJoe Perches			CHK("INVALID_UTF8",
2415000d1cc1SJoe Perches			    "Invalid UTF-8, patch and commit message should be encoded in UTF-8\n" . $hereptr);
241600df344fSAndy Whitcroft		}
24170a920b5bSAndy Whitcroft
241815662b3eSJoe Perches# Check if it's the start of a commit log
241915662b3eSJoe Perches# (not a header line and we haven't seen the patch filename)
242015662b3eSJoe Perches		if ($in_header_lines && $realfile =~ /^$/ &&
242129ee1b0cSJoe Perches		    !($rawline =~ /^\s+\S/ ||
242229ee1b0cSJoe Perches		      $rawline =~ /^(commit\b|from\b|[\w-]+:).*$/i)) {
242315662b3eSJoe Perches			$in_header_lines = 0;
242415662b3eSJoe Perches			$in_commit_log = 1;
242515662b3eSJoe Perches		}
242615662b3eSJoe Perches
2427fa64205dSPasi Savanainen# Check if there is UTF-8 in a commit log when a mail header has explicitly
2428fa64205dSPasi Savanainen# declined it, i.e defined some charset where it is missing.
2429fa64205dSPasi Savanainen		if ($in_header_lines &&
2430fa64205dSPasi Savanainen		    $rawline =~ /^Content-Type:.+charset="(.+)".*$/ &&
2431fa64205dSPasi Savanainen		    $1 !~ /utf-8/i) {
2432fa64205dSPasi Savanainen			$non_utf8_charset = 1;
2433fa64205dSPasi Savanainen		}
2434fa64205dSPasi Savanainen
2435fa64205dSPasi Savanainen		if ($in_commit_log && $non_utf8_charset && $realfile =~ /^$/ &&
243615662b3eSJoe Perches		    $rawline =~ /$NON_ASCII_UTF8/) {
2437fa64205dSPasi Savanainen			WARN("UTF8_BEFORE_PATCH",
243815662b3eSJoe Perches			    "8-bit UTF-8 used in possible commit log\n" . $herecurr);
243915662b3eSJoe Perches		}
244015662b3eSJoe Perches
244166b47b4aSKees Cook# Check for various typo / spelling mistakes
244266d7a382SJoe Perches		if (defined($misspellings) &&
244366d7a382SJoe Perches		    ($in_commit_log || $line =~ /^(?:\+|Subject:)/i)) {
2444ebfd7d62SJoe Perches			while ($rawline =~ /(?:^|[^a-z@])($misspellings)(?:\b|$|[^a-z@])/gi) {
244566b47b4aSKees Cook				my $typo = $1;
244666b47b4aSKees Cook				my $typo_fix = $spelling_fix{lc($typo)};
244766b47b4aSKees Cook				$typo_fix = ucfirst($typo_fix) if ($typo =~ /^[A-Z]/);
244866b47b4aSKees Cook				$typo_fix = uc($typo_fix) if ($typo =~ /^[A-Z]+$/);
244966b47b4aSKees Cook				my $msg_type = \&WARN;
245066b47b4aSKees Cook				$msg_type = \&CHK if ($file);
245166b47b4aSKees Cook				if (&{$msg_type}("TYPO_SPELLING",
245266b47b4aSKees Cook						 "'$typo' may be misspelled - perhaps '$typo_fix'?\n" . $herecurr) &&
245366b47b4aSKees Cook				    $fix) {
245466b47b4aSKees Cook					$fixed[$fixlinenr] =~ s/(^|[^A-Za-z@])($typo)($|[^A-Za-z@])/$1$typo_fix$3/;
245566b47b4aSKees Cook				}
245666b47b4aSKees Cook			}
245766b47b4aSKees Cook		}
245866b47b4aSKees Cook
245930670854SAndy Whitcroft# ignore non-hunk lines and lines being removed
246030670854SAndy Whitcroft		next if (!$hunk_line || $line =~ /^-/);
246100df344fSAndy Whitcroft
24620a920b5bSAndy Whitcroft#trailing whitespace
24639c0ca6f9SAndy Whitcroft		if ($line =~ /^\+.*\015/) {
2464c2fdda0dSAndy Whitcroft			my $herevet = "$here\n" . cat_vet($rawline) . "\n";
2465d5e616fcSJoe Perches			if (ERROR("DOS_LINE_ENDINGS",
2466d5e616fcSJoe Perches				  "DOS line endings\n" . $herevet) &&
2467d5e616fcSJoe Perches			    $fix) {
2468194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/[\s\015]+$//;
2469d5e616fcSJoe Perches			}
2470c2fdda0dSAndy Whitcroft		} elsif ($rawline =~ /^\+.*\S\s+$/ || $rawline =~ /^\+\s+$/) {
2471c2fdda0dSAndy Whitcroft			my $herevet = "$here\n" . cat_vet($rawline) . "\n";
24723705ce5bSJoe Perches			if (ERROR("TRAILING_WHITESPACE",
24733705ce5bSJoe Perches				  "trailing whitespace\n" . $herevet) &&
24743705ce5bSJoe Perches			    $fix) {
2475194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\s+$//;
24763705ce5bSJoe Perches			}
24773705ce5bSJoe Perches
2478d2c0a235SAndy Whitcroft			$rpt_cleaners = 1;
24790a920b5bSAndy Whitcroft		}
24805368df20SAndy Whitcroft
24814783f894SJosh Triplett# Check for FSF mailing addresses.
2482109d8cb2SAlexander Duyck		if ($rawline =~ /\bwrite to the Free/i ||
24833e2232f2SJoe Perches		    $rawline =~ /\b59\s+Temple\s+Pl/i ||
24843e2232f2SJoe Perches		    $rawline =~ /\b51\s+Franklin\s+St/i) {
24854783f894SJosh Triplett			my $herevet = "$here\n" . cat_vet($rawline) . "\n";
24864783f894SJosh Triplett			my $msg_type = \&ERROR;
24874783f894SJosh Triplett			$msg_type = \&CHK if ($file);
24884783f894SJosh Triplett			&{$msg_type}("FSF_MAILING_ADDRESS",
24894783f894SJosh Triplett				     "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)
24904783f894SJosh Triplett		}
24914783f894SJosh Triplett
24923354957aSAndi Kleen# check for Kconfig help text having a real description
24939fe287d7SAndy Whitcroft# Only applies when adding the entry originally, after that we do not have
24949fe287d7SAndy Whitcroft# sufficient context to determine whether it is indeed long enough.
24953354957aSAndi Kleen		if ($realfile =~ /Kconfig/ &&
24968d73e0e7SJoe Perches		    $line =~ /^\+\s*config\s+/) {
24973354957aSAndi Kleen			my $length = 0;
24989fe287d7SAndy Whitcroft			my $cnt = $realcnt;
24999fe287d7SAndy Whitcroft			my $ln = $linenr + 1;
25009fe287d7SAndy Whitcroft			my $f;
2501a1385803SAndy Whitcroft			my $is_start = 0;
25029fe287d7SAndy Whitcroft			my $is_end = 0;
2503a1385803SAndy Whitcroft			for (; $cnt > 0 && defined $lines[$ln - 1]; $ln++) {
25049fe287d7SAndy Whitcroft				$f = $lines[$ln - 1];
25059fe287d7SAndy Whitcroft				$cnt-- if ($lines[$ln - 1] !~ /^-/);
25069fe287d7SAndy Whitcroft				$is_end = $lines[$ln - 1] =~ /^\+/;
25079fe287d7SAndy Whitcroft
25089fe287d7SAndy Whitcroft				next if ($f =~ /^-/);
25098d73e0e7SJoe Perches				last if (!$file && $f =~ /^\@\@/);
2510a1385803SAndy Whitcroft
25118d73e0e7SJoe Perches				if ($lines[$ln - 1] =~ /^\+\s*(?:bool|tristate)\s*\"/) {
2512a1385803SAndy Whitcroft					$is_start = 1;
25138d73e0e7SJoe Perches				} elsif ($lines[$ln - 1] =~ /^\+\s*(?:---)?help(?:---)?$/) {
2514a1385803SAndy Whitcroft					$length = -1;
2515a1385803SAndy Whitcroft				}
2516a1385803SAndy Whitcroft
25179fe287d7SAndy Whitcroft				$f =~ s/^.//;
25183354957aSAndi Kleen				$f =~ s/#.*//;
25193354957aSAndi Kleen				$f =~ s/^\s+//;
25203354957aSAndi Kleen				next if ($f =~ /^$/);
25219fe287d7SAndy Whitcroft				if ($f =~ /^\s*config\s/) {
25229fe287d7SAndy Whitcroft					$is_end = 1;
25239fe287d7SAndy Whitcroft					last;
25249fe287d7SAndy Whitcroft				}
25253354957aSAndi Kleen				$length++;
25263354957aSAndi Kleen			}
252756193274SVadim Bendebury			if ($is_start && $is_end && $length < $min_conf_desc_length) {
2528000d1cc1SJoe Perches				WARN("CONFIG_DESCRIPTION",
252956193274SVadim Bendebury				     "please write a paragraph that describes the config symbol fully\n" . $herecurr);
253056193274SVadim Bendebury			}
2531a1385803SAndy Whitcroft			#print "is_start<$is_start> is_end<$is_end> length<$length>\n";
25323354957aSAndi Kleen		}
25333354957aSAndi Kleen
25341ba8dfd1SKees Cook# discourage the addition of CONFIG_EXPERIMENTAL in Kconfig.
25351ba8dfd1SKees Cook		if ($realfile =~ /Kconfig/ &&
25361ba8dfd1SKees Cook		    $line =~ /.\s*depends on\s+.*\bEXPERIMENTAL\b/) {
25371ba8dfd1SKees Cook			WARN("CONFIG_EXPERIMENTAL",
25381ba8dfd1SKees Cook			     "Use of CONFIG_EXPERIMENTAL is deprecated. For alternatives, see https://lkml.org/lkml/2012/10/23/580\n");
25391ba8dfd1SKees Cook		}
25401ba8dfd1SKees Cook
2541327953e9SChristoph Jaeger# discourage the use of boolean for type definition attributes of Kconfig options
2542327953e9SChristoph Jaeger		if ($realfile =~ /Kconfig/ &&
2543327953e9SChristoph Jaeger		    $line =~ /^\+\s*\bboolean\b/) {
2544327953e9SChristoph Jaeger			WARN("CONFIG_TYPE_BOOLEAN",
2545327953e9SChristoph Jaeger			     "Use of boolean is deprecated, please use bool instead.\n" . $herecurr);
2546327953e9SChristoph Jaeger		}
2547327953e9SChristoph Jaeger
2548c68e5878SArnaud Lacombe		if (($realfile =~ /Makefile.*/ || $realfile =~ /Kbuild.*/) &&
2549c68e5878SArnaud Lacombe		    ($line =~ /\+(EXTRA_[A-Z]+FLAGS).*/)) {
2550c68e5878SArnaud Lacombe			my $flag = $1;
2551c68e5878SArnaud Lacombe			my $replacement = {
2552c68e5878SArnaud Lacombe				'EXTRA_AFLAGS' =>   'asflags-y',
2553c68e5878SArnaud Lacombe				'EXTRA_CFLAGS' =>   'ccflags-y',
2554c68e5878SArnaud Lacombe				'EXTRA_CPPFLAGS' => 'cppflags-y',
2555c68e5878SArnaud Lacombe				'EXTRA_LDFLAGS' =>  'ldflags-y',
2556c68e5878SArnaud Lacombe			};
2557c68e5878SArnaud Lacombe
2558c68e5878SArnaud Lacombe			WARN("DEPRECATED_VARIABLE",
2559c68e5878SArnaud Lacombe			     "Use of $flag is deprecated, please use \`$replacement->{$flag} instead.\n" . $herecurr) if ($replacement->{$flag});
2560c68e5878SArnaud Lacombe		}
2561c68e5878SArnaud Lacombe
2562bff5da43SRob Herring# check for DT compatible documentation
25637dd05b38SFlorian Vaussard		if (defined $root &&
25647dd05b38SFlorian Vaussard			(($realfile =~ /\.dtsi?$/ && $line =~ /^\+\s*compatible\s*=\s*\"/) ||
25657dd05b38SFlorian Vaussard			 ($realfile =~ /\.[ch]$/ && $line =~ /^\+.*\.compatible\s*=\s*\"/))) {
25667dd05b38SFlorian Vaussard
2567bff5da43SRob Herring			my @compats = $rawline =~ /\"([a-zA-Z0-9\-\,\.\+_]+)\"/g;
2568bff5da43SRob Herring
2569cc93319bSFlorian Vaussard			my $dt_path = $root . "/Documentation/devicetree/bindings/";
2570cc93319bSFlorian Vaussard			my $vp_file = $dt_path . "vendor-prefixes.txt";
2571cc93319bSFlorian Vaussard
2572bff5da43SRob Herring			foreach my $compat (@compats) {
2573bff5da43SRob Herring				my $compat2 = $compat;
2574185d566bSRob Herring				$compat2 =~ s/\,[a-zA-Z0-9]*\-/\,<\.\*>\-/;
2575185d566bSRob Herring				my $compat3 = $compat;
2576185d566bSRob Herring				$compat3 =~ s/\,([a-z]*)[0-9]*\-/\,$1<\.\*>\-/;
2577185d566bSRob Herring				`grep -Erq "$compat|$compat2|$compat3" $dt_path`;
2578bff5da43SRob Herring				if ( $? >> 8 ) {
2579bff5da43SRob Herring					WARN("UNDOCUMENTED_DT_STRING",
2580bff5da43SRob Herring					     "DT compatible string \"$compat\" appears un-documented -- check $dt_path\n" . $herecurr);
2581bff5da43SRob Herring				}
2582bff5da43SRob Herring
25834fbf32a6SFlorian Vaussard				next if $compat !~ /^([a-zA-Z0-9\-]+)\,/;
25844fbf32a6SFlorian Vaussard				my $vendor = $1;
2585cc93319bSFlorian Vaussard				`grep -Eq "^$vendor\\b" $vp_file`;
2586bff5da43SRob Herring				if ( $? >> 8 ) {
2587bff5da43SRob Herring					WARN("UNDOCUMENTED_DT_STRING",
2588cc93319bSFlorian Vaussard					     "DT compatible string vendor \"$vendor\" appears un-documented -- check $vp_file\n" . $herecurr);
2589bff5da43SRob Herring				}
2590bff5da43SRob Herring			}
2591bff5da43SRob Herring		}
2592bff5da43SRob Herring
25935368df20SAndy Whitcroft# check we are in a valid source file if not then ignore this hunk
2594de4c924cSGeert Uytterhoeven		next if ($realfile !~ /\.(h|c|s|S|pl|sh|dtsi|dts)$/);
25955368df20SAndy Whitcroft
259647e0c88bSJoe Perches# line length limit (with some exclusions)
259747e0c88bSJoe Perches#
259847e0c88bSJoe Perches# There are a few types of lines that may extend beyond $max_line_length:
259947e0c88bSJoe Perches#	logging functions like pr_info that end in a string
260047e0c88bSJoe Perches#	lines with a single string
260147e0c88bSJoe Perches#	#defines that are a single string
260247e0c88bSJoe Perches#
260347e0c88bSJoe Perches# There are 3 different line length message types:
260447e0c88bSJoe Perches# LONG_LINE_COMMENT	a comment starts before but extends beyond $max_linelength
260547e0c88bSJoe Perches# LONG_LINE_STRING	a string starts before but extends beyond $max_line_length
260647e0c88bSJoe Perches# LONG_LINE		all other lines longer than $max_line_length
260747e0c88bSJoe Perches#
260847e0c88bSJoe Perches# if LONG_LINE is ignored, the other 2 types are also ignored
260947e0c88bSJoe Perches#
261047e0c88bSJoe Perches
2611b4749e96SJoe Perches		if ($line =~ /^\+/ && $length > $max_line_length) {
261247e0c88bSJoe Perches			my $msg_type = "LONG_LINE";
261347e0c88bSJoe Perches
261447e0c88bSJoe Perches			# Check the allowed long line types first
261547e0c88bSJoe Perches
261647e0c88bSJoe Perches			# logging functions that end in a string that starts
261747e0c88bSJoe Perches			# before $max_line_length
261847e0c88bSJoe Perches			if ($line =~ /^\+\s*$logFunctions\s*\(\s*(?:(?:KERN_\S+\s*|[^"]*))?($String\s*(?:|,|\)\s*;)\s*)$/ &&
261947e0c88bSJoe Perches			    length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) {
262047e0c88bSJoe Perches				$msg_type = "";
262147e0c88bSJoe Perches
262247e0c88bSJoe Perches			# lines with only strings (w/ possible termination)
262347e0c88bSJoe Perches			# #defines with only strings
262447e0c88bSJoe Perches			} elsif ($line =~ /^\+\s*$String\s*(?:\s*|,|\)\s*;)\s*$/ ||
262547e0c88bSJoe Perches				 $line =~ /^\+\s*#\s*define\s+\w+\s+$String$/) {
262647e0c88bSJoe Perches				$msg_type = "";
262747e0c88bSJoe Perches
262847e0c88bSJoe Perches			# Otherwise set the alternate message types
262947e0c88bSJoe Perches
263047e0c88bSJoe Perches			# a comment starts before $max_line_length
263147e0c88bSJoe Perches			} elsif ($line =~ /($;[\s$;]*)$/ &&
263247e0c88bSJoe Perches				 length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) {
263347e0c88bSJoe Perches				$msg_type = "LONG_LINE_COMMENT"
263447e0c88bSJoe Perches
263547e0c88bSJoe Perches			# a quoted string starts before $max_line_length
263647e0c88bSJoe Perches			} elsif ($sline =~ /\s*($String(?:\s*(?:\\|,\s*|\)\s*;\s*))?)$/ &&
263747e0c88bSJoe Perches				 length(expand_tabs(substr($line, 1, length($line) - length($1) - 1))) <= $max_line_length) {
263847e0c88bSJoe Perches				$msg_type = "LONG_LINE_STRING"
263947e0c88bSJoe Perches			}
264047e0c88bSJoe Perches
264147e0c88bSJoe Perches			if ($msg_type ne "" &&
264247e0c88bSJoe Perches			    (show_type("LONG_LINE") || show_type($msg_type))) {
264347e0c88bSJoe Perches				WARN($msg_type,
26446cd7f386SJoe Perches				     "line over $max_line_length characters\n" . $herecurr);
26450a920b5bSAndy Whitcroft			}
264647e0c88bSJoe Perches		}
26470a920b5bSAndy Whitcroft
26488905a67cSAndy Whitcroft# check for adding lines without a newline.
26498905a67cSAndy Whitcroft		if ($line =~ /^\+/ && defined $lines[$linenr] && $lines[$linenr] =~ /^\\ No newline at end of file/) {
2650000d1cc1SJoe Perches			WARN("MISSING_EOF_NEWLINE",
2651000d1cc1SJoe Perches			     "adding a line without newline at end of file\n" . $herecurr);
26528905a67cSAndy Whitcroft		}
26538905a67cSAndy Whitcroft
265442e41c54SMike Frysinger# Blackfin: use hi/lo macros
265542e41c54SMike Frysinger		if ($realfile =~ m@arch/blackfin/.*\.S$@) {
265642e41c54SMike Frysinger			if ($line =~ /\.[lL][[:space:]]*=.*&[[:space:]]*0x[fF][fF][fF][fF]/) {
265742e41c54SMike Frysinger				my $herevet = "$here\n" . cat_vet($line) . "\n";
2658000d1cc1SJoe Perches				ERROR("LO_MACRO",
2659000d1cc1SJoe Perches				      "use the LO() macro, not (... & 0xFFFF)\n" . $herevet);
266042e41c54SMike Frysinger			}
266142e41c54SMike Frysinger			if ($line =~ /\.[hH][[:space:]]*=.*>>[[:space:]]*16/) {
266242e41c54SMike Frysinger				my $herevet = "$here\n" . cat_vet($line) . "\n";
2663000d1cc1SJoe Perches				ERROR("HI_MACRO",
2664000d1cc1SJoe Perches				      "use the HI() macro, not (... >> 16)\n" . $herevet);
266542e41c54SMike Frysinger			}
266642e41c54SMike Frysinger		}
266742e41c54SMike Frysinger
2668b9ea10d6SAndy Whitcroft# check we are in a valid source file C or perl if not then ignore this hunk
2669de4c924cSGeert Uytterhoeven		next if ($realfile !~ /\.(h|c|pl|dtsi|dts)$/);
26700a920b5bSAndy Whitcroft
26710a920b5bSAndy Whitcroft# at the beginning of a line any tabs must come first and anything
26720a920b5bSAndy Whitcroft# more than 8 must use tabs.
2673c2fdda0dSAndy Whitcroft		if ($rawline =~ /^\+\s* \t\s*\S/ ||
2674c2fdda0dSAndy Whitcroft		    $rawline =~ /^\+\s*        \s*/) {
2675c2fdda0dSAndy Whitcroft			my $herevet = "$here\n" . cat_vet($rawline) . "\n";
2676d2c0a235SAndy Whitcroft			$rpt_cleaners = 1;
26773705ce5bSJoe Perches			if (ERROR("CODE_INDENT",
26783705ce5bSJoe Perches				  "code indent should use tabs where possible\n" . $herevet) &&
26793705ce5bSJoe Perches			    $fix) {
2680194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e;
26813705ce5bSJoe Perches			}
26820a920b5bSAndy Whitcroft		}
26830a920b5bSAndy Whitcroft
268408e44365SAlberto Panizzo# check for space before tabs.
268508e44365SAlberto Panizzo		if ($rawline =~ /^\+/ && $rawline =~ / \t/) {
268608e44365SAlberto Panizzo			my $herevet = "$here\n" . cat_vet($rawline) . "\n";
26873705ce5bSJoe Perches			if (WARN("SPACE_BEFORE_TAB",
26883705ce5bSJoe Perches				"please, no space before tabs\n" . $herevet) &&
26893705ce5bSJoe Perches			    $fix) {
2690194f66fcSJoe Perches				while ($fixed[$fixlinenr] =~
2691d2207ccbSJoe Perches					   s/(^\+.*) {8,8}\t/$1\t\t/) {}
2692194f66fcSJoe Perches				while ($fixed[$fixlinenr] =~
2693c76f4cb3SJoe Perches					   s/(^\+.*) +\t/$1\t/) {}
26943705ce5bSJoe Perches			}
269508e44365SAlberto Panizzo		}
269608e44365SAlberto Panizzo
2697d1fe9c09SJoe Perches# check for && or || at the start of a line
2698d1fe9c09SJoe Perches		if ($rawline =~ /^\+\s*(&&|\|\|)/) {
2699d1fe9c09SJoe Perches			CHK("LOGICAL_CONTINUATIONS",
2700d1fe9c09SJoe Perches			    "Logical continuations should be on the previous line\n" . $hereprev);
2701d1fe9c09SJoe Perches		}
2702d1fe9c09SJoe Perches
2703d1fe9c09SJoe Perches# check multi-line statement indentation matches previous line
2704d1fe9c09SJoe Perches		if ($^V && $^V ge 5.10.0 &&
270591cb5195SJoe Perches		    $prevline =~ /^\+([ \t]*)((?:$c90_Keywords(?:\s+if)\s*)|(?:$Declare\s*)?(?:$Ident|\(\s*\*\s*$Ident\s*\))\s*|$Ident\s*=\s*$Ident\s*)\(.*(\&\&|\|\||,)\s*$/) {
2706d1fe9c09SJoe Perches			$prevline =~ /^\+(\t*)(.*)$/;
2707d1fe9c09SJoe Perches			my $oldindent = $1;
2708d1fe9c09SJoe Perches			my $rest = $2;
2709d1fe9c09SJoe Perches
2710d1fe9c09SJoe Perches			my $pos = pos_last_openparen($rest);
2711d1fe9c09SJoe Perches			if ($pos >= 0) {
2712b34a26f3SJoe Perches				$line =~ /^(\+| )([ \t]*)/;
2713b34a26f3SJoe Perches				my $newindent = $2;
2714d1fe9c09SJoe Perches
2715d1fe9c09SJoe Perches				my $goodtabindent = $oldindent .
2716d1fe9c09SJoe Perches					"\t" x ($pos / 8) .
2717d1fe9c09SJoe Perches					" "  x ($pos % 8);
2718d1fe9c09SJoe Perches				my $goodspaceindent = $oldindent . " "  x $pos;
2719d1fe9c09SJoe Perches
2720d1fe9c09SJoe Perches				if ($newindent ne $goodtabindent &&
2721d1fe9c09SJoe Perches				    $newindent ne $goodspaceindent) {
27223705ce5bSJoe Perches
27233705ce5bSJoe Perches					if (CHK("PARENTHESIS_ALIGNMENT",
27243705ce5bSJoe Perches						"Alignment should match open parenthesis\n" . $hereprev) &&
27253705ce5bSJoe Perches					    $fix && $line =~ /^\+/) {
2726194f66fcSJoe Perches						$fixed[$fixlinenr] =~
27273705ce5bSJoe Perches						    s/^\+[ \t]*/\+$goodtabindent/;
27283705ce5bSJoe Perches					}
2729d1fe9c09SJoe Perches				}
2730d1fe9c09SJoe Perches			}
2731d1fe9c09SJoe Perches		}
2732d1fe9c09SJoe Perches
27336ab3a970SJoe Perches# check for space after cast like "(int) foo" or "(struct foo) bar"
27346ab3a970SJoe Perches# avoid checking a few false positives:
27356ab3a970SJoe Perches#   "sizeof(<type>)" or "__alignof__(<type>)"
27366ab3a970SJoe Perches#   function pointer declarations like "(*foo)(int) = bar;"
27376ab3a970SJoe Perches#   structure definitions like "(struct foo) { 0 };"
27386ab3a970SJoe Perches#   multiline macros that define functions
27396ab3a970SJoe Perches#   known attributes or the __attribute__ keyword
27406ab3a970SJoe Perches		if ($line =~ /^\+(.*)\(\s*$Type\s*\)([ \t]++)((?![={]|\\$|$Attribute|__attribute__))/ &&
27416ab3a970SJoe Perches		    (!defined($1) || $1 !~ /\b(?:sizeof|__alignof__)\s*$/)) {
27423705ce5bSJoe Perches			if (CHK("SPACING",
2743f27c95dbSJoe Perches				"No space is necessary after a cast\n" . $herecurr) &&
27443705ce5bSJoe Perches			    $fix) {
2745194f66fcSJoe Perches				$fixed[$fixlinenr] =~
2746f27c95dbSJoe Perches				    s/(\(\s*$Type\s*\))[ \t]+/$1/;
27473705ce5bSJoe Perches			}
2748aad4f614SJoe Perches		}
2749aad4f614SJoe Perches
275005880600SJoe Perches		if ($realfile =~ m@^(drivers/net/|net/)@ &&
2751fdb4bcd6SJoe Perches		    $prevrawline =~ /^\+[ \t]*\/\*[ \t]*$/ &&
275285ad978cSJoe Perches		    $rawline =~ /^\+[ \t]*\*/ &&
275385ad978cSJoe Perches		    $realline > 2) {
275405880600SJoe Perches			WARN("NETWORKING_BLOCK_COMMENT_STYLE",
275505880600SJoe Perches			     "networking block comments don't use an empty /* line, use /* Comment...\n" . $hereprev);
275605880600SJoe Perches		}
275705880600SJoe Perches
275805880600SJoe Perches		if ($realfile =~ m@^(drivers/net/|net/)@ &&
2759a605e32eSJoe Perches		    $prevrawline =~ /^\+[ \t]*\/\*/ &&		#starting /*
2760a605e32eSJoe Perches		    $prevrawline !~ /\*\/[ \t]*$/ &&		#no trailing */
276161135e96SJoe Perches		    $rawline =~ /^\+/ &&			#line is new
2762a605e32eSJoe Perches		    $rawline !~ /^\+[ \t]*\*/) {		#no leading *
2763a605e32eSJoe Perches			WARN("NETWORKING_BLOCK_COMMENT_STYLE",
2764a605e32eSJoe Perches			     "networking block comments start with * on subsequent lines\n" . $hereprev);
2765a605e32eSJoe Perches		}
2766a605e32eSJoe Perches
2767a605e32eSJoe Perches		if ($realfile =~ m@^(drivers/net/|net/)@ &&
2768c24f9f19SJoe Perches		    $rawline !~ m@^\+[ \t]*\*/[ \t]*$@ &&	#trailing */
2769c24f9f19SJoe Perches		    $rawline !~ m@^\+.*/\*.*\*/[ \t]*$@ &&	#inline /*...*/
2770c24f9f19SJoe Perches		    $rawline !~ m@^\+.*\*{2,}/[ \t]*$@ &&	#trailing **/
2771c24f9f19SJoe Perches		    $rawline =~ m@^\+[ \t]*.+\*\/[ \t]*$@) {	#non blank */
277205880600SJoe Perches			WARN("NETWORKING_BLOCK_COMMENT_STYLE",
277305880600SJoe Perches			     "networking block comments put the trailing */ on a separate line\n" . $herecurr);
277405880600SJoe Perches		}
277505880600SJoe Perches
27767f619191SJoe Perches# check for missing blank lines after struct/union declarations
27777f619191SJoe Perches# with exceptions for various attributes and macros
27787f619191SJoe Perches		if ($prevline =~ /^[\+ ]};?\s*$/ &&
27797f619191SJoe Perches		    $line =~ /^\+/ &&
27807f619191SJoe Perches		    !($line =~ /^\+\s*$/ ||
27817f619191SJoe Perches		      $line =~ /^\+\s*EXPORT_SYMBOL/ ||
27827f619191SJoe Perches		      $line =~ /^\+\s*MODULE_/i ||
27837f619191SJoe Perches		      $line =~ /^\+\s*\#\s*(?:end|elif|else)/ ||
27847f619191SJoe Perches		      $line =~ /^\+[a-z_]*init/ ||
27857f619191SJoe Perches		      $line =~ /^\+\s*(?:static\s+)?[A-Z_]*ATTR/ ||
27867f619191SJoe Perches		      $line =~ /^\+\s*DECLARE/ ||
27877f619191SJoe Perches		      $line =~ /^\+\s*__setup/)) {
2788d752fcc8SJoe Perches			if (CHK("LINE_SPACING",
2789d752fcc8SJoe Perches				"Please use a blank line after function/struct/union/enum declarations\n" . $hereprev) &&
2790d752fcc8SJoe Perches			    $fix) {
2791f2d7e4d4SJoe Perches				fix_insert_line($fixlinenr, "\+");
2792d752fcc8SJoe Perches			}
27937f619191SJoe Perches		}
27947f619191SJoe Perches
2795365dd4eaSJoe Perches# check for multiple consecutive blank lines
2796365dd4eaSJoe Perches		if ($prevline =~ /^[\+ ]\s*$/ &&
2797365dd4eaSJoe Perches		    $line =~ /^\+\s*$/ &&
2798365dd4eaSJoe Perches		    $last_blank_line != ($linenr - 1)) {
2799d752fcc8SJoe Perches			if (CHK("LINE_SPACING",
2800d752fcc8SJoe Perches				"Please don't use multiple blank lines\n" . $hereprev) &&
2801d752fcc8SJoe Perches			    $fix) {
2802f2d7e4d4SJoe Perches				fix_delete_line($fixlinenr, $rawline);
2803d752fcc8SJoe Perches			}
2804d752fcc8SJoe Perches
2805365dd4eaSJoe Perches			$last_blank_line = $linenr;
2806365dd4eaSJoe Perches		}
2807365dd4eaSJoe Perches
28083b617e3bSJoe Perches# check for missing blank lines after declarations
28093f7bac03SJoe Perches		if ($sline =~ /^\+\s+\S/ &&			#Not at char 1
28103f7bac03SJoe Perches			# actual declarations
28113f7bac03SJoe Perches		    ($prevline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ ||
28125a4e1fd3SJoe Perches			# function pointer declarations
28135a4e1fd3SJoe Perches		     $prevline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ ||
28143f7bac03SJoe Perches			# foo bar; where foo is some local typedef or #define
28153f7bac03SJoe Perches		     $prevline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ ||
28163f7bac03SJoe Perches			# known declaration macros
28173f7bac03SJoe Perches		     $prevline =~ /^\+\s+$declaration_macros/) &&
28183f7bac03SJoe Perches			# for "else if" which can look like "$Ident $Ident"
28193f7bac03SJoe Perches		    !($prevline =~ /^\+\s+$c90_Keywords\b/ ||
28203f7bac03SJoe Perches			# other possible extensions of declaration lines
28213f7bac03SJoe Perches		      $prevline =~ /(?:$Compare|$Assignment|$Operators)\s*$/ ||
28223f7bac03SJoe Perches			# not starting a section or a macro "\" extended line
28233f7bac03SJoe Perches		      $prevline =~ /(?:\{\s*|\\)$/) &&
28243f7bac03SJoe Perches			# looks like a declaration
28253f7bac03SJoe Perches		    !($sline =~ /^\+\s+$Declare\s*$Ident\s*[=,;:\[]/ ||
28265a4e1fd3SJoe Perches			# function pointer declarations
28275a4e1fd3SJoe Perches		      $sline =~ /^\+\s+$Declare\s*\(\s*\*\s*$Ident\s*\)\s*[=,;:\[\(]/ ||
28283f7bac03SJoe Perches			# foo bar; where foo is some local typedef or #define
28293f7bac03SJoe Perches		      $sline =~ /^\+\s+$Ident(?:\s+|\s*\*\s*)$Ident\s*[=,;\[]/ ||
28303f7bac03SJoe Perches			# known declaration macros
28313f7bac03SJoe Perches		      $sline =~ /^\+\s+$declaration_macros/ ||
28323f7bac03SJoe Perches			# start of struct or union or enum
28333b617e3bSJoe Perches		      $sline =~ /^\+\s+(?:union|struct|enum|typedef)\b/ ||
28343f7bac03SJoe Perches			# start or end of block or continuation of declaration
28353f7bac03SJoe Perches		      $sline =~ /^\+\s+(?:$|[\{\}\.\#\"\?\:\(\[])/ ||
28363f7bac03SJoe Perches			# bitfield continuation
28373f7bac03SJoe Perches		      $sline =~ /^\+\s+$Ident\s*:\s*\d+\s*[,;]/ ||
28383f7bac03SJoe Perches			# other possible extensions of declaration lines
28393f7bac03SJoe Perches		      $sline =~ /^\+\s+\(?\s*(?:$Compare|$Assignment|$Operators)/) &&
28403f7bac03SJoe Perches			# indentation of previous and current line are the same
28413f7bac03SJoe Perches		    (($prevline =~ /\+(\s+)\S/) && $sline =~ /^\+$1\S/)) {
2842d752fcc8SJoe Perches			if (WARN("LINE_SPACING",
2843d752fcc8SJoe Perches				 "Missing a blank line after declarations\n" . $hereprev) &&
2844d752fcc8SJoe Perches			    $fix) {
2845f2d7e4d4SJoe Perches				fix_insert_line($fixlinenr, "\+");
2846d752fcc8SJoe Perches			}
28473b617e3bSJoe Perches		}
28483b617e3bSJoe Perches
28495f7ddae6SRaffaele Recalcati# check for spaces at the beginning of a line.
28506b4c5bebSAndy Whitcroft# Exceptions:
28516b4c5bebSAndy Whitcroft#  1) within comments
28526b4c5bebSAndy Whitcroft#  2) indented preprocessor commands
28536b4c5bebSAndy Whitcroft#  3) hanging labels
28543705ce5bSJoe Perches		if ($rawline =~ /^\+ / && $line !~ /^\+ *(?:$;|#|$Ident:)/)  {
28555f7ddae6SRaffaele Recalcati			my $herevet = "$here\n" . cat_vet($rawline) . "\n";
28563705ce5bSJoe Perches			if (WARN("LEADING_SPACE",
28573705ce5bSJoe Perches				 "please, no spaces at the start of a line\n" . $herevet) &&
28583705ce5bSJoe Perches			    $fix) {
2859194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/^\+([ \t]+)/"\+" . tabify($1)/e;
28603705ce5bSJoe Perches			}
28615f7ddae6SRaffaele Recalcati		}
28625f7ddae6SRaffaele Recalcati
2863b9ea10d6SAndy Whitcroft# check we are in a valid C source file if not then ignore this hunk
2864b9ea10d6SAndy Whitcroft		next if ($realfile !~ /\.(h|c)$/);
2865b9ea10d6SAndy Whitcroft
2866032a4c0fSJoe Perches# check indentation of any line with a bare else
2867840080a0SJoe Perches# (but not if it is a multiple line "if (foo) return bar; else return baz;")
2868032a4c0fSJoe Perches# if the previous line is a break or return and is indented 1 tab more...
2869032a4c0fSJoe Perches		if ($sline =~ /^\+([\t]+)(?:}[ \t]*)?else(?:[ \t]*{)?\s*$/) {
2870032a4c0fSJoe Perches			my $tabs = length($1) + 1;
2871840080a0SJoe Perches			if ($prevline =~ /^\+\t{$tabs,$tabs}break\b/ ||
2872840080a0SJoe Perches			    ($prevline =~ /^\+\t{$tabs,$tabs}return\b/ &&
2873840080a0SJoe Perches			     defined $lines[$linenr] &&
2874840080a0SJoe Perches			     $lines[$linenr] !~ /^[ \+]\t{$tabs,$tabs}return/)) {
2875032a4c0fSJoe Perches				WARN("UNNECESSARY_ELSE",
2876032a4c0fSJoe Perches				     "else is not generally useful after a break or return\n" . $hereprev);
2877032a4c0fSJoe Perches			}
2878032a4c0fSJoe Perches		}
2879032a4c0fSJoe Perches
2880c00df19aSJoe Perches# check indentation of a line with a break;
2881c00df19aSJoe Perches# if the previous line is a goto or return and is indented the same # of tabs
2882c00df19aSJoe Perches		if ($sline =~ /^\+([\t]+)break\s*;\s*$/) {
2883c00df19aSJoe Perches			my $tabs = $1;
2884c00df19aSJoe Perches			if ($prevline =~ /^\+$tabs(?:goto|return)\b/) {
2885c00df19aSJoe Perches				WARN("UNNECESSARY_BREAK",
2886c00df19aSJoe Perches				     "break is not useful after a goto or return\n" . $hereprev);
2887c00df19aSJoe Perches			}
2888c00df19aSJoe Perches		}
2889c00df19aSJoe Perches
28901ba8dfd1SKees Cook# discourage the addition of CONFIG_EXPERIMENTAL in #if(def).
28911ba8dfd1SKees Cook		if ($line =~ /^\+\s*\#\s*if.*\bCONFIG_EXPERIMENTAL\b/) {
28921ba8dfd1SKees Cook			WARN("CONFIG_EXPERIMENTAL",
28931ba8dfd1SKees Cook			     "Use of CONFIG_EXPERIMENTAL is deprecated. For alternatives, see https://lkml.org/lkml/2012/10/23/580\n");
28941ba8dfd1SKees Cook		}
28951ba8dfd1SKees Cook
2896c2fdda0dSAndy Whitcroft# check for RCS/CVS revision markers
2897cf655043SAndy Whitcroft		if ($rawline =~ /^\+.*\$(Revision|Log|Id)(?:\$|)/) {
2898000d1cc1SJoe Perches			WARN("CVS_KEYWORD",
2899000d1cc1SJoe Perches			     "CVS style keyword markers, these will _not_ be updated\n". $herecurr);
2900c2fdda0dSAndy Whitcroft		}
290122f2a2efSAndy Whitcroft
290242e41c54SMike Frysinger# Blackfin: don't use __builtin_bfin_[cs]sync
290342e41c54SMike Frysinger		if ($line =~ /__builtin_bfin_csync/) {
290442e41c54SMike Frysinger			my $herevet = "$here\n" . cat_vet($line) . "\n";
2905000d1cc1SJoe Perches			ERROR("CSYNC",
2906000d1cc1SJoe Perches			      "use the CSYNC() macro in asm/blackfin.h\n" . $herevet);
290742e41c54SMike Frysinger		}
290842e41c54SMike Frysinger		if ($line =~ /__builtin_bfin_ssync/) {
290942e41c54SMike Frysinger			my $herevet = "$here\n" . cat_vet($line) . "\n";
2910000d1cc1SJoe Perches			ERROR("SSYNC",
2911000d1cc1SJoe Perches			      "use the SSYNC() macro in asm/blackfin.h\n" . $herevet);
291242e41c54SMike Frysinger		}
291342e41c54SMike Frysinger
291456e77d70SJoe Perches# check for old HOTPLUG __dev<foo> section markings
291556e77d70SJoe Perches		if ($line =~ /\b(__dev(init|exit)(data|const|))\b/) {
291656e77d70SJoe Perches			WARN("HOTPLUG_SECTION",
291756e77d70SJoe Perches			     "Using $1 is unnecessary\n" . $herecurr);
291856e77d70SJoe Perches		}
291956e77d70SJoe Perches
29209c0ca6f9SAndy Whitcroft# Check for potential 'bare' types
29212b474a1aSAndy Whitcroft		my ($stat, $cond, $line_nr_next, $remain_next, $off_next,
29222b474a1aSAndy Whitcroft		    $realline_next);
29233e469cdcSAndy Whitcroft#print "LINE<$line>\n";
29243e469cdcSAndy Whitcroft		if ($linenr >= $suppress_statement &&
29251b5539b1SJoe Perches		    $realcnt && $sline =~ /.\s*\S/) {
2926170d3a22SAndy Whitcroft			($stat, $cond, $line_nr_next, $remain_next, $off_next) =
2927f5fe35ddSAndy Whitcroft				ctx_statement_block($linenr, $realcnt, 0);
2928171ae1a4SAndy Whitcroft			$stat =~ s/\n./\n /g;
2929171ae1a4SAndy Whitcroft			$cond =~ s/\n./\n /g;
2930171ae1a4SAndy Whitcroft
29313e469cdcSAndy Whitcroft#print "linenr<$linenr> <$stat>\n";
29323e469cdcSAndy Whitcroft			# If this statement has no statement boundaries within
29333e469cdcSAndy Whitcroft			# it there is no point in retrying a statement scan
29343e469cdcSAndy Whitcroft			# until we hit end of it.
29353e469cdcSAndy Whitcroft			my $frag = $stat; $frag =~ s/;+\s*$//;
29363e469cdcSAndy Whitcroft			if ($frag !~ /(?:{|;)/) {
29373e469cdcSAndy Whitcroft#print "skip<$line_nr_next>\n";
29383e469cdcSAndy Whitcroft				$suppress_statement = $line_nr_next;
29393e469cdcSAndy Whitcroft			}
2940f74bd194SAndy Whitcroft
29412b474a1aSAndy Whitcroft			# Find the real next line.
29422b474a1aSAndy Whitcroft			$realline_next = $line_nr_next;
29432b474a1aSAndy Whitcroft			if (defined $realline_next &&
29442b474a1aSAndy Whitcroft			    (!defined $lines[$realline_next - 1] ||
29452b474a1aSAndy Whitcroft			     substr($lines[$realline_next - 1], $off_next) =~ /^\s*$/)) {
29462b474a1aSAndy Whitcroft				$realline_next++;
29472b474a1aSAndy Whitcroft			}
29482b474a1aSAndy Whitcroft
2949171ae1a4SAndy Whitcroft			my $s = $stat;
2950171ae1a4SAndy Whitcroft			$s =~ s/{.*$//s;
2951cf655043SAndy Whitcroft
2952c2fdda0dSAndy Whitcroft			# Ignore goto labels.
2953171ae1a4SAndy Whitcroft			if ($s =~ /$Ident:\*$/s) {
2954c2fdda0dSAndy Whitcroft
2955c2fdda0dSAndy Whitcroft			# Ignore functions being called
2956171ae1a4SAndy Whitcroft			} elsif ($s =~ /^.\s*$Ident\s*\(/s) {
2957c2fdda0dSAndy Whitcroft
2958463f2864SAndy Whitcroft			} elsif ($s =~ /^.\s*else\b/s) {
2959463f2864SAndy Whitcroft
2960c45dcabdSAndy Whitcroft			# declarations always start with types
2961d2506586SAndy Whitcroft			} 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) {
2962c45dcabdSAndy Whitcroft				my $type = $1;
2963c45dcabdSAndy Whitcroft				$type =~ s/\s+/ /g;
2964c45dcabdSAndy Whitcroft				possible($type, "A:" . $s);
2965c45dcabdSAndy Whitcroft
29666c72ffaaSAndy Whitcroft			# definitions in global scope can only start with types
2967a6a84062SAndy Whitcroft			} elsif ($s =~ /^.(?:$Storage\s+)?(?:$Inline\s+)?(?:const\s+)?($Ident)\b\s*(?!:)/s) {
2968c45dcabdSAndy Whitcroft				possible($1, "B:" . $s);
2969c2fdda0dSAndy Whitcroft			}
29708905a67cSAndy Whitcroft
29716c72ffaaSAndy Whitcroft			# any (foo ... *) is a pointer cast, and foo is a type
297265863862SAndy Whitcroft			while ($s =~ /\(($Ident)(?:\s+$Sparse)*[\s\*]+\s*\)/sg) {
2973c45dcabdSAndy Whitcroft				possible($1, "C:" . $s);
29749c0ca6f9SAndy Whitcroft			}
29758905a67cSAndy Whitcroft
29768905a67cSAndy Whitcroft			# Check for any sort of function declaration.
29778905a67cSAndy Whitcroft			# int foo(something bar, other baz);
29788905a67cSAndy Whitcroft			# void (*store_gdt)(x86_descr_ptr *);
2979171ae1a4SAndy Whitcroft			if ($prev_values eq 'E' && $s =~ /^(.(?:typedef\s*)?(?:(?:$Storage|$Inline)\s*)*\s*$Type\s*(?:\b$Ident|\(\*\s*$Ident\))\s*)\(/s) {
29808905a67cSAndy Whitcroft				my ($name_len) = length($1);
29818905a67cSAndy Whitcroft
2982cf655043SAndy Whitcroft				my $ctx = $s;
2983773647a0SAndy Whitcroft				substr($ctx, 0, $name_len + 1, '');
29848905a67cSAndy Whitcroft				$ctx =~ s/\)[^\)]*$//;
2985cf655043SAndy Whitcroft
29868905a67cSAndy Whitcroft				for my $arg (split(/\s*,\s*/, $ctx)) {
2987c45dcabdSAndy Whitcroft					if ($arg =~ /^(?:const\s+)?($Ident)(?:\s+$Sparse)*\s*\**\s*(:?\b$Ident)?$/s || $arg =~ /^($Ident)$/s) {
29888905a67cSAndy Whitcroft
2989c45dcabdSAndy Whitcroft						possible($1, "D:" . $s);
29908905a67cSAndy Whitcroft					}
29918905a67cSAndy Whitcroft				}
29928905a67cSAndy Whitcroft			}
29938905a67cSAndy Whitcroft
29949c0ca6f9SAndy Whitcroft		}
29959c0ca6f9SAndy Whitcroft
299600df344fSAndy Whitcroft#
299700df344fSAndy Whitcroft# Checks which may be anchored in the context.
299800df344fSAndy Whitcroft#
299900df344fSAndy Whitcroft
300000df344fSAndy Whitcroft# Check for switch () and associated case and default
300100df344fSAndy Whitcroft# statements should be at the same indent.
300200df344fSAndy Whitcroft		if ($line=~/\bswitch\s*\(.*\)/) {
300300df344fSAndy Whitcroft			my $err = '';
300400df344fSAndy Whitcroft			my $sep = '';
300500df344fSAndy Whitcroft			my @ctx = ctx_block_outer($linenr, $realcnt);
300600df344fSAndy Whitcroft			shift(@ctx);
300700df344fSAndy Whitcroft			for my $ctx (@ctx) {
300800df344fSAndy Whitcroft				my ($clen, $cindent) = line_stats($ctx);
300900df344fSAndy Whitcroft				if ($ctx =~ /^\+\s*(case\s+|default:)/ &&
301000df344fSAndy Whitcroft							$indent != $cindent) {
301100df344fSAndy Whitcroft					$err .= "$sep$ctx\n";
301200df344fSAndy Whitcroft					$sep = '';
301300df344fSAndy Whitcroft				} else {
301400df344fSAndy Whitcroft					$sep = "[...]\n";
301500df344fSAndy Whitcroft				}
301600df344fSAndy Whitcroft			}
301700df344fSAndy Whitcroft			if ($err ne '') {
3018000d1cc1SJoe Perches				ERROR("SWITCH_CASE_INDENT_LEVEL",
3019000d1cc1SJoe Perches				      "switch and case should be at the same indent\n$hereline$err");
3020de7d4f0eSAndy Whitcroft			}
3021de7d4f0eSAndy Whitcroft		}
3022de7d4f0eSAndy Whitcroft
3023de7d4f0eSAndy Whitcroft# if/while/etc brace do not go on next line, unless defining a do while loop,
3024de7d4f0eSAndy Whitcroft# or if that brace on the next line is for something else
30250fe3dc2bSJoe Perches		if ($line =~ /(.*)\b((?:if|while|for|switch|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b|else\b)/ && $line !~ /^.\s*\#/) {
3026773647a0SAndy Whitcroft			my $pre_ctx = "$1$2";
3027773647a0SAndy Whitcroft
30289c0ca6f9SAndy Whitcroft			my ($level, @ctx) = ctx_statement_level($linenr, $realcnt, 0);
30298eef05ddSJoe Perches
30308eef05ddSJoe Perches			if ($line =~ /^\+\t{6,}/) {
30318eef05ddSJoe Perches				WARN("DEEP_INDENTATION",
30328eef05ddSJoe Perches				     "Too many leading tabs - consider code refactoring\n" . $herecurr);
30338eef05ddSJoe Perches			}
30348eef05ddSJoe Perches
3035de7d4f0eSAndy Whitcroft			my $ctx_cnt = $realcnt - $#ctx - 1;
3036de7d4f0eSAndy Whitcroft			my $ctx = join("\n", @ctx);
3037de7d4f0eSAndy Whitcroft
3038548596d5SAndy Whitcroft			my $ctx_ln = $linenr;
3039548596d5SAndy Whitcroft			my $ctx_skip = $realcnt;
3040de7d4f0eSAndy Whitcroft
3041548596d5SAndy Whitcroft			while ($ctx_skip > $ctx_cnt || ($ctx_skip == $ctx_cnt &&
3042548596d5SAndy Whitcroft					defined $lines[$ctx_ln - 1] &&
3043548596d5SAndy Whitcroft					$lines[$ctx_ln - 1] =~ /^-/)) {
3044548596d5SAndy Whitcroft				##print "SKIP<$ctx_skip> CNT<$ctx_cnt>\n";
3045548596d5SAndy Whitcroft				$ctx_skip-- if (!defined $lines[$ctx_ln - 1] || $lines[$ctx_ln - 1] !~ /^-/);
3046773647a0SAndy Whitcroft				$ctx_ln++;
3047773647a0SAndy Whitcroft			}
3048548596d5SAndy Whitcroft
304953210168SAndy Whitcroft			#print "realcnt<$realcnt> ctx_cnt<$ctx_cnt>\n";
305053210168SAndy Whitcroft			#print "pre<$pre_ctx>\nline<$line>\nctx<$ctx>\nnext<$lines[$ctx_ln - 1]>\n";
3051773647a0SAndy Whitcroft
3052773647a0SAndy Whitcroft			if ($ctx !~ /{\s*/ && defined($lines[$ctx_ln - 1]) && $lines[$ctx_ln - 1] =~ /^\+\s*{/) {
3053000d1cc1SJoe Perches				ERROR("OPEN_BRACE",
3054000d1cc1SJoe Perches				      "that open brace { should be on the previous line\n" .
305501464f30SAndy Whitcroft					"$here\n$ctx\n$rawlines[$ctx_ln - 1]\n");
305600df344fSAndy Whitcroft			}
3057773647a0SAndy Whitcroft			if ($level == 0 && $pre_ctx !~ /}\s*while\s*\($/ &&
3058773647a0SAndy Whitcroft			    $ctx =~ /\)\s*\;\s*$/ &&
3059773647a0SAndy Whitcroft			    defined $lines[$ctx_ln - 1])
3060773647a0SAndy Whitcroft			{
30619c0ca6f9SAndy Whitcroft				my ($nlength, $nindent) = line_stats($lines[$ctx_ln - 1]);
30629c0ca6f9SAndy Whitcroft				if ($nindent > $indent) {
3063000d1cc1SJoe Perches					WARN("TRAILING_SEMICOLON",
3064000d1cc1SJoe Perches					     "trailing semicolon indicates no statements, indent implies otherwise\n" .
306501464f30SAndy Whitcroft						"$here\n$ctx\n$rawlines[$ctx_ln - 1]\n");
30669c0ca6f9SAndy Whitcroft				}
30679c0ca6f9SAndy Whitcroft			}
306800df344fSAndy Whitcroft		}
306900df344fSAndy Whitcroft
30704d001e4dSAndy Whitcroft# Check relative indent for conditionals and blocks.
30710fe3dc2bSJoe Perches		if ($line =~ /\b(?:(?:if|while|for|(?:[a-z_]+|)for_each[a-z_]+)\s*\(|do\b)/ && $line !~ /^.\s*#/ && $line !~ /\}\s*while\s*/) {
30723e469cdcSAndy Whitcroft			($stat, $cond, $line_nr_next, $remain_next, $off_next) =
30733e469cdcSAndy Whitcroft				ctx_statement_block($linenr, $realcnt, 0)
30743e469cdcSAndy Whitcroft					if (!defined $stat);
30754d001e4dSAndy Whitcroft			my ($s, $c) = ($stat, $cond);
30764d001e4dSAndy Whitcroft
30774d001e4dSAndy Whitcroft			substr($s, 0, length($c), '');
30784d001e4dSAndy Whitcroft
30794d001e4dSAndy Whitcroft			# Make sure we remove the line prefixes as we have
30804d001e4dSAndy Whitcroft			# none on the first line, and are going to readd them
30814d001e4dSAndy Whitcroft			# where necessary.
30824d001e4dSAndy Whitcroft			$s =~ s/\n./\n/gs;
30834d001e4dSAndy Whitcroft
30844d001e4dSAndy Whitcroft			# Find out how long the conditional actually is.
30856f779c18SAndy Whitcroft			my @newlines = ($c =~ /\n/gs);
30866f779c18SAndy Whitcroft			my $cond_lines = 1 + $#newlines;
30874d001e4dSAndy Whitcroft
30884d001e4dSAndy Whitcroft			# We want to check the first line inside the block
30894d001e4dSAndy Whitcroft			# starting at the end of the conditional, so remove:
30904d001e4dSAndy Whitcroft			#  1) any blank line termination
30914d001e4dSAndy Whitcroft			#  2) any opening brace { on end of the line
30924d001e4dSAndy Whitcroft			#  3) any do (...) {
30934d001e4dSAndy Whitcroft			my $continuation = 0;
30944d001e4dSAndy Whitcroft			my $check = 0;
30954d001e4dSAndy Whitcroft			$s =~ s/^.*\bdo\b//;
30964d001e4dSAndy Whitcroft			$s =~ s/^\s*{//;
30974d001e4dSAndy Whitcroft			if ($s =~ s/^\s*\\//) {
30984d001e4dSAndy Whitcroft				$continuation = 1;
30994d001e4dSAndy Whitcroft			}
31009bd49efeSAndy Whitcroft			if ($s =~ s/^\s*?\n//) {
31014d001e4dSAndy Whitcroft				$check = 1;
31024d001e4dSAndy Whitcroft				$cond_lines++;
31034d001e4dSAndy Whitcroft			}
31044d001e4dSAndy Whitcroft
31054d001e4dSAndy Whitcroft			# Also ignore a loop construct at the end of a
31064d001e4dSAndy Whitcroft			# preprocessor statement.
31074d001e4dSAndy Whitcroft			if (($prevline =~ /^.\s*#\s*define\s/ ||
31084d001e4dSAndy Whitcroft			    $prevline =~ /\\\s*$/) && $continuation == 0) {
31094d001e4dSAndy Whitcroft				$check = 0;
31104d001e4dSAndy Whitcroft			}
31114d001e4dSAndy Whitcroft
31129bd49efeSAndy Whitcroft			my $cond_ptr = -1;
3113740504c6SAndy Whitcroft			$continuation = 0;
31149bd49efeSAndy Whitcroft			while ($cond_ptr != $cond_lines) {
31159bd49efeSAndy Whitcroft				$cond_ptr = $cond_lines;
31164d001e4dSAndy Whitcroft
3117f16fa28fSAndy Whitcroft				# If we see an #else/#elif then the code
3118f16fa28fSAndy Whitcroft				# is not linear.
3119f16fa28fSAndy Whitcroft				if ($s =~ /^\s*\#\s*(?:else|elif)/) {
3120f16fa28fSAndy Whitcroft					$check = 0;
3121f16fa28fSAndy Whitcroft				}
3122f16fa28fSAndy Whitcroft
31239bd49efeSAndy Whitcroft				# Ignore:
31249bd49efeSAndy Whitcroft				#  1) blank lines, they should be at 0,
31259bd49efeSAndy Whitcroft				#  2) preprocessor lines, and
31269bd49efeSAndy Whitcroft				#  3) labels.
3127740504c6SAndy Whitcroft				if ($continuation ||
3128740504c6SAndy Whitcroft				    $s =~ /^\s*?\n/ ||
31299bd49efeSAndy Whitcroft				    $s =~ /^\s*#\s*?/ ||
31309bd49efeSAndy Whitcroft				    $s =~ /^\s*$Ident\s*:/) {
3131740504c6SAndy Whitcroft					$continuation = ($s =~ /^.*?\\\n/) ? 1 : 0;
313230dad6ebSAndy Whitcroft					if ($s =~ s/^.*?\n//) {
31339bd49efeSAndy Whitcroft						$cond_lines++;
31349bd49efeSAndy Whitcroft					}
31354d001e4dSAndy Whitcroft				}
313630dad6ebSAndy Whitcroft			}
31374d001e4dSAndy Whitcroft
31384d001e4dSAndy Whitcroft			my (undef, $sindent) = line_stats("+" . $s);
31394d001e4dSAndy Whitcroft			my $stat_real = raw_line($linenr, $cond_lines);
31404d001e4dSAndy Whitcroft
31414d001e4dSAndy Whitcroft			# Check if either of these lines are modified, else
31424d001e4dSAndy Whitcroft			# this is not this patch's fault.
31434d001e4dSAndy Whitcroft			if (!defined($stat_real) ||
31444d001e4dSAndy Whitcroft			    $stat !~ /^\+/ && $stat_real !~ /^\+/) {
31454d001e4dSAndy Whitcroft				$check = 0;
31464d001e4dSAndy Whitcroft			}
31474d001e4dSAndy Whitcroft			if (defined($stat_real) && $cond_lines > 1) {
31484d001e4dSAndy Whitcroft				$stat_real = "[...]\n$stat_real";
31494d001e4dSAndy Whitcroft			}
31504d001e4dSAndy Whitcroft
31519bd49efeSAndy Whitcroft			#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";
31524d001e4dSAndy Whitcroft
31534d001e4dSAndy Whitcroft			if ($check && (($sindent % 8) != 0 ||
31544d001e4dSAndy Whitcroft			    ($sindent <= $indent && $s ne ''))) {
3155000d1cc1SJoe Perches				WARN("SUSPECT_CODE_INDENT",
3156000d1cc1SJoe Perches				     "suspect code indent for conditional statements ($indent, $sindent)\n" . $herecurr . "$stat_real\n");
31574d001e4dSAndy Whitcroft			}
31584d001e4dSAndy Whitcroft		}
31594d001e4dSAndy Whitcroft
31606c72ffaaSAndy Whitcroft		# Track the 'values' across context and added lines.
31616c72ffaaSAndy Whitcroft		my $opline = $line; $opline =~ s/^./ /;
31621f65f947SAndy Whitcroft		my ($curr_values, $curr_vars) =
31631f65f947SAndy Whitcroft				annotate_values($opline . "\n", $prev_values);
31646c72ffaaSAndy Whitcroft		$curr_values = $prev_values . $curr_values;
3165c2fdda0dSAndy Whitcroft		if ($dbg_values) {
3166c2fdda0dSAndy Whitcroft			my $outline = $opline; $outline =~ s/\t/ /g;
3167cf655043SAndy Whitcroft			print "$linenr > .$outline\n";
3168cf655043SAndy Whitcroft			print "$linenr > $curr_values\n";
31691f65f947SAndy Whitcroft			print "$linenr >  $curr_vars\n";
3170c2fdda0dSAndy Whitcroft		}
31716c72ffaaSAndy Whitcroft		$prev_values = substr($curr_values, -1);
31726c72ffaaSAndy Whitcroft
317300df344fSAndy Whitcroft#ignore lines not being added
31743705ce5bSJoe Perches		next if ($line =~ /^[^\+]/);
317500df344fSAndy Whitcroft
3176653d4876SAndy Whitcroft# TEST: allow direct testing of the type matcher.
31777429c690SAndy Whitcroft		if ($dbg_type) {
31787429c690SAndy Whitcroft			if ($line =~ /^.\s*$Declare\s*$/) {
3179000d1cc1SJoe Perches				ERROR("TEST_TYPE",
3180000d1cc1SJoe Perches				      "TEST: is type\n" . $herecurr);
31817429c690SAndy Whitcroft			} elsif ($dbg_type > 1 && $line =~ /^.+($Declare)/) {
3182000d1cc1SJoe Perches				ERROR("TEST_NOT_TYPE",
3183000d1cc1SJoe Perches				      "TEST: is not type ($1 is)\n". $herecurr);
31847429c690SAndy Whitcroft			}
3185653d4876SAndy Whitcroft			next;
3186653d4876SAndy Whitcroft		}
3187a1ef277eSAndy Whitcroft# TEST: allow direct testing of the attribute matcher.
3188a1ef277eSAndy Whitcroft		if ($dbg_attr) {
31899360b0e5SAndy Whitcroft			if ($line =~ /^.\s*$Modifier\s*$/) {
3190000d1cc1SJoe Perches				ERROR("TEST_ATTR",
3191000d1cc1SJoe Perches				      "TEST: is attr\n" . $herecurr);
31929360b0e5SAndy Whitcroft			} elsif ($dbg_attr > 1 && $line =~ /^.+($Modifier)/) {
3193000d1cc1SJoe Perches				ERROR("TEST_NOT_ATTR",
3194000d1cc1SJoe Perches				      "TEST: is not attr ($1 is)\n". $herecurr);
3195a1ef277eSAndy Whitcroft			}
3196a1ef277eSAndy Whitcroft			next;
3197a1ef277eSAndy Whitcroft		}
3198653d4876SAndy Whitcroft
3199f0a594c1SAndy Whitcroft# check for initialisation to aggregates open brace on the next line
320099423c20SAndy Whitcroft		if ($line =~ /^.\s*{/ &&
320199423c20SAndy Whitcroft		    $prevline =~ /(?:^|[^=])=\s*$/) {
3202d752fcc8SJoe Perches			if (ERROR("OPEN_BRACE",
3203d752fcc8SJoe Perches				  "that open brace { should be on the previous line\n" . $hereprev) &&
3204f2d7e4d4SJoe Perches			    $fix && $prevline =~ /^\+/ && $line =~ /^\+/) {
3205f2d7e4d4SJoe Perches				fix_delete_line($fixlinenr - 1, $prevrawline);
3206f2d7e4d4SJoe Perches				fix_delete_line($fixlinenr, $rawline);
3207d752fcc8SJoe Perches				my $fixedline = $prevrawline;
3208d752fcc8SJoe Perches				$fixedline =~ s/\s*=\s*$/ = {/;
3209f2d7e4d4SJoe Perches				fix_insert_line($fixlinenr, $fixedline);
3210d752fcc8SJoe Perches				$fixedline = $line;
3211d752fcc8SJoe Perches				$fixedline =~ s/^(.\s*){\s*/$1/;
3212f2d7e4d4SJoe Perches				fix_insert_line($fixlinenr, $fixedline);
3213d752fcc8SJoe Perches			}
3214f0a594c1SAndy Whitcroft		}
3215f0a594c1SAndy Whitcroft
321600df344fSAndy Whitcroft#
321700df344fSAndy Whitcroft# Checks which are anchored on the added line.
321800df344fSAndy Whitcroft#
321900df344fSAndy Whitcroft
3220653d4876SAndy Whitcroft# check for malformed paths in #include statements (uses RAW line)
3221c45dcabdSAndy Whitcroft		if ($rawline =~ m{^.\s*\#\s*include\s+[<"](.*)[">]}) {
3222653d4876SAndy Whitcroft			my $path = $1;
3223653d4876SAndy Whitcroft			if ($path =~ m{//}) {
3224000d1cc1SJoe Perches				ERROR("MALFORMED_INCLUDE",
3225495e9d84SJoe Perches				      "malformed #include filename\n" . $herecurr);
3226495e9d84SJoe Perches			}
3227495e9d84SJoe Perches			if ($path =~ "^uapi/" && $realfile =~ m@\binclude/uapi/@) {
3228495e9d84SJoe Perches				ERROR("UAPI_INCLUDE",
3229495e9d84SJoe Perches				      "No #include in ...include/uapi/... should use a uapi/ path prefix\n" . $herecurr);
3230653d4876SAndy Whitcroft			}
3231653d4876SAndy Whitcroft		}
3232653d4876SAndy Whitcroft
323300df344fSAndy Whitcroft# no C99 // comments
323400df344fSAndy Whitcroft		if ($line =~ m{//}) {
32353705ce5bSJoe Perches			if (ERROR("C99_COMMENTS",
32363705ce5bSJoe Perches				  "do not use C99 // comments\n" . $herecurr) &&
32373705ce5bSJoe Perches			    $fix) {
3238194f66fcSJoe Perches				my $line = $fixed[$fixlinenr];
32393705ce5bSJoe Perches				if ($line =~ /\/\/(.*)$/) {
32403705ce5bSJoe Perches					my $comment = trim($1);
3241194f66fcSJoe Perches					$fixed[$fixlinenr] =~ s@\/\/(.*)$@/\* $comment \*/@;
32423705ce5bSJoe Perches				}
32433705ce5bSJoe Perches			}
324400df344fSAndy Whitcroft		}
324500df344fSAndy Whitcroft		# Remove C99 comments.
32460a920b5bSAndy Whitcroft		$line =~ s@//.*@@;
32476c72ffaaSAndy Whitcroft		$opline =~ s@//.*@@;
32480a920b5bSAndy Whitcroft
32492b474a1aSAndy Whitcroft# EXPORT_SYMBOL should immediately follow the thing it is exporting, consider
32502b474a1aSAndy Whitcroft# the whole statement.
32512b474a1aSAndy Whitcroft#print "APW <$lines[$realline_next - 1]>\n";
32522b474a1aSAndy Whitcroft		if (defined $realline_next &&
32532b474a1aSAndy Whitcroft		    exists $lines[$realline_next - 1] &&
32542b474a1aSAndy Whitcroft		    !defined $suppress_export{$realline_next} &&
32552b474a1aSAndy Whitcroft		    ($lines[$realline_next - 1] =~ /EXPORT_SYMBOL.*\((.*)\)/ ||
32562b474a1aSAndy Whitcroft		     $lines[$realline_next - 1] =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) {
32573cbf62dfSAndy Whitcroft			# Handle definitions which produce identifiers with
32583cbf62dfSAndy Whitcroft			# a prefix:
32593cbf62dfSAndy Whitcroft			#   XXX(foo);
32603cbf62dfSAndy Whitcroft			#   EXPORT_SYMBOL(something_foo);
3261653d4876SAndy Whitcroft			my $name = $1;
326287a53877SAndy Whitcroft			if ($stat =~ /^(?:.\s*}\s*\n)?.([A-Z_]+)\s*\(\s*($Ident)/ &&
32633cbf62dfSAndy Whitcroft			    $name =~ /^${Ident}_$2/) {
32643cbf62dfSAndy Whitcroft#print "FOO C name<$name>\n";
32653cbf62dfSAndy Whitcroft				$suppress_export{$realline_next} = 1;
32663cbf62dfSAndy Whitcroft
32673cbf62dfSAndy Whitcroft			} elsif ($stat !~ /(?:
32682b474a1aSAndy Whitcroft				\n.}\s*$|
326948012058SAndy Whitcroft				^.DEFINE_$Ident\(\Q$name\E\)|
327048012058SAndy Whitcroft				^.DECLARE_$Ident\(\Q$name\E\)|
327148012058SAndy Whitcroft				^.LIST_HEAD\(\Q$name\E\)|
32722b474a1aSAndy Whitcroft				^.(?:$Storage\s+)?$Type\s*\(\s*\*\s*\Q$name\E\s*\)\s*\(|
32732b474a1aSAndy Whitcroft				\b\Q$name\E(?:\s+$Attribute)*\s*(?:;|=|\[|\()
327448012058SAndy Whitcroft			    )/x) {
32752b474a1aSAndy Whitcroft#print "FOO A<$lines[$realline_next - 1]> stat<$stat> name<$name>\n";
32762b474a1aSAndy Whitcroft				$suppress_export{$realline_next} = 2;
32772b474a1aSAndy Whitcroft			} else {
32782b474a1aSAndy Whitcroft				$suppress_export{$realline_next} = 1;
32790a920b5bSAndy Whitcroft			}
32800a920b5bSAndy Whitcroft		}
32812b474a1aSAndy Whitcroft		if (!defined $suppress_export{$linenr} &&
32822b474a1aSAndy Whitcroft		    $prevline =~ /^.\s*$/ &&
32832b474a1aSAndy Whitcroft		    ($line =~ /EXPORT_SYMBOL.*\((.*)\)/ ||
32842b474a1aSAndy Whitcroft		     $line =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/)) {
32852b474a1aSAndy Whitcroft#print "FOO B <$lines[$linenr - 1]>\n";
32862b474a1aSAndy Whitcroft			$suppress_export{$linenr} = 2;
32872b474a1aSAndy Whitcroft		}
32882b474a1aSAndy Whitcroft		if (defined $suppress_export{$linenr} &&
32892b474a1aSAndy Whitcroft		    $suppress_export{$linenr} == 2) {
3290000d1cc1SJoe Perches			WARN("EXPORT_SYMBOL",
3291000d1cc1SJoe Perches			     "EXPORT_SYMBOL(foo); should immediately follow its function/variable\n" . $herecurr);
32922b474a1aSAndy Whitcroft		}
32930a920b5bSAndy Whitcroft
32945150bda4SJoe Eloff# check for global initialisers.
32955129e87cSJoe Perches		if ($line =~ /^\+$Type\s*$Ident(?:\s+$Modifier)*\s*=\s*(?:0|NULL|false)\s*;/) {
3296d5e616fcSJoe Perches			if (ERROR("GLOBAL_INITIALISERS",
3297000d1cc1SJoe Perches				  "do not initialise globals to 0 or NULL\n" .
3298d5e616fcSJoe Perches				      $herecurr) &&
3299d5e616fcSJoe Perches			    $fix) {
33005129e87cSJoe Perches				$fixed[$fixlinenr] =~ s/(^.$Type\s*$Ident(?:\s+$Modifier)*)\s*=\s*(0|NULL|false)\s*;/$1;/;
3301d5e616fcSJoe Perches			}
3302f0a594c1SAndy Whitcroft		}
33030a920b5bSAndy Whitcroft# check for static initialisers.
3304d5e616fcSJoe Perches		if ($line =~ /^\+.*\bstatic\s.*=\s*(0|NULL|false)\s*;/) {
3305d5e616fcSJoe Perches			if (ERROR("INITIALISED_STATIC",
3306000d1cc1SJoe Perches				  "do not initialise statics to 0 or NULL\n" .
3307d5e616fcSJoe Perches				      $herecurr) &&
3308d5e616fcSJoe Perches			    $fix) {
3309194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/(\bstatic\s.*?)\s*=\s*(0|NULL|false)\s*;/$1;/;
3310d5e616fcSJoe Perches			}
33110a920b5bSAndy Whitcroft		}
33120a920b5bSAndy Whitcroft
33131813087dSJoe Perches# check for misordered declarations of char/short/int/long with signed/unsigned
33141813087dSJoe Perches		while ($sline =~ m{(\b$TypeMisordered\b)}g) {
33151813087dSJoe Perches			my $tmp = trim($1);
33161813087dSJoe Perches			WARN("MISORDERED_TYPE",
33171813087dSJoe Perches			     "type '$tmp' should be specified in [[un]signed] [short|int|long|long long] order\n" . $herecurr);
33181813087dSJoe Perches		}
33191813087dSJoe Perches
3320cb710ecaSJoe Perches# check for static const char * arrays.
3321cb710ecaSJoe Perches		if ($line =~ /\bstatic\s+const\s+char\s*\*\s*(\w+)\s*\[\s*\]\s*=\s*/) {
3322000d1cc1SJoe Perches			WARN("STATIC_CONST_CHAR_ARRAY",
3323000d1cc1SJoe Perches			     "static const char * array should probably be static const char * const\n" .
3324cb710ecaSJoe Perches				$herecurr);
3325cb710ecaSJoe Perches               }
3326cb710ecaSJoe Perches
3327cb710ecaSJoe Perches# check for static char foo[] = "bar" declarations.
3328cb710ecaSJoe Perches		if ($line =~ /\bstatic\s+char\s+(\w+)\s*\[\s*\]\s*=\s*"/) {
3329000d1cc1SJoe Perches			WARN("STATIC_CONST_CHAR_ARRAY",
3330000d1cc1SJoe Perches			     "static char array declaration should probably be static const char\n" .
3331cb710ecaSJoe Perches				$herecurr);
3332cb710ecaSJoe Perches               }
3333cb710ecaSJoe Perches
3334ab7e23f3SJoe Perches# check for const <foo> const where <foo> is not a pointer or array type
3335ab7e23f3SJoe Perches		if ($sline =~ /\bconst\s+($BasicType)\s+const\b/) {
3336ab7e23f3SJoe Perches			my $found = $1;
3337ab7e23f3SJoe Perches			if ($sline =~ /\bconst\s+\Q$found\E\s+const\b\s*\*/) {
3338ab7e23f3SJoe Perches				WARN("CONST_CONST",
3339ab7e23f3SJoe Perches				     "'const $found const *' should probably be 'const $found * const'\n" . $herecurr);
3340ab7e23f3SJoe Perches			} elsif ($sline !~ /\bconst\s+\Q$found\E\s+const\s+\w+\s*\[/) {
3341ab7e23f3SJoe Perches				WARN("CONST_CONST",
3342ab7e23f3SJoe Perches				     "'const $found const' should probably be 'const $found'\n" . $herecurr);
3343ab7e23f3SJoe Perches			}
3344ab7e23f3SJoe Perches		}
3345ab7e23f3SJoe Perches
33469b0fa60dSJoe Perches# check for non-global char *foo[] = {"bar", ...} declarations.
33479b0fa60dSJoe Perches		if ($line =~ /^.\s+(?:static\s+|const\s+)?char\s+\*\s*\w+\s*\[\s*\]\s*=\s*\{/) {
33489b0fa60dSJoe Perches			WARN("STATIC_CONST_CHAR_ARRAY",
33499b0fa60dSJoe Perches			     "char * array declaration might be better as static const\n" .
33509b0fa60dSJoe Perches				$herecurr);
33519b0fa60dSJoe Perches               }
33529b0fa60dSJoe Perches
3353b598b670SJoe Perches# check for sizeof(foo)/sizeof(foo[0]) that could be ARRAY_SIZE(foo)
3354b598b670SJoe Perches		if ($line =~ m@\bsizeof\s*\(\s*($Lval)\s*\)@) {
3355b598b670SJoe Perches			my $array = $1;
3356b598b670SJoe Perches			if ($line =~ m@\b(sizeof\s*\(\s*\Q$array\E\s*\)\s*/\s*sizeof\s*\(\s*\Q$array\E\s*\[\s*0\s*\]\s*\))@) {
3357b598b670SJoe Perches				my $array_div = $1;
3358b598b670SJoe Perches				if (WARN("ARRAY_SIZE",
3359b598b670SJoe Perches					 "Prefer ARRAY_SIZE($array)\n" . $herecurr) &&
3360b598b670SJoe Perches				    $fix) {
3361b598b670SJoe Perches					$fixed[$fixlinenr] =~ s/\Q$array_div\E/ARRAY_SIZE($array)/;
3362b598b670SJoe Perches				}
3363b598b670SJoe Perches			}
3364b598b670SJoe Perches		}
3365b598b670SJoe Perches
3366b36190c5SJoe Perches# check for function declarations without arguments like "int foo()"
3367b36190c5SJoe Perches		if ($line =~ /(\b$Type\s+$Ident)\s*\(\s*\)/) {
3368b36190c5SJoe Perches			if (ERROR("FUNCTION_WITHOUT_ARGS",
3369b36190c5SJoe Perches				  "Bad function definition - $1() should probably be $1(void)\n" . $herecurr) &&
3370b36190c5SJoe Perches			    $fix) {
3371194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/(\b($Type)\s+($Ident))\s*\(\s*\)/$2 $3(void)/;
3372b36190c5SJoe Perches			}
3373b36190c5SJoe Perches		}
3374b36190c5SJoe Perches
337592e112fdSJoe Perches# check for uses of DEFINE_PCI_DEVICE_TABLE
337692e112fdSJoe Perches		if ($line =~ /\bDEFINE_PCI_DEVICE_TABLE\s*\(\s*(\w+)\s*\)\s*=/) {
337792e112fdSJoe Perches			if (WARN("DEFINE_PCI_DEVICE_TABLE",
337892e112fdSJoe Perches				 "Prefer struct pci_device_id over deprecated DEFINE_PCI_DEVICE_TABLE\n" . $herecurr) &&
337992e112fdSJoe Perches			    $fix) {
3380194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\b(?:static\s+|)DEFINE_PCI_DEVICE_TABLE\s*\(\s*(\w+)\s*\)\s*=\s*/static const struct pci_device_id $1\[\] = /;
338192e112fdSJoe Perches			}
338293ed0e2dSJoe Perches		}
338393ed0e2dSJoe Perches
3384653d4876SAndy Whitcroft# check for new typedefs, only function parameters and sparse annotations
3385653d4876SAndy Whitcroft# make sense.
3386653d4876SAndy Whitcroft		if ($line =~ /\btypedef\s/ &&
33878054576dSAndy Whitcroft		    $line !~ /\btypedef\s+$Type\s*\(\s*\*?$Ident\s*\)\s*\(/ &&
3388c45dcabdSAndy Whitcroft		    $line !~ /\btypedef\s+$Type\s+$Ident\s*\(/ &&
33898ed22cadSAndy Whitcroft		    $line !~ /\b$typeTypedefs\b/ &&
3390653d4876SAndy Whitcroft		    $line !~ /\b__bitwise(?:__|)\b/) {
3391000d1cc1SJoe Perches			WARN("NEW_TYPEDEFS",
3392000d1cc1SJoe Perches			     "do not add new typedefs\n" . $herecurr);
33930a920b5bSAndy Whitcroft		}
33940a920b5bSAndy Whitcroft
33950a920b5bSAndy Whitcroft# * goes on variable not on type
339665863862SAndy Whitcroft		# (char*[ const])
3397bfcb2cc7SAndy Whitcroft		while ($line =~ m{(\($NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)\))}g) {
3398bfcb2cc7SAndy Whitcroft			#print "AA<$1>\n";
33993705ce5bSJoe Perches			my ($ident, $from, $to) = ($1, $2, $2);
3400d8aaf121SAndy Whitcroft
340165863862SAndy Whitcroft			# Should start with a space.
340265863862SAndy Whitcroft			$to =~ s/^(\S)/ $1/;
340365863862SAndy Whitcroft			# Should not end with a space.
340465863862SAndy Whitcroft			$to =~ s/\s+$//;
340565863862SAndy Whitcroft			# '*'s should not have spaces between.
3406f9a0b3d1SAndy Whitcroft			while ($to =~ s/\*\s+\*/\*\*/) {
340765863862SAndy Whitcroft			}
3408d8aaf121SAndy Whitcroft
34093705ce5bSJoe Perches##			print "1: from<$from> to<$to> ident<$ident>\n";
341065863862SAndy Whitcroft			if ($from ne $to) {
34113705ce5bSJoe Perches				if (ERROR("POINTER_LOCATION",
34123705ce5bSJoe Perches					  "\"(foo$from)\" should be \"(foo$to)\"\n" .  $herecurr) &&
34133705ce5bSJoe Perches				    $fix) {
34143705ce5bSJoe Perches					my $sub_from = $ident;
34153705ce5bSJoe Perches					my $sub_to = $ident;
34163705ce5bSJoe Perches					$sub_to =~ s/\Q$from\E/$to/;
3417194f66fcSJoe Perches					$fixed[$fixlinenr] =~
34183705ce5bSJoe Perches					    s@\Q$sub_from\E@$sub_to@;
34193705ce5bSJoe Perches				}
342065863862SAndy Whitcroft			}
3421bfcb2cc7SAndy Whitcroft		}
3422bfcb2cc7SAndy Whitcroft		while ($line =~ m{(\b$NonptrType(\s*(?:$Modifier\b\s*|\*\s*)+)($Ident))}g) {
3423bfcb2cc7SAndy Whitcroft			#print "BB<$1>\n";
34243705ce5bSJoe Perches			my ($match, $from, $to, $ident) = ($1, $2, $2, $3);
3425d8aaf121SAndy Whitcroft
342665863862SAndy Whitcroft			# Should start with a space.
342765863862SAndy Whitcroft			$to =~ s/^(\S)/ $1/;
342865863862SAndy Whitcroft			# Should not end with a space.
342965863862SAndy Whitcroft			$to =~ s/\s+$//;
343065863862SAndy Whitcroft			# '*'s should not have spaces between.
3431f9a0b3d1SAndy Whitcroft			while ($to =~ s/\*\s+\*/\*\*/) {
343265863862SAndy Whitcroft			}
343365863862SAndy Whitcroft			# Modifiers should have spaces.
343465863862SAndy Whitcroft			$to =~ s/(\b$Modifier$)/$1 /;
343565863862SAndy Whitcroft
34363705ce5bSJoe Perches##			print "2: from<$from> to<$to> ident<$ident>\n";
3437667026e7SAndy Whitcroft			if ($from ne $to && $ident !~ /^$Modifier$/) {
34383705ce5bSJoe Perches				if (ERROR("POINTER_LOCATION",
34393705ce5bSJoe Perches					  "\"foo${from}bar\" should be \"foo${to}bar\"\n" .  $herecurr) &&
34403705ce5bSJoe Perches				    $fix) {
34413705ce5bSJoe Perches
34423705ce5bSJoe Perches					my $sub_from = $match;
34433705ce5bSJoe Perches					my $sub_to = $match;
34443705ce5bSJoe Perches					$sub_to =~ s/\Q$from\E/$to/;
3445194f66fcSJoe Perches					$fixed[$fixlinenr] =~
34463705ce5bSJoe Perches					    s@\Q$sub_from\E@$sub_to@;
34473705ce5bSJoe Perches				}
344865863862SAndy Whitcroft			}
34490a920b5bSAndy Whitcroft		}
34500a920b5bSAndy Whitcroft
34510a920b5bSAndy Whitcroft# # no BUG() or BUG_ON()
34520a920b5bSAndy Whitcroft# 		if ($line =~ /\b(BUG|BUG_ON)\b/) {
34530a920b5bSAndy Whitcroft# 			print "Try to use WARN_ON & Recovery code rather than BUG() or BUG_ON()\n";
34540a920b5bSAndy Whitcroft# 			print "$herecurr";
34550a920b5bSAndy Whitcroft# 			$clean = 0;
34560a920b5bSAndy Whitcroft# 		}
34570a920b5bSAndy Whitcroft
34588905a67cSAndy Whitcroft		if ($line =~ /\bLINUX_VERSION_CODE\b/) {
3459000d1cc1SJoe Perches			WARN("LINUX_VERSION_CODE",
3460000d1cc1SJoe Perches			     "LINUX_VERSION_CODE should be avoided, code should be for the version to which it is merged\n" . $herecurr);
34618905a67cSAndy Whitcroft		}
34628905a67cSAndy Whitcroft
346317441227SJoe Perches# check for uses of printk_ratelimit
346417441227SJoe Perches		if ($line =~ /\bprintk_ratelimit\s*\(/) {
3465000d1cc1SJoe Perches			WARN("PRINTK_RATELIMITED",
3466000d1cc1SJoe Perches			     "Prefer printk_ratelimited or pr_<level>_ratelimited to printk_ratelimit\n" . $herecurr);
346717441227SJoe Perches		}
346817441227SJoe Perches
346900df344fSAndy Whitcroft# printk should use KERN_* levels.  Note that follow on printk's on the
347000df344fSAndy Whitcroft# same line do not need a level, so we use the current block context
347100df344fSAndy Whitcroft# to try and find and validate the current printk.  In summary the current
347225985edcSLucas De Marchi# printk includes all preceding printk's which have no newline on the end.
347300df344fSAndy Whitcroft# we assume the first bad printk is the one to report.
3474f0a594c1SAndy Whitcroft		if ($line =~ /\bprintk\((?!KERN_)\s*"/) {
347500df344fSAndy Whitcroft			my $ok = 0;
347600df344fSAndy Whitcroft			for (my $ln = $linenr - 1; $ln >= $first_line; $ln--) {
347700df344fSAndy Whitcroft				#print "CHECK<$lines[$ln - 1]\n";
347825985edcSLucas De Marchi				# we have a preceding printk if it ends
347900df344fSAndy Whitcroft				# with "\n" ignore it, else it is to blame
348000df344fSAndy Whitcroft				if ($lines[$ln - 1] =~ m{\bprintk\(}) {
348100df344fSAndy Whitcroft					if ($rawlines[$ln - 1] !~ m{\\n"}) {
348200df344fSAndy Whitcroft						$ok = 1;
348300df344fSAndy Whitcroft					}
348400df344fSAndy Whitcroft					last;
348500df344fSAndy Whitcroft				}
348600df344fSAndy Whitcroft			}
348700df344fSAndy Whitcroft			if ($ok == 0) {
3488000d1cc1SJoe Perches				WARN("PRINTK_WITHOUT_KERN_LEVEL",
3489000d1cc1SJoe Perches				     "printk() should include KERN_ facility level\n" . $herecurr);
34900a920b5bSAndy Whitcroft			}
349100df344fSAndy Whitcroft		}
34920a920b5bSAndy Whitcroft
3493243f3803SJoe Perches		if ($line =~ /\bprintk\s*\(\s*KERN_([A-Z]+)/) {
3494243f3803SJoe Perches			my $orig = $1;
3495243f3803SJoe Perches			my $level = lc($orig);
3496243f3803SJoe Perches			$level = "warn" if ($level eq "warning");
34978f26b837SJoe Perches			my $level2 = $level;
34988f26b837SJoe Perches			$level2 = "dbg" if ($level eq "debug");
3499243f3803SJoe Perches			WARN("PREFER_PR_LEVEL",
3500daa8b059SYogesh Chaudhari			     "Prefer [subsystem eg: netdev]_$level2([subsystem]dev, ... then dev_$level2(dev, ... then pr_$level(...  to printk(KERN_$orig ...\n" . $herecurr);
3501243f3803SJoe Perches		}
3502243f3803SJoe Perches
3503243f3803SJoe Perches		if ($line =~ /\bpr_warning\s*\(/) {
3504d5e616fcSJoe Perches			if (WARN("PREFER_PR_LEVEL",
3505d5e616fcSJoe Perches				 "Prefer pr_warn(... to pr_warning(...\n" . $herecurr) &&
3506d5e616fcSJoe Perches			    $fix) {
3507194f66fcSJoe Perches				$fixed[$fixlinenr] =~
3508d5e616fcSJoe Perches				    s/\bpr_warning\b/pr_warn/;
3509d5e616fcSJoe Perches			}
3510243f3803SJoe Perches		}
3511243f3803SJoe Perches
3512dc139313SJoe Perches		if ($line =~ /\bdev_printk\s*\(\s*KERN_([A-Z]+)/) {
3513dc139313SJoe Perches			my $orig = $1;
3514dc139313SJoe Perches			my $level = lc($orig);
3515dc139313SJoe Perches			$level = "warn" if ($level eq "warning");
3516dc139313SJoe Perches			$level = "dbg" if ($level eq "debug");
3517dc139313SJoe Perches			WARN("PREFER_DEV_LEVEL",
3518dc139313SJoe Perches			     "Prefer dev_$level(... to dev_printk(KERN_$orig, ...\n" . $herecurr);
3519dc139313SJoe Perches		}
3520dc139313SJoe Perches
352191c9afafSAndy Lutomirski# ENOSYS means "bad syscall nr" and nothing else.  This will have a small
352291c9afafSAndy Lutomirski# number of false positives, but assembly files are not checked, so at
352391c9afafSAndy Lutomirski# least the arch entry code will not trigger this warning.
352491c9afafSAndy Lutomirski		if ($line =~ /\bENOSYS\b/) {
352591c9afafSAndy Lutomirski			WARN("ENOSYS",
352691c9afafSAndy Lutomirski			     "ENOSYS means 'invalid syscall nr' and nothing else\n" . $herecurr);
352791c9afafSAndy Lutomirski		}
352891c9afafSAndy Lutomirski
3529653d4876SAndy Whitcroft# function brace can't be on same line, except for #defines of do while,
3530653d4876SAndy Whitcroft# or if closed on same line
35318d182478SJoe Perches		if (($line=~/$Type\s*$Ident\(.*\).*\s*{/) and
3532c45dcabdSAndy Whitcroft		    !($line=~/\#\s*define.*do\s{/) and !($line=~/}/)) {
35338d182478SJoe Perches			if (ERROR("OPEN_BRACE",
35348d182478SJoe Perches				  "open brace '{' following function declarations go on the next line\n" . $herecurr) &&
35358d182478SJoe Perches			    $fix) {
35368d182478SJoe Perches				fix_delete_line($fixlinenr, $rawline);
35378d182478SJoe Perches				my $fixed_line = $rawline;
35388d182478SJoe Perches				$fixed_line =~ /(^..*$Type\s*$Ident\(.*\)\s*){(.*)$/;
35398d182478SJoe Perches				my $line1 = $1;
35408d182478SJoe Perches				my $line2 = $2;
35418d182478SJoe Perches				fix_insert_line($fixlinenr, ltrim($line1));
35428d182478SJoe Perches				fix_insert_line($fixlinenr, "\+{");
35438d182478SJoe Perches				if ($line2 !~ /^\s*$/) {
35448d182478SJoe Perches					fix_insert_line($fixlinenr, "\+\t" . trim($line2));
35458d182478SJoe Perches				}
35468d182478SJoe Perches			}
35470a920b5bSAndy Whitcroft		}
3548653d4876SAndy Whitcroft
35498905a67cSAndy Whitcroft# open braces for enum, union and struct go on the same line.
35508905a67cSAndy Whitcroft		if ($line =~ /^.\s*{/ &&
35518905a67cSAndy Whitcroft		    $prevline =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident)?\s*$/) {
35528d182478SJoe Perches			if (ERROR("OPEN_BRACE",
35538d182478SJoe Perches				  "open brace '{' following $1 go on the same line\n" . $hereprev) &&
35548d182478SJoe Perches			    $fix && $prevline =~ /^\+/ && $line =~ /^\+/) {
35558d182478SJoe Perches				fix_delete_line($fixlinenr - 1, $prevrawline);
35568d182478SJoe Perches				fix_delete_line($fixlinenr, $rawline);
35578d182478SJoe Perches				my $fixedline = rtrim($prevrawline) . " {";
35588d182478SJoe Perches				fix_insert_line($fixlinenr, $fixedline);
35598d182478SJoe Perches				$fixedline = $rawline;
35608d182478SJoe Perches				$fixedline =~ s/^(.\s*){\s*/$1\t/;
35618d182478SJoe Perches				if ($fixedline !~ /^\+\s*$/) {
35628d182478SJoe Perches					fix_insert_line($fixlinenr, $fixedline);
35638d182478SJoe Perches				}
35648d182478SJoe Perches			}
35658905a67cSAndy Whitcroft		}
35668905a67cSAndy Whitcroft
35670c73b4ebSAndy Whitcroft# missing space after union, struct or enum definition
35683705ce5bSJoe Perches		if ($line =~ /^.\s*(?:typedef\s+)?(enum|union|struct)(?:\s+$Ident){1,2}[=\{]/) {
35693705ce5bSJoe Perches			if (WARN("SPACING",
35703705ce5bSJoe Perches				 "missing space after $1 definition\n" . $herecurr) &&
35713705ce5bSJoe Perches			    $fix) {
3572194f66fcSJoe Perches				$fixed[$fixlinenr] =~
35733705ce5bSJoe Perches				    s/^(.\s*(?:typedef\s+)?(?:enum|union|struct)(?:\s+$Ident){1,2})([=\{])/$1 $2/;
35743705ce5bSJoe Perches			}
35750c73b4ebSAndy Whitcroft		}
35760c73b4ebSAndy Whitcroft
357731070b5dSJoe Perches# Function pointer declarations
357831070b5dSJoe Perches# check spacing between type, funcptr, and args
357931070b5dSJoe Perches# canonical declaration is "type (*funcptr)(args...)"
358091f72e9cSJoe Perches		if ($line =~ /^.\s*($Declare)\((\s*)\*(\s*)($Ident)(\s*)\)(\s*)\(/) {
358131070b5dSJoe Perches			my $declare = $1;
358231070b5dSJoe Perches			my $pre_pointer_space = $2;
358331070b5dSJoe Perches			my $post_pointer_space = $3;
358431070b5dSJoe Perches			my $funcname = $4;
358531070b5dSJoe Perches			my $post_funcname_space = $5;
358631070b5dSJoe Perches			my $pre_args_space = $6;
358731070b5dSJoe Perches
358891f72e9cSJoe Perches# the $Declare variable will capture all spaces after the type
358991f72e9cSJoe Perches# so check it for a missing trailing missing space but pointer return types
359091f72e9cSJoe Perches# don't need a space so don't warn for those.
359191f72e9cSJoe Perches			my $post_declare_space = "";
359291f72e9cSJoe Perches			if ($declare =~ /(\s+)$/) {
359391f72e9cSJoe Perches				$post_declare_space = $1;
359491f72e9cSJoe Perches				$declare = rtrim($declare);
359591f72e9cSJoe Perches			}
359691f72e9cSJoe Perches			if ($declare !~ /\*$/ && $post_declare_space =~ /^$/) {
359731070b5dSJoe Perches				WARN("SPACING",
359831070b5dSJoe Perches				     "missing space after return type\n" . $herecurr);
359991f72e9cSJoe Perches				$post_declare_space = " ";
360031070b5dSJoe Perches			}
360131070b5dSJoe Perches
360231070b5dSJoe Perches# unnecessary space "type  (*funcptr)(args...)"
360391f72e9cSJoe Perches# This test is not currently implemented because these declarations are
360491f72e9cSJoe Perches# equivalent to
360591f72e9cSJoe Perches#	int  foo(int bar, ...)
360691f72e9cSJoe Perches# and this is form shouldn't/doesn't generate a checkpatch warning.
360791f72e9cSJoe Perches#
360891f72e9cSJoe Perches#			elsif ($declare =~ /\s{2,}$/) {
360991f72e9cSJoe Perches#				WARN("SPACING",
361091f72e9cSJoe Perches#				     "Multiple spaces after return type\n" . $herecurr);
361191f72e9cSJoe Perches#			}
361231070b5dSJoe Perches
361331070b5dSJoe Perches# unnecessary space "type ( *funcptr)(args...)"
361431070b5dSJoe Perches			if (defined $pre_pointer_space &&
361531070b5dSJoe Perches			    $pre_pointer_space =~ /^\s/) {
361631070b5dSJoe Perches				WARN("SPACING",
361731070b5dSJoe Perches				     "Unnecessary space after function pointer open parenthesis\n" . $herecurr);
361831070b5dSJoe Perches			}
361931070b5dSJoe Perches
362031070b5dSJoe Perches# unnecessary space "type (* funcptr)(args...)"
362131070b5dSJoe Perches			if (defined $post_pointer_space &&
362231070b5dSJoe Perches			    $post_pointer_space =~ /^\s/) {
362331070b5dSJoe Perches				WARN("SPACING",
362431070b5dSJoe Perches				     "Unnecessary space before function pointer name\n" . $herecurr);
362531070b5dSJoe Perches			}
362631070b5dSJoe Perches
362731070b5dSJoe Perches# unnecessary space "type (*funcptr )(args...)"
362831070b5dSJoe Perches			if (defined $post_funcname_space &&
362931070b5dSJoe Perches			    $post_funcname_space =~ /^\s/) {
363031070b5dSJoe Perches				WARN("SPACING",
363131070b5dSJoe Perches				     "Unnecessary space after function pointer name\n" . $herecurr);
363231070b5dSJoe Perches			}
363331070b5dSJoe Perches
363431070b5dSJoe Perches# unnecessary space "type (*funcptr) (args...)"
363531070b5dSJoe Perches			if (defined $pre_args_space &&
363631070b5dSJoe Perches			    $pre_args_space =~ /^\s/) {
363731070b5dSJoe Perches				WARN("SPACING",
363831070b5dSJoe Perches				     "Unnecessary space before function pointer arguments\n" . $herecurr);
363931070b5dSJoe Perches			}
364031070b5dSJoe Perches
364131070b5dSJoe Perches			if (show_type("SPACING") && $fix) {
3642194f66fcSJoe Perches				$fixed[$fixlinenr] =~
364391f72e9cSJoe Perches				    s/^(.\s*)$Declare\s*\(\s*\*\s*$Ident\s*\)\s*\(/$1 . $declare . $post_declare_space . '(*' . $funcname . ')('/ex;
364431070b5dSJoe Perches			}
364531070b5dSJoe Perches		}
364631070b5dSJoe Perches
36478d31cfceSAndy Whitcroft# check for spacing round square brackets; allowed:
36488d31cfceSAndy Whitcroft#  1. with a type on the left -- int [] a;
3649fe2a7dbcSAndy Whitcroft#  2. at the beginning of a line for slice initialisers -- [0...10] = 5,
3650fe2a7dbcSAndy Whitcroft#  3. inside a curly brace -- = { [0...10] = 5 }
36518d31cfceSAndy Whitcroft		while ($line =~ /(.*?\s)\[/g) {
36528d31cfceSAndy Whitcroft			my ($where, $prefix) = ($-[1], $1);
36538d31cfceSAndy Whitcroft			if ($prefix !~ /$Type\s+$/ &&
3654fe2a7dbcSAndy Whitcroft			    ($where != 0 || $prefix !~ /^.\s+$/) &&
3655daebc534SAndy Whitcroft			    $prefix !~ /[{,]\s+$/) {
36563705ce5bSJoe Perches				if (ERROR("BRACKET_SPACE",
36573705ce5bSJoe Perches					  "space prohibited before open square bracket '['\n" . $herecurr) &&
36583705ce5bSJoe Perches				    $fix) {
3659194f66fcSJoe Perches				    $fixed[$fixlinenr] =~
36603705ce5bSJoe Perches					s/^(\+.*?)\s+\[/$1\[/;
36613705ce5bSJoe Perches				}
36628d31cfceSAndy Whitcroft			}
36638d31cfceSAndy Whitcroft		}
36648d31cfceSAndy Whitcroft
3665f0a594c1SAndy Whitcroft# check for spaces between functions and their parentheses.
36666c72ffaaSAndy Whitcroft		while ($line =~ /($Ident)\s+\(/g) {
3667c2fdda0dSAndy Whitcroft			my $name = $1;
3668773647a0SAndy Whitcroft			my $ctx_before = substr($line, 0, $-[1]);
3669773647a0SAndy Whitcroft			my $ctx = "$ctx_before$name";
3670c2fdda0dSAndy Whitcroft
3671c2fdda0dSAndy Whitcroft			# Ignore those directives where spaces _are_ permitted.
3672773647a0SAndy Whitcroft			if ($name =~ /^(?:
3673773647a0SAndy Whitcroft				if|for|while|switch|return|case|
3674773647a0SAndy Whitcroft				volatile|__volatile__|
3675773647a0SAndy Whitcroft				__attribute__|format|__extension__|
3676773647a0SAndy Whitcroft				asm|__asm__)$/x)
3677773647a0SAndy Whitcroft			{
3678c2fdda0dSAndy Whitcroft			# cpp #define statements have non-optional spaces, ie
3679c2fdda0dSAndy Whitcroft			# if there is a space between the name and the open
3680c2fdda0dSAndy Whitcroft			# parenthesis it is simply not a parameter group.
3681c45dcabdSAndy Whitcroft			} elsif ($ctx_before =~ /^.\s*\#\s*define\s*$/) {
3682773647a0SAndy Whitcroft
3683773647a0SAndy Whitcroft			# cpp #elif statement condition may start with a (
3684c45dcabdSAndy Whitcroft			} elsif ($ctx =~ /^.\s*\#\s*elif\s*$/) {
3685c2fdda0dSAndy Whitcroft
3686c2fdda0dSAndy Whitcroft			# If this whole things ends with a type its most
3687c2fdda0dSAndy Whitcroft			# likely a typedef for a function.
3688773647a0SAndy Whitcroft			} elsif ($ctx =~ /$Type$/) {
3689c2fdda0dSAndy Whitcroft
3690c2fdda0dSAndy Whitcroft			} else {
36913705ce5bSJoe Perches				if (WARN("SPACING",
36923705ce5bSJoe Perches					 "space prohibited between function name and open parenthesis '('\n" . $herecurr) &&
36933705ce5bSJoe Perches					     $fix) {
3694194f66fcSJoe Perches					$fixed[$fixlinenr] =~
36953705ce5bSJoe Perches					    s/\b$name\s+\(/$name\(/;
36963705ce5bSJoe Perches				}
3697f0a594c1SAndy Whitcroft			}
36986c72ffaaSAndy Whitcroft		}
36999a4cad4eSEric Nelson
3700653d4876SAndy Whitcroft# Check operator spacing.
37010a920b5bSAndy Whitcroft		if (!($line=~/\#\s*include/)) {
37023705ce5bSJoe Perches			my $fixed_line = "";
37033705ce5bSJoe Perches			my $line_fixed = 0;
37043705ce5bSJoe Perches
37059c0ca6f9SAndy Whitcroft			my $ops = qr{
37069c0ca6f9SAndy Whitcroft				<<=|>>=|<=|>=|==|!=|
37079c0ca6f9SAndy Whitcroft				\+=|-=|\*=|\/=|%=|\^=|\|=|&=|
37089c0ca6f9SAndy Whitcroft				=>|->|<<|>>|<|>|=|!|~|
37091f65f947SAndy Whitcroft				&&|\|\||,|\^|\+\+|--|&|\||\+|-|\*|\/|%|
371084731623SJoe Perches				\?:|\?|:
37119c0ca6f9SAndy Whitcroft			}x;
3712cf655043SAndy Whitcroft			my @elements = split(/($ops|;)/, $opline);
37133705ce5bSJoe Perches
37143705ce5bSJoe Perches##			print("element count: <" . $#elements . ">\n");
37153705ce5bSJoe Perches##			foreach my $el (@elements) {
37163705ce5bSJoe Perches##				print("el: <$el>\n");
37173705ce5bSJoe Perches##			}
37183705ce5bSJoe Perches
37193705ce5bSJoe Perches			my @fix_elements = ();
372000df344fSAndy Whitcroft			my $off = 0;
37216c72ffaaSAndy Whitcroft
37223705ce5bSJoe Perches			foreach my $el (@elements) {
37233705ce5bSJoe Perches				push(@fix_elements, substr($rawline, $off, length($el)));
37243705ce5bSJoe Perches				$off += length($el);
37253705ce5bSJoe Perches			}
37263705ce5bSJoe Perches
37273705ce5bSJoe Perches			$off = 0;
37283705ce5bSJoe Perches
37296c72ffaaSAndy Whitcroft			my $blank = copy_spacing($opline);
3730b34c648bSJoe Perches			my $last_after = -1;
37316c72ffaaSAndy Whitcroft
37320a920b5bSAndy Whitcroft			for (my $n = 0; $n < $#elements; $n += 2) {
37333705ce5bSJoe Perches
37343705ce5bSJoe Perches				my $good = $fix_elements[$n] . $fix_elements[$n + 1];
37353705ce5bSJoe Perches
37363705ce5bSJoe Perches##				print("n: <$n> good: <$good>\n");
37373705ce5bSJoe Perches
37384a0df2efSAndy Whitcroft				$off += length($elements[$n]);
37394a0df2efSAndy Whitcroft
374025985edcSLucas De Marchi				# Pick up the preceding and succeeding characters.
3741773647a0SAndy Whitcroft				my $ca = substr($opline, 0, $off);
3742773647a0SAndy Whitcroft				my $cc = '';
3743773647a0SAndy Whitcroft				if (length($opline) >= ($off + length($elements[$n + 1]))) {
3744773647a0SAndy Whitcroft					$cc = substr($opline, $off + length($elements[$n + 1]));
3745773647a0SAndy Whitcroft				}
3746773647a0SAndy Whitcroft				my $cb = "$ca$;$cc";
3747773647a0SAndy Whitcroft
37484a0df2efSAndy Whitcroft				my $a = '';
37494a0df2efSAndy Whitcroft				$a = 'V' if ($elements[$n] ne '');
37504a0df2efSAndy Whitcroft				$a = 'W' if ($elements[$n] =~ /\s$/);
3751cf655043SAndy Whitcroft				$a = 'C' if ($elements[$n] =~ /$;$/);
37524a0df2efSAndy Whitcroft				$a = 'B' if ($elements[$n] =~ /(\[|\()$/);
37534a0df2efSAndy Whitcroft				$a = 'O' if ($elements[$n] eq '');
3754773647a0SAndy Whitcroft				$a = 'E' if ($ca =~ /^\s*$/);
37554a0df2efSAndy Whitcroft
37560a920b5bSAndy Whitcroft				my $op = $elements[$n + 1];
37574a0df2efSAndy Whitcroft
37584a0df2efSAndy Whitcroft				my $c = '';
37590a920b5bSAndy Whitcroft				if (defined $elements[$n + 2]) {
37604a0df2efSAndy Whitcroft					$c = 'V' if ($elements[$n + 2] ne '');
37614a0df2efSAndy Whitcroft					$c = 'W' if ($elements[$n + 2] =~ /^\s/);
3762cf655043SAndy Whitcroft					$c = 'C' if ($elements[$n + 2] =~ /^$;/);
37634a0df2efSAndy Whitcroft					$c = 'B' if ($elements[$n + 2] =~ /^(\)|\]|;)/);
37644a0df2efSAndy Whitcroft					$c = 'O' if ($elements[$n + 2] eq '');
37658b1b3378SAndy Whitcroft					$c = 'E' if ($elements[$n + 2] =~ /^\s*\\$/);
37664a0df2efSAndy Whitcroft				} else {
37674a0df2efSAndy Whitcroft					$c = 'E';
37680a920b5bSAndy Whitcroft				}
37690a920b5bSAndy Whitcroft
37704a0df2efSAndy Whitcroft				my $ctx = "${a}x${c}";
37714a0df2efSAndy Whitcroft
37724a0df2efSAndy Whitcroft				my $at = "(ctx:$ctx)";
37734a0df2efSAndy Whitcroft
37746c72ffaaSAndy Whitcroft				my $ptr = substr($blank, 0, $off) . "^";
3775de7d4f0eSAndy Whitcroft				my $hereptr = "$hereline$ptr\n";
37760a920b5bSAndy Whitcroft
377774048ed8SAndy Whitcroft				# Pull out the value of this operator.
37786c72ffaaSAndy Whitcroft				my $op_type = substr($curr_values, $off + 1, 1);
37790a920b5bSAndy Whitcroft
37801f65f947SAndy Whitcroft				# Get the full operator variant.
37811f65f947SAndy Whitcroft				my $opv = $op . substr($curr_vars, $off, 1);
37821f65f947SAndy Whitcroft
378313214adfSAndy Whitcroft				# Ignore operators passed as parameters.
378413214adfSAndy Whitcroft				if ($op_type ne 'V' &&
3785d7fe8065SSam Bobroff				    $ca =~ /\s$/ && $cc =~ /^\s*[,\)]/) {
378613214adfSAndy Whitcroft
3787cf655043SAndy Whitcroft#				# Ignore comments
3788cf655043SAndy Whitcroft#				} elsif ($op =~ /^$;+$/) {
378913214adfSAndy Whitcroft
3790d8aaf121SAndy Whitcroft				# ; should have either the end of line or a space or \ after it
379113214adfSAndy Whitcroft				} elsif ($op eq ';') {
3792cf655043SAndy Whitcroft					if ($ctx !~ /.x[WEBC]/ &&
3793cf655043SAndy Whitcroft					    $cc !~ /^\\/ && $cc !~ /^;/) {
37943705ce5bSJoe Perches						if (ERROR("SPACING",
37953705ce5bSJoe Perches							  "space required after that '$op' $at\n" . $hereptr)) {
3796b34c648bSJoe Perches							$good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " ";
37973705ce5bSJoe Perches							$line_fixed = 1;
37983705ce5bSJoe Perches						}
3799d8aaf121SAndy Whitcroft					}
3800d8aaf121SAndy Whitcroft
3801d8aaf121SAndy Whitcroft				# // is a comment
3802d8aaf121SAndy Whitcroft				} elsif ($op eq '//') {
38030a920b5bSAndy Whitcroft
3804b00e4814SJoe Perches				#   :   when part of a bitfield
3805b00e4814SJoe Perches				} elsif ($opv eq ':B') {
3806b00e4814SJoe Perches					# skip the bitfield test for now
3807b00e4814SJoe Perches
38081f65f947SAndy Whitcroft				# No spaces for:
38091f65f947SAndy Whitcroft				#   ->
3810b00e4814SJoe Perches				} elsif ($op eq '->') {
38114a0df2efSAndy Whitcroft					if ($ctx =~ /Wx.|.xW/) {
38123705ce5bSJoe Perches						if (ERROR("SPACING",
38133705ce5bSJoe Perches							  "spaces prohibited around that '$op' $at\n" . $hereptr)) {
3814b34c648bSJoe Perches							$good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]);
38153705ce5bSJoe Perches							if (defined $fix_elements[$n + 2]) {
38163705ce5bSJoe Perches								$fix_elements[$n + 2] =~ s/^\s+//;
38173705ce5bSJoe Perches							}
3818b34c648bSJoe Perches							$line_fixed = 1;
38193705ce5bSJoe Perches						}
38200a920b5bSAndy Whitcroft					}
38210a920b5bSAndy Whitcroft
38222381097bSJoe Perches				# , must not have a space before and must have a space on the right.
38230a920b5bSAndy Whitcroft				} elsif ($op eq ',') {
38242381097bSJoe Perches					my $rtrim_before = 0;
38252381097bSJoe Perches					my $space_after = 0;
38262381097bSJoe Perches					if ($ctx =~ /Wx./) {
38272381097bSJoe Perches						if (ERROR("SPACING",
38282381097bSJoe Perches							  "space prohibited before that '$op' $at\n" . $hereptr)) {
38292381097bSJoe Perches							$line_fixed = 1;
38302381097bSJoe Perches							$rtrim_before = 1;
38312381097bSJoe Perches						}
38322381097bSJoe Perches					}
3833cf655043SAndy Whitcroft					if ($ctx !~ /.x[WEC]/ && $cc !~ /^}/) {
38343705ce5bSJoe Perches						if (ERROR("SPACING",
38353705ce5bSJoe Perches							  "space required after that '$op' $at\n" . $hereptr)) {
38363705ce5bSJoe Perches							$line_fixed = 1;
3837b34c648bSJoe Perches							$last_after = $n;
38382381097bSJoe Perches							$space_after = 1;
38392381097bSJoe Perches						}
38402381097bSJoe Perches					}
38412381097bSJoe Perches					if ($rtrim_before || $space_after) {
38422381097bSJoe Perches						if ($rtrim_before) {
38432381097bSJoe Perches							$good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]);
38442381097bSJoe Perches						} else {
38452381097bSJoe Perches							$good = $fix_elements[$n] . trim($fix_elements[$n + 1]);
38462381097bSJoe Perches						}
38472381097bSJoe Perches						if ($space_after) {
38482381097bSJoe Perches							$good .= " ";
38493705ce5bSJoe Perches						}
38500a920b5bSAndy Whitcroft					}
38510a920b5bSAndy Whitcroft
38529c0ca6f9SAndy Whitcroft				# '*' as part of a type definition -- reported already.
385374048ed8SAndy Whitcroft				} elsif ($opv eq '*_') {
38549c0ca6f9SAndy Whitcroft					#warn "'*' is part of type\n";
38559c0ca6f9SAndy Whitcroft
38569c0ca6f9SAndy Whitcroft				# unary operators should have a space before and
38579c0ca6f9SAndy Whitcroft				# none after.  May be left adjacent to another
38589c0ca6f9SAndy Whitcroft				# unary operator, or a cast
38599c0ca6f9SAndy Whitcroft				} elsif ($op eq '!' || $op eq '~' ||
386074048ed8SAndy Whitcroft					 $opv eq '*U' || $opv eq '-U' ||
38610d413866SAndy Whitcroft					 $opv eq '&U' || $opv eq '&&U') {
3862cf655043SAndy Whitcroft					if ($ctx !~ /[WEBC]x./ && $ca !~ /(?:\)|!|~|\*|-|\&|\||\+\+|\-\-|\{)$/) {
38633705ce5bSJoe Perches						if (ERROR("SPACING",
38643705ce5bSJoe Perches							  "space required before that '$op' $at\n" . $hereptr)) {
3865b34c648bSJoe Perches							if ($n != $last_after + 2) {
3866b34c648bSJoe Perches								$good = $fix_elements[$n] . " " . ltrim($fix_elements[$n + 1]);
38673705ce5bSJoe Perches								$line_fixed = 1;
38683705ce5bSJoe Perches							}
38690a920b5bSAndy Whitcroft						}
3870b34c648bSJoe Perches					}
3871a3340b35SAndy Whitcroft					if ($op eq '*' && $cc =~/\s*$Modifier\b/) {
3872171ae1a4SAndy Whitcroft						# A unary '*' may be const
3873171ae1a4SAndy Whitcroft
3874171ae1a4SAndy Whitcroft					} elsif ($ctx =~ /.xW/) {
38753705ce5bSJoe Perches						if (ERROR("SPACING",
38763705ce5bSJoe Perches							  "space prohibited after that '$op' $at\n" . $hereptr)) {
3877b34c648bSJoe Perches							$good = $fix_elements[$n] . rtrim($fix_elements[$n + 1]);
38783705ce5bSJoe Perches							if (defined $fix_elements[$n + 2]) {
38793705ce5bSJoe Perches								$fix_elements[$n + 2] =~ s/^\s+//;
38803705ce5bSJoe Perches							}
3881b34c648bSJoe Perches							$line_fixed = 1;
38823705ce5bSJoe Perches						}
38830a920b5bSAndy Whitcroft					}
38840a920b5bSAndy Whitcroft
38850a920b5bSAndy Whitcroft				# unary ++ and unary -- are allowed no space on one side.
38860a920b5bSAndy Whitcroft				} elsif ($op eq '++' or $op eq '--') {
3887773647a0SAndy Whitcroft					if ($ctx !~ /[WEOBC]x[^W]/ && $ctx !~ /[^W]x[WOBEC]/) {
38883705ce5bSJoe Perches						if (ERROR("SPACING",
38893705ce5bSJoe Perches							  "space required one side of that '$op' $at\n" . $hereptr)) {
3890b34c648bSJoe Perches							$good = $fix_elements[$n] . trim($fix_elements[$n + 1]) . " ";
38913705ce5bSJoe Perches							$line_fixed = 1;
38923705ce5bSJoe Perches						}
38930a920b5bSAndy Whitcroft					}
3894773647a0SAndy Whitcroft					if ($ctx =~ /Wx[BE]/ ||
3895773647a0SAndy Whitcroft					    ($ctx =~ /Wx./ && $cc =~ /^;/)) {
38963705ce5bSJoe Perches						if (ERROR("SPACING",
38973705ce5bSJoe Perches							  "space prohibited before that '$op' $at\n" . $hereptr)) {
3898b34c648bSJoe Perches							$good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]);
38993705ce5bSJoe Perches							$line_fixed = 1;
39003705ce5bSJoe Perches						}
3901653d4876SAndy Whitcroft					}
3902773647a0SAndy Whitcroft					if ($ctx =~ /ExW/) {
39033705ce5bSJoe Perches						if (ERROR("SPACING",
39043705ce5bSJoe Perches							  "space prohibited after that '$op' $at\n" . $hereptr)) {
3905b34c648bSJoe Perches							$good = $fix_elements[$n] . trim($fix_elements[$n + 1]);
39063705ce5bSJoe Perches							if (defined $fix_elements[$n + 2]) {
39073705ce5bSJoe Perches								$fix_elements[$n + 2] =~ s/^\s+//;
3908773647a0SAndy Whitcroft							}
3909b34c648bSJoe Perches							$line_fixed = 1;
39103705ce5bSJoe Perches						}
39113705ce5bSJoe Perches					}
39120a920b5bSAndy Whitcroft
39130a920b5bSAndy Whitcroft				# << and >> may either have or not have spaces both sides
39149c0ca6f9SAndy Whitcroft				} elsif ($op eq '<<' or $op eq '>>' or
39159c0ca6f9SAndy Whitcroft					 $op eq '&' or $op eq '^' or $op eq '|' or
39169c0ca6f9SAndy Whitcroft					 $op eq '+' or $op eq '-' or
3917c2fdda0dSAndy Whitcroft					 $op eq '*' or $op eq '/' or
3918c2fdda0dSAndy Whitcroft					 $op eq '%')
39190a920b5bSAndy Whitcroft				{
3920d2e025f3SJoe Perches					if ($check) {
3921d2e025f3SJoe Perches						if (defined $fix_elements[$n + 2] && $ctx !~ /[EW]x[EW]/) {
3922d2e025f3SJoe Perches							if (CHK("SPACING",
3923d2e025f3SJoe Perches								"spaces preferred around that '$op' $at\n" . $hereptr)) {
3924d2e025f3SJoe Perches								$good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " ";
3925d2e025f3SJoe Perches								$fix_elements[$n + 2] =~ s/^\s+//;
3926d2e025f3SJoe Perches								$line_fixed = 1;
3927d2e025f3SJoe Perches							}
3928d2e025f3SJoe Perches						} elsif (!defined $fix_elements[$n + 2] && $ctx !~ /Wx[OE]/) {
3929d2e025f3SJoe Perches							if (CHK("SPACING",
3930d2e025f3SJoe Perches								"space preferred before that '$op' $at\n" . $hereptr)) {
3931d2e025f3SJoe Perches								$good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]);
3932d2e025f3SJoe Perches								$line_fixed = 1;
3933d2e025f3SJoe Perches							}
3934d2e025f3SJoe Perches						}
3935d2e025f3SJoe Perches					} elsif ($ctx =~ /Wx[^WCE]|[^WCE]xW/) {
39363705ce5bSJoe Perches						if (ERROR("SPACING",
39373705ce5bSJoe Perches							  "need consistent spacing around '$op' $at\n" . $hereptr)) {
3938b34c648bSJoe Perches							$good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " ";
3939b34c648bSJoe Perches							if (defined $fix_elements[$n + 2]) {
3940b34c648bSJoe Perches								$fix_elements[$n + 2] =~ s/^\s+//;
3941b34c648bSJoe Perches							}
39423705ce5bSJoe Perches							$line_fixed = 1;
39433705ce5bSJoe Perches						}
39440a920b5bSAndy Whitcroft					}
39450a920b5bSAndy Whitcroft
39461f65f947SAndy Whitcroft				# A colon needs no spaces before when it is
39471f65f947SAndy Whitcroft				# terminating a case value or a label.
39481f65f947SAndy Whitcroft				} elsif ($opv eq ':C' || $opv eq ':L') {
39491f65f947SAndy Whitcroft					if ($ctx =~ /Wx./) {
39503705ce5bSJoe Perches						if (ERROR("SPACING",
39513705ce5bSJoe Perches							  "space prohibited before that '$op' $at\n" . $hereptr)) {
3952b34c648bSJoe Perches							$good = rtrim($fix_elements[$n]) . trim($fix_elements[$n + 1]);
39533705ce5bSJoe Perches							$line_fixed = 1;
39543705ce5bSJoe Perches						}
39551f65f947SAndy Whitcroft					}
39561f65f947SAndy Whitcroft
39570a920b5bSAndy Whitcroft				# All the others need spaces both sides.
3958cf655043SAndy Whitcroft				} elsif ($ctx !~ /[EWC]x[CWE]/) {
39591f65f947SAndy Whitcroft					my $ok = 0;
39601f65f947SAndy Whitcroft
396122f2a2efSAndy Whitcroft					# Ignore email addresses <foo@bar>
39621f65f947SAndy Whitcroft					if (($op eq '<' &&
39631f65f947SAndy Whitcroft					     $cc =~ /^\S+\@\S+>/) ||
39641f65f947SAndy Whitcroft					    ($op eq '>' &&
39651f65f947SAndy Whitcroft					     $ca =~ /<\S+\@\S+$/))
39661f65f947SAndy Whitcroft					{
39671f65f947SAndy Whitcroft					    	$ok = 1;
39681f65f947SAndy Whitcroft					}
39691f65f947SAndy Whitcroft
3970e0df7e1fSJoe Perches					# for asm volatile statements
3971e0df7e1fSJoe Perches					# ignore a colon with another
3972e0df7e1fSJoe Perches					# colon immediately before or after
3973e0df7e1fSJoe Perches					if (($op eq ':') &&
3974e0df7e1fSJoe Perches					    ($ca =~ /:$/ || $cc =~ /^:/)) {
3975e0df7e1fSJoe Perches						$ok = 1;
3976e0df7e1fSJoe Perches					}
3977e0df7e1fSJoe Perches
397884731623SJoe Perches					# messages are ERROR, but ?: are CHK
39791f65f947SAndy Whitcroft					if ($ok == 0) {
398084731623SJoe Perches						my $msg_type = \&ERROR;
398184731623SJoe Perches						$msg_type = \&CHK if (($op eq '?:' || $op eq '?' || $op eq ':') && $ctx =~ /VxV/);
398284731623SJoe Perches
398384731623SJoe Perches						if (&{$msg_type}("SPACING",
39843705ce5bSJoe Perches								 "spaces required around that '$op' $at\n" . $hereptr)) {
3985b34c648bSJoe Perches							$good = rtrim($fix_elements[$n]) . " " . trim($fix_elements[$n + 1]) . " ";
3986b34c648bSJoe Perches							if (defined $fix_elements[$n + 2]) {
3987b34c648bSJoe Perches								$fix_elements[$n + 2] =~ s/^\s+//;
3988b34c648bSJoe Perches							}
39893705ce5bSJoe Perches							$line_fixed = 1;
39903705ce5bSJoe Perches						}
39910a920b5bSAndy Whitcroft					}
399222f2a2efSAndy Whitcroft				}
39934a0df2efSAndy Whitcroft				$off += length($elements[$n + 1]);
39943705ce5bSJoe Perches
39953705ce5bSJoe Perches##				print("n: <$n> GOOD: <$good>\n");
39963705ce5bSJoe Perches
39973705ce5bSJoe Perches				$fixed_line = $fixed_line . $good;
39980a920b5bSAndy Whitcroft			}
39993705ce5bSJoe Perches
40003705ce5bSJoe Perches			if (($#elements % 2) == 0) {
40013705ce5bSJoe Perches				$fixed_line = $fixed_line . $fix_elements[$#elements];
40023705ce5bSJoe Perches			}
40033705ce5bSJoe Perches
4004194f66fcSJoe Perches			if ($fix && $line_fixed && $fixed_line ne $fixed[$fixlinenr]) {
4005194f66fcSJoe Perches				$fixed[$fixlinenr] = $fixed_line;
40063705ce5bSJoe Perches			}
40073705ce5bSJoe Perches
40083705ce5bSJoe Perches
40090a920b5bSAndy Whitcroft		}
40100a920b5bSAndy Whitcroft
4011786b6326SJoe Perches# check for whitespace before a non-naked semicolon
4012d2e248e7SJoe Perches		if ($line =~ /^\+.*\S\s+;\s*$/) {
4013786b6326SJoe Perches			if (WARN("SPACING",
4014786b6326SJoe Perches				 "space prohibited before semicolon\n" . $herecurr) &&
4015786b6326SJoe Perches			    $fix) {
4016194f66fcSJoe Perches				1 while $fixed[$fixlinenr] =~
4017786b6326SJoe Perches				    s/^(\+.*\S)\s+;/$1;/;
4018786b6326SJoe Perches			}
4019786b6326SJoe Perches		}
4020786b6326SJoe Perches
4021f0a594c1SAndy Whitcroft# check for multiple assignments
4022f0a594c1SAndy Whitcroft		if ($line =~ /^.\s*$Lval\s*=\s*$Lval\s*=(?!=)/) {
4023000d1cc1SJoe Perches			CHK("MULTIPLE_ASSIGNMENTS",
4024000d1cc1SJoe Perches			    "multiple assignments should be avoided\n" . $herecurr);
4025f0a594c1SAndy Whitcroft		}
4026f0a594c1SAndy Whitcroft
402722f2a2efSAndy Whitcroft## # check for multiple declarations, allowing for a function declaration
402822f2a2efSAndy Whitcroft## # continuation.
402922f2a2efSAndy Whitcroft## 		if ($line =~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Ident.*/ &&
403022f2a2efSAndy Whitcroft## 		    $line !~ /^.\s*$Type\s+$Ident(?:\s*=[^,{]*)?\s*,\s*$Type\s*$Ident.*/) {
403122f2a2efSAndy Whitcroft##
403222f2a2efSAndy Whitcroft## 			# Remove any bracketed sections to ensure we do not
403322f2a2efSAndy Whitcroft## 			# falsly report the parameters of functions.
403422f2a2efSAndy Whitcroft## 			my $ln = $line;
403522f2a2efSAndy Whitcroft## 			while ($ln =~ s/\([^\(\)]*\)//g) {
403622f2a2efSAndy Whitcroft## 			}
403722f2a2efSAndy Whitcroft## 			if ($ln =~ /,/) {
4038000d1cc1SJoe Perches## 				WARN("MULTIPLE_DECLARATION",
4039000d1cc1SJoe Perches##				     "declaring multiple variables together should be avoided\n" . $herecurr);
404022f2a2efSAndy Whitcroft## 			}
404122f2a2efSAndy Whitcroft## 		}
4042f0a594c1SAndy Whitcroft
40430a920b5bSAndy Whitcroft#need space before brace following if, while, etc
404422f2a2efSAndy Whitcroft		if (($line =~ /\(.*\){/ && $line !~ /\($Type\){/) ||
404522f2a2efSAndy Whitcroft		    $line =~ /do{/) {
40463705ce5bSJoe Perches			if (ERROR("SPACING",
40473705ce5bSJoe Perches				  "space required before the open brace '{'\n" . $herecurr) &&
40483705ce5bSJoe Perches			    $fix) {
4049194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/^(\+.*(?:do|\))){/$1 {/;
40503705ce5bSJoe Perches			}
4051de7d4f0eSAndy Whitcroft		}
4052de7d4f0eSAndy Whitcroft
4053c4a62ef9SJoe Perches## # check for blank lines before declarations
4054c4a62ef9SJoe Perches##		if ($line =~ /^.\t+$Type\s+$Ident(?:\s*=.*)?;/ &&
4055c4a62ef9SJoe Perches##		    $prevrawline =~ /^.\s*$/) {
4056c4a62ef9SJoe Perches##			WARN("SPACING",
4057c4a62ef9SJoe Perches##			     "No blank lines before declarations\n" . $hereprev);
4058c4a62ef9SJoe Perches##		}
4059c4a62ef9SJoe Perches##
4060c4a62ef9SJoe Perches
4061de7d4f0eSAndy Whitcroft# closing brace should have a space following it when it has anything
4062de7d4f0eSAndy Whitcroft# on the line
4063de7d4f0eSAndy Whitcroft		if ($line =~ /}(?!(?:,|;|\)))\S/) {
4064d5e616fcSJoe Perches			if (ERROR("SPACING",
4065d5e616fcSJoe Perches				  "space required after that close brace '}'\n" . $herecurr) &&
4066d5e616fcSJoe Perches			    $fix) {
4067194f66fcSJoe Perches				$fixed[$fixlinenr] =~
4068d5e616fcSJoe Perches				    s/}((?!(?:,|;|\)))\S)/} $1/;
4069d5e616fcSJoe Perches			}
40700a920b5bSAndy Whitcroft		}
40710a920b5bSAndy Whitcroft
407222f2a2efSAndy Whitcroft# check spacing on square brackets
407322f2a2efSAndy Whitcroft		if ($line =~ /\[\s/ && $line !~ /\[\s*$/) {
40743705ce5bSJoe Perches			if (ERROR("SPACING",
40753705ce5bSJoe Perches				  "space prohibited after that open square bracket '['\n" . $herecurr) &&
40763705ce5bSJoe Perches			    $fix) {
4077194f66fcSJoe Perches				$fixed[$fixlinenr] =~
40783705ce5bSJoe Perches				    s/\[\s+/\[/;
40793705ce5bSJoe Perches			}
408022f2a2efSAndy Whitcroft		}
408122f2a2efSAndy Whitcroft		if ($line =~ /\s\]/) {
40823705ce5bSJoe Perches			if (ERROR("SPACING",
40833705ce5bSJoe Perches				  "space prohibited before that close square bracket ']'\n" . $herecurr) &&
40843705ce5bSJoe Perches			    $fix) {
4085194f66fcSJoe Perches				$fixed[$fixlinenr] =~
40863705ce5bSJoe Perches				    s/\s+\]/\]/;
40873705ce5bSJoe Perches			}
408822f2a2efSAndy Whitcroft		}
408922f2a2efSAndy Whitcroft
4090c45dcabdSAndy Whitcroft# check spacing on parentheses
40919c0ca6f9SAndy Whitcroft		if ($line =~ /\(\s/ && $line !~ /\(\s*(?:\\)?$/ &&
40929c0ca6f9SAndy Whitcroft		    $line !~ /for\s*\(\s+;/) {
40933705ce5bSJoe Perches			if (ERROR("SPACING",
40943705ce5bSJoe Perches				  "space prohibited after that open parenthesis '('\n" . $herecurr) &&
40953705ce5bSJoe Perches			    $fix) {
4096194f66fcSJoe Perches				$fixed[$fixlinenr] =~
40973705ce5bSJoe Perches				    s/\(\s+/\(/;
40983705ce5bSJoe Perches			}
409922f2a2efSAndy Whitcroft		}
410013214adfSAndy Whitcroft		if ($line =~ /(\s+)\)/ && $line !~ /^.\s*\)/ &&
4101c45dcabdSAndy Whitcroft		    $line !~ /for\s*\(.*;\s+\)/ &&
4102c45dcabdSAndy Whitcroft		    $line !~ /:\s+\)/) {
41033705ce5bSJoe Perches			if (ERROR("SPACING",
41043705ce5bSJoe Perches				  "space prohibited before that close parenthesis ')'\n" . $herecurr) &&
41053705ce5bSJoe Perches			    $fix) {
4106194f66fcSJoe Perches				$fixed[$fixlinenr] =~
41073705ce5bSJoe Perches				    s/\s+\)/\)/;
41083705ce5bSJoe Perches			}
410922f2a2efSAndy Whitcroft		}
411022f2a2efSAndy Whitcroft
4111e2826fd0SJoe Perches# check unnecessary parentheses around addressof/dereference single $Lvals
4112e2826fd0SJoe Perches# ie: &(foo->bar) should be &foo->bar and *(foo->bar) should be *foo->bar
4113e2826fd0SJoe Perches
4114e2826fd0SJoe Perches		while ($line =~ /(?:[^&]&\s*|\*)\(\s*($Ident\s*(?:$Member\s*)+)\s*\)/g) {
4115ea4acbb1SJoe Perches			my $var = $1;
4116ea4acbb1SJoe Perches			if (CHK("UNNECESSARY_PARENTHESES",
4117ea4acbb1SJoe Perches				"Unnecessary parentheses around $var\n" . $herecurr) &&
4118ea4acbb1SJoe Perches			    $fix) {
4119ea4acbb1SJoe Perches				$fixed[$fixlinenr] =~ s/\(\s*\Q$var\E\s*\)/$var/;
4120ea4acbb1SJoe Perches			}
4121ea4acbb1SJoe Perches		}
4122ea4acbb1SJoe Perches
4123ea4acbb1SJoe Perches# check for unnecessary parentheses around function pointer uses
4124ea4acbb1SJoe Perches# ie: (foo->bar)(); should be foo->bar();
4125ea4acbb1SJoe Perches# but not "if (foo->bar) (" to avoid some false positives
4126ea4acbb1SJoe Perches		if ($line =~ /(\bif\s*|)(\(\s*$Ident\s*(?:$Member\s*)+\))[ \t]*\(/ && $1 !~ /^if/) {
4127ea4acbb1SJoe Perches			my $var = $2;
4128ea4acbb1SJoe Perches			if (CHK("UNNECESSARY_PARENTHESES",
4129ea4acbb1SJoe Perches				"Unnecessary parentheses around function pointer $var\n" . $herecurr) &&
4130ea4acbb1SJoe Perches			    $fix) {
4131ea4acbb1SJoe Perches				my $var2 = deparenthesize($var);
4132ea4acbb1SJoe Perches				$var2 =~ s/\s//g;
4133ea4acbb1SJoe Perches				$fixed[$fixlinenr] =~ s/\Q$var\E/$var2/;
4134ea4acbb1SJoe Perches			}
4135e2826fd0SJoe Perches		}
4136e2826fd0SJoe Perches
41370a920b5bSAndy Whitcroft#goto labels aren't indented, allow a single space however
41384a0df2efSAndy Whitcroft		if ($line=~/^.\s+[A-Za-z\d_]+:(?![0-9]+)/ and
41390a920b5bSAndy Whitcroft		   !($line=~/^. [A-Za-z\d_]+:/) and !($line=~/^.\s+default:/)) {
41403705ce5bSJoe Perches			if (WARN("INDENTED_LABEL",
41413705ce5bSJoe Perches				 "labels should not be indented\n" . $herecurr) &&
41423705ce5bSJoe Perches			    $fix) {
4143194f66fcSJoe Perches				$fixed[$fixlinenr] =~
41443705ce5bSJoe Perches				    s/^(.)\s+/$1/;
41453705ce5bSJoe Perches			}
41460a920b5bSAndy Whitcroft		}
41470a920b5bSAndy Whitcroft
41485b9553abSJoe Perches# return is not a function
4149507e5141SJoe Perches		if (defined($stat) && $stat =~ /^.\s*return(\s*)\(/s) {
4150c45dcabdSAndy Whitcroft			my $spacing = $1;
4151507e5141SJoe Perches			if ($^V && $^V ge 5.10.0 &&
41525b9553abSJoe Perches			    $stat =~ /^.\s*return\s*($balanced_parens)\s*;\s*$/) {
41535b9553abSJoe Perches				my $value = $1;
41545b9553abSJoe Perches				$value = deparenthesize($value);
41555b9553abSJoe Perches				if ($value =~ m/^\s*$FuncArg\s*(?:\?|$)/) {
4156000d1cc1SJoe Perches					ERROR("RETURN_PARENTHESES",
4157000d1cc1SJoe Perches					      "return is not a function, parentheses are not required\n" . $herecurr);
41585b9553abSJoe Perches				}
4159c45dcabdSAndy Whitcroft			} elsif ($spacing !~ /\s+/) {
4160000d1cc1SJoe Perches				ERROR("SPACING",
4161000d1cc1SJoe Perches				      "space required before the open parenthesis '('\n" . $herecurr);
4162c45dcabdSAndy Whitcroft			}
4163c45dcabdSAndy Whitcroft		}
4164507e5141SJoe Perches
4165b43ae21bSJoe Perches# unnecessary return in a void function
4166b43ae21bSJoe Perches# at end-of-function, with the previous line a single leading tab, then return;
4167b43ae21bSJoe Perches# and the line before that not a goto label target like "out:"
4168b43ae21bSJoe Perches		if ($sline =~ /^[ \+]}\s*$/ &&
4169b43ae21bSJoe Perches		    $prevline =~ /^\+\treturn\s*;\s*$/ &&
4170b43ae21bSJoe Perches		    $linenr >= 3 &&
4171b43ae21bSJoe Perches		    $lines[$linenr - 3] =~ /^[ +]/ &&
4172b43ae21bSJoe Perches		    $lines[$linenr - 3] !~ /^[ +]\s*$Ident\s*:/) {
41739819cf25SJoe Perches			WARN("RETURN_VOID",
4174b43ae21bSJoe Perches			     "void function return statements are not generally useful\n" . $hereprev);
41759819cf25SJoe Perches               }
41769819cf25SJoe Perches
4177189248d8SJoe Perches# if statements using unnecessary parentheses - ie: if ((foo == bar))
4178189248d8SJoe Perches		if ($^V && $^V ge 5.10.0 &&
4179189248d8SJoe Perches		    $line =~ /\bif\s*((?:\(\s*){2,})/) {
4180189248d8SJoe Perches			my $openparens = $1;
4181189248d8SJoe Perches			my $count = $openparens =~ tr@\(@\(@;
4182189248d8SJoe Perches			my $msg = "";
4183189248d8SJoe Perches			if ($line =~ /\bif\s*(?:\(\s*){$count,$count}$LvalOrFunc\s*($Compare)\s*$LvalOrFunc(?:\s*\)){$count,$count}/) {
4184189248d8SJoe Perches				my $comp = $4;	#Not $1 because of $LvalOrFunc
4185189248d8SJoe Perches				$msg = " - maybe == should be = ?" if ($comp eq "==");
4186189248d8SJoe Perches				WARN("UNNECESSARY_PARENTHESES",
4187189248d8SJoe Perches				     "Unnecessary parentheses$msg\n" . $herecurr);
4188189248d8SJoe Perches			}
4189189248d8SJoe Perches		}
4190189248d8SJoe Perches
4191f34e4a4fSJoe Perches# Return of what appears to be an errno should normally be negative
4192f34e4a4fSJoe Perches		if ($sline =~ /\breturn(?:\s*\(+\s*|\s+)(E[A-Z]+)(?:\s*\)+\s*|\s*)[;:,]/) {
419353a3c448SAndy Whitcroft			my $name = $1;
419453a3c448SAndy Whitcroft			if ($name ne 'EOF' && $name ne 'ERROR') {
4195000d1cc1SJoe Perches				WARN("USE_NEGATIVE_ERRNO",
4196f34e4a4fSJoe Perches				     "return of an errno should typically be negative (ie: return -$1)\n" . $herecurr);
419753a3c448SAndy Whitcroft			}
419853a3c448SAndy Whitcroft		}
4199c45dcabdSAndy Whitcroft
42000a920b5bSAndy Whitcroft# Need a space before open parenthesis after if, while etc
42014a0df2efSAndy Whitcroft		if ($line =~ /\b(if|while|for|switch)\(/) {
42023705ce5bSJoe Perches			if (ERROR("SPACING",
42033705ce5bSJoe Perches				  "space required before the open parenthesis '('\n" . $herecurr) &&
42043705ce5bSJoe Perches			    $fix) {
4205194f66fcSJoe Perches				$fixed[$fixlinenr] =~
42063705ce5bSJoe Perches				    s/\b(if|while|for|switch)\(/$1 \(/;
42073705ce5bSJoe Perches			}
42080a920b5bSAndy Whitcroft		}
42090a920b5bSAndy Whitcroft
4210f5fe35ddSAndy Whitcroft# Check for illegal assignment in if conditional -- and check for trailing
4211f5fe35ddSAndy Whitcroft# statements after the conditional.
4212170d3a22SAndy Whitcroft		if ($line =~ /do\s*(?!{)/) {
42133e469cdcSAndy Whitcroft			($stat, $cond, $line_nr_next, $remain_next, $off_next) =
42143e469cdcSAndy Whitcroft				ctx_statement_block($linenr, $realcnt, 0)
42153e469cdcSAndy Whitcroft					if (!defined $stat);
4216170d3a22SAndy Whitcroft			my ($stat_next) = ctx_statement_block($line_nr_next,
4217170d3a22SAndy Whitcroft						$remain_next, $off_next);
4218170d3a22SAndy Whitcroft			$stat_next =~ s/\n./\n /g;
4219170d3a22SAndy Whitcroft			##print "stat<$stat> stat_next<$stat_next>\n";
4220170d3a22SAndy Whitcroft
4221170d3a22SAndy Whitcroft			if ($stat_next =~ /^\s*while\b/) {
4222170d3a22SAndy Whitcroft				# If the statement carries leading newlines,
4223170d3a22SAndy Whitcroft				# then count those as offsets.
4224170d3a22SAndy Whitcroft				my ($whitespace) =
4225170d3a22SAndy Whitcroft					($stat_next =~ /^((?:\s*\n[+-])*\s*)/s);
4226170d3a22SAndy Whitcroft				my $offset =
4227170d3a22SAndy Whitcroft					statement_rawlines($whitespace) - 1;
4228170d3a22SAndy Whitcroft
4229170d3a22SAndy Whitcroft				$suppress_whiletrailers{$line_nr_next +
4230170d3a22SAndy Whitcroft								$offset} = 1;
4231170d3a22SAndy Whitcroft			}
4232170d3a22SAndy Whitcroft		}
4233170d3a22SAndy Whitcroft		if (!defined $suppress_whiletrailers{$linenr} &&
4234c11230f4SJoe Perches		    defined($stat) && defined($cond) &&
4235170d3a22SAndy Whitcroft		    $line =~ /\b(?:if|while|for)\s*\(/ && $line !~ /^.\s*#/) {
4236171ae1a4SAndy Whitcroft			my ($s, $c) = ($stat, $cond);
42378905a67cSAndy Whitcroft
4238b53c8e10SAndy Whitcroft			if ($c =~ /\bif\s*\(.*[^<>!=]=[^=].*/s) {
4239000d1cc1SJoe Perches				ERROR("ASSIGN_IN_IF",
4240000d1cc1SJoe Perches				      "do not use assignment in if condition\n" . $herecurr);
42418905a67cSAndy Whitcroft			}
42428905a67cSAndy Whitcroft
42438905a67cSAndy Whitcroft			# Find out what is on the end of the line after the
42448905a67cSAndy Whitcroft			# conditional.
4245773647a0SAndy Whitcroft			substr($s, 0, length($c), '');
42468905a67cSAndy Whitcroft			$s =~ s/\n.*//g;
424713214adfSAndy Whitcroft			$s =~ s/$;//g; 	# Remove any comments
424853210168SAndy Whitcroft			if (length($c) && $s !~ /^\s*{?\s*\\*\s*$/ &&
424953210168SAndy Whitcroft			    $c !~ /}\s*while\s*/)
4250773647a0SAndy Whitcroft			{
4251bb44ad39SAndy Whitcroft				# Find out how long the conditional actually is.
4252bb44ad39SAndy Whitcroft				my @newlines = ($c =~ /\n/gs);
4253bb44ad39SAndy Whitcroft				my $cond_lines = 1 + $#newlines;
425442bdf74cSHidetoshi Seto				my $stat_real = '';
4255bb44ad39SAndy Whitcroft
425642bdf74cSHidetoshi Seto				$stat_real = raw_line($linenr, $cond_lines)
425742bdf74cSHidetoshi Seto							. "\n" if ($cond_lines);
4258bb44ad39SAndy Whitcroft				if (defined($stat_real) && $cond_lines > 1) {
4259bb44ad39SAndy Whitcroft					$stat_real = "[...]\n$stat_real";
4260bb44ad39SAndy Whitcroft				}
4261bb44ad39SAndy Whitcroft
4262000d1cc1SJoe Perches				ERROR("TRAILING_STATEMENTS",
4263000d1cc1SJoe Perches				      "trailing statements should be on next line\n" . $herecurr . $stat_real);
42648905a67cSAndy Whitcroft			}
42658905a67cSAndy Whitcroft		}
42668905a67cSAndy Whitcroft
426713214adfSAndy Whitcroft# Check for bitwise tests written as boolean
426813214adfSAndy Whitcroft		if ($line =~ /
426913214adfSAndy Whitcroft			(?:
427013214adfSAndy Whitcroft				(?:\[|\(|\&\&|\|\|)
427113214adfSAndy Whitcroft				\s*0[xX][0-9]+\s*
427213214adfSAndy Whitcroft				(?:\&\&|\|\|)
427313214adfSAndy Whitcroft			|
427413214adfSAndy Whitcroft				(?:\&\&|\|\|)
427513214adfSAndy Whitcroft				\s*0[xX][0-9]+\s*
427613214adfSAndy Whitcroft				(?:\&\&|\|\||\)|\])
427713214adfSAndy Whitcroft			)/x)
427813214adfSAndy Whitcroft		{
4279000d1cc1SJoe Perches			WARN("HEXADECIMAL_BOOLEAN_TEST",
4280000d1cc1SJoe Perches			     "boolean test with hexadecimal, perhaps just 1 \& or \|?\n" . $herecurr);
428113214adfSAndy Whitcroft		}
428213214adfSAndy Whitcroft
42838905a67cSAndy Whitcroft# if and else should not have general statements after it
428413214adfSAndy Whitcroft		if ($line =~ /^.\s*(?:}\s*)?else\b(.*)/) {
428513214adfSAndy Whitcroft			my $s = $1;
428613214adfSAndy Whitcroft			$s =~ s/$;//g; 	# Remove any comments
428713214adfSAndy Whitcroft			if ($s !~ /^\s*(?:\sif|(?:{|)\s*\\?\s*$)/) {
4288000d1cc1SJoe Perches				ERROR("TRAILING_STATEMENTS",
4289000d1cc1SJoe Perches				      "trailing statements should be on next line\n" . $herecurr);
42900a920b5bSAndy Whitcroft			}
429113214adfSAndy Whitcroft		}
429239667782SAndy Whitcroft# if should not continue a brace
429339667782SAndy Whitcroft		if ($line =~ /}\s*if\b/) {
4294000d1cc1SJoe Perches			ERROR("TRAILING_STATEMENTS",
4295048b123fSRasmus Villemoes			      "trailing statements should be on next line (or did you mean 'else if'?)\n" .
429639667782SAndy Whitcroft				$herecurr);
429739667782SAndy Whitcroft		}
4298a1080bf8SAndy Whitcroft# case and default should not have general statements after them
4299a1080bf8SAndy Whitcroft		if ($line =~ /^.\s*(?:case\s*.*|default\s*):/g &&
4300a1080bf8SAndy Whitcroft		    $line !~ /\G(?:
43013fef12d6SAndy Whitcroft			(?:\s*$;*)(?:\s*{)?(?:\s*$;*)(?:\s*\\)?\s*$|
4302a1080bf8SAndy Whitcroft			\s*return\s+
4303a1080bf8SAndy Whitcroft		    )/xg)
4304a1080bf8SAndy Whitcroft		{
4305000d1cc1SJoe Perches			ERROR("TRAILING_STATEMENTS",
4306000d1cc1SJoe Perches			      "trailing statements should be on next line\n" . $herecurr);
4307a1080bf8SAndy Whitcroft		}
43080a920b5bSAndy Whitcroft
43090a920b5bSAndy Whitcroft		# Check for }<nl>else {, these must be at the same
43100a920b5bSAndy Whitcroft		# indent level to be relevant to each other.
43118b8856f4SJoe Perches		if ($prevline=~/}\s*$/ and $line=~/^.\s*else\s*/ &&
43120a920b5bSAndy Whitcroft		    $previndent == $indent) {
43138b8856f4SJoe Perches			if (ERROR("ELSE_AFTER_BRACE",
43148b8856f4SJoe Perches				  "else should follow close brace '}'\n" . $hereprev) &&
43158b8856f4SJoe Perches			    $fix && $prevline =~ /^\+/ && $line =~ /^\+/) {
43168b8856f4SJoe Perches				fix_delete_line($fixlinenr - 1, $prevrawline);
43178b8856f4SJoe Perches				fix_delete_line($fixlinenr, $rawline);
43188b8856f4SJoe Perches				my $fixedline = $prevrawline;
43198b8856f4SJoe Perches				$fixedline =~ s/}\s*$//;
43208b8856f4SJoe Perches				if ($fixedline !~ /^\+\s*$/) {
43218b8856f4SJoe Perches					fix_insert_line($fixlinenr, $fixedline);
43228b8856f4SJoe Perches				}
43238b8856f4SJoe Perches				$fixedline = $rawline;
43248b8856f4SJoe Perches				$fixedline =~ s/^(.\s*)else/$1} else/;
43258b8856f4SJoe Perches				fix_insert_line($fixlinenr, $fixedline);
43268b8856f4SJoe Perches			}
43270a920b5bSAndy Whitcroft		}
43280a920b5bSAndy Whitcroft
43298b8856f4SJoe Perches		if ($prevline=~/}\s*$/ and $line=~/^.\s*while\s*/ &&
4330c2fdda0dSAndy Whitcroft		    $previndent == $indent) {
4331c2fdda0dSAndy Whitcroft			my ($s, $c) = ctx_statement_block($linenr, $realcnt, 0);
4332c2fdda0dSAndy Whitcroft
4333c2fdda0dSAndy Whitcroft			# Find out what is on the end of the line after the
4334c2fdda0dSAndy Whitcroft			# conditional.
4335773647a0SAndy Whitcroft			substr($s, 0, length($c), '');
4336c2fdda0dSAndy Whitcroft			$s =~ s/\n.*//g;
4337c2fdda0dSAndy Whitcroft
4338c2fdda0dSAndy Whitcroft			if ($s =~ /^\s*;/) {
43398b8856f4SJoe Perches				if (ERROR("WHILE_AFTER_BRACE",
43408b8856f4SJoe Perches					  "while should follow close brace '}'\n" . $hereprev) &&
43418b8856f4SJoe Perches				    $fix && $prevline =~ /^\+/ && $line =~ /^\+/) {
43428b8856f4SJoe Perches					fix_delete_line($fixlinenr - 1, $prevrawline);
43438b8856f4SJoe Perches					fix_delete_line($fixlinenr, $rawline);
43448b8856f4SJoe Perches					my $fixedline = $prevrawline;
43458b8856f4SJoe Perches					my $trailing = $rawline;
43468b8856f4SJoe Perches					$trailing =~ s/^\+//;
43478b8856f4SJoe Perches					$trailing = trim($trailing);
43488b8856f4SJoe Perches					$fixedline =~ s/}\s*$/} $trailing/;
43498b8856f4SJoe Perches					fix_insert_line($fixlinenr, $fixedline);
43508b8856f4SJoe Perches				}
4351c2fdda0dSAndy Whitcroft			}
4352c2fdda0dSAndy Whitcroft		}
4353c2fdda0dSAndy Whitcroft
435495e2c602SJoe Perches#Specific variable tests
4355323c1260SJoe Perches		while ($line =~ m{($Constant|$Lval)}g) {
4356323c1260SJoe Perches			my $var = $1;
435795e2c602SJoe Perches
435895e2c602SJoe Perches#gcc binary extension
435995e2c602SJoe Perches			if ($var =~ /^$Binary$/) {
4360d5e616fcSJoe Perches				if (WARN("GCC_BINARY_CONSTANT",
4361d5e616fcSJoe Perches					 "Avoid gcc v4.3+ binary constant extension: <$var>\n" . $herecurr) &&
4362d5e616fcSJoe Perches				    $fix) {
4363d5e616fcSJoe Perches					my $hexval = sprintf("0x%x", oct($var));
4364194f66fcSJoe Perches					$fixed[$fixlinenr] =~
4365d5e616fcSJoe Perches					    s/\b$var\b/$hexval/;
4366d5e616fcSJoe Perches				}
436795e2c602SJoe Perches			}
436895e2c602SJoe Perches
436995e2c602SJoe Perches#CamelCase
4370807bd26cSJoe Perches			if ($var !~ /^$Constant$/ &&
4371be79794bSJoe Perches			    $var =~ /[A-Z][a-z]|[a-z][A-Z]/ &&
437222735ce8SJoe Perches#Ignore Page<foo> variants
4373807bd26cSJoe Perches			    $var !~ /^(?:Clear|Set|TestClear|TestSet|)Page[A-Z]/ &&
437422735ce8SJoe Perches#Ignore SI style variants like nS, mV and dB (ie: max_uV, regulator_min_uA_show)
4375f5123576SJulius Werner			    $var !~ /^(?:[a-z_]*?)_?[a-z][A-Z](?:_[a-z_]+)?$/ &&
4376f5123576SJulius Werner#Ignore some three character SI units explicitly, like MiB and KHz
4377f5123576SJulius Werner			    $var !~ /^(?:[a-z_]*?)_?(?:[KMGT]iB|[KMGT]?Hz)(?:_[a-z_]+)?$/) {
43787e781f67SJoe Perches				while ($var =~ m{($Ident)}g) {
43797e781f67SJoe Perches					my $word = $1;
43807e781f67SJoe Perches					next if ($word !~ /[A-Z][a-z]|[a-z][A-Z]/);
4381d8b07710SJoe Perches					if ($check) {
4382d8b07710SJoe Perches						seed_camelcase_includes();
4383d8b07710SJoe Perches						if (!$file && !$camelcase_file_seeded) {
4384d8b07710SJoe Perches							seed_camelcase_file($realfile);
4385d8b07710SJoe Perches							$camelcase_file_seeded = 1;
4386d8b07710SJoe Perches						}
4387d8b07710SJoe Perches					}
43887e781f67SJoe Perches					if (!defined $camelcase{$word}) {
43897e781f67SJoe Perches						$camelcase{$word} = 1;
4390be79794bSJoe Perches						CHK("CAMELCASE",
43917e781f67SJoe Perches						    "Avoid CamelCase: <$word>\n" . $herecurr);
43927e781f67SJoe Perches					}
4393323c1260SJoe Perches				}
4394323c1260SJoe Perches			}
43953445686aSJoe Perches		}
43960a920b5bSAndy Whitcroft
43970a920b5bSAndy Whitcroft#no spaces allowed after \ in define
4398d5e616fcSJoe Perches		if ($line =~ /\#\s*define.*\\\s+$/) {
4399d5e616fcSJoe Perches			if (WARN("WHITESPACE_AFTER_LINE_CONTINUATION",
4400d5e616fcSJoe Perches				 "Whitespace after \\ makes next lines useless\n" . $herecurr) &&
4401d5e616fcSJoe Perches			    $fix) {
4402194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\s+$//;
4403d5e616fcSJoe Perches			}
44040a920b5bSAndy Whitcroft		}
44050a920b5bSAndy Whitcroft
44060e212e0aSFabian Frederick# warn if <asm/foo.h> is #included and <linux/foo.h> is available and includes
44070e212e0aSFabian Frederick# itself <asm/foo.h> (uses RAW line)
4408c45dcabdSAndy Whitcroft		if ($tree && $rawline =~ m{^.\s*\#\s*include\s*\<asm\/(.*)\.h\>}) {
4409e09dec48SAndy Whitcroft			my $file = "$1.h";
4410e09dec48SAndy Whitcroft			my $checkfile = "include/linux/$file";
4411e09dec48SAndy Whitcroft			if (-f "$root/$checkfile" &&
4412e09dec48SAndy Whitcroft			    $realfile ne $checkfile &&
44137840a94cSWolfram Sang			    $1 !~ /$allowed_asm_includes/)
4414c45dcabdSAndy Whitcroft			{
44150e212e0aSFabian Frederick				my $asminclude = `grep -Ec "#include\\s+<asm/$file>" $root/$checkfile`;
44160e212e0aSFabian Frederick				if ($asminclude > 0) {
4417e09dec48SAndy Whitcroft					if ($realfile =~ m{^arch/}) {
4418000d1cc1SJoe Perches						CHK("ARCH_INCLUDE_LINUX",
4419000d1cc1SJoe Perches						    "Consider using #include <linux/$file> instead of <asm/$file>\n" . $herecurr);
4420e09dec48SAndy Whitcroft					} else {
4421000d1cc1SJoe Perches						WARN("INCLUDE_LINUX",
4422000d1cc1SJoe Perches						     "Use #include <linux/$file> instead of <asm/$file>\n" . $herecurr);
4423e09dec48SAndy Whitcroft					}
44240a920b5bSAndy Whitcroft				}
44250a920b5bSAndy Whitcroft			}
44260e212e0aSFabian Frederick		}
44270a920b5bSAndy Whitcroft
4428653d4876SAndy Whitcroft# multi-statement macros should be enclosed in a do while loop, grab the
4429653d4876SAndy Whitcroft# first statement and ensure its the whole macro if its not enclosed
4430cf655043SAndy Whitcroft# in a known good container
4431b8f96a31SAndy Whitcroft		if ($realfile !~ m@/vmlinux.lds.h$@ &&
4432b8f96a31SAndy Whitcroft		    $line =~ /^.\s*\#\s*define\s*$Ident(\()?/) {
4433d8aaf121SAndy Whitcroft			my $ln = $linenr;
4434d8aaf121SAndy Whitcroft			my $cnt = $realcnt;
4435c45dcabdSAndy Whitcroft			my ($off, $dstat, $dcond, $rest);
4436c45dcabdSAndy Whitcroft			my $ctx = '';
443708a2843eSJoe Perches			my $has_flow_statement = 0;
443808a2843eSJoe Perches			my $has_arg_concat = 0;
4439c45dcabdSAndy Whitcroft			($dstat, $dcond, $ln, $cnt, $off) =
4440f74bd194SAndy Whitcroft				ctx_statement_block($linenr, $realcnt, 0);
4441f74bd194SAndy Whitcroft			$ctx = $dstat;
4442c45dcabdSAndy Whitcroft			#print "dstat<$dstat> dcond<$dcond> cnt<$cnt> off<$off>\n";
4443a3bb97a7SAndy Whitcroft			#print "LINE<$lines[$ln-1]> len<" . length($lines[$ln-1]) . "\n";
4444c45dcabdSAndy Whitcroft
444508a2843eSJoe Perches			$has_flow_statement = 1 if ($ctx =~ /\b(goto|return)\b/);
444608a2843eSJoe Perches			$has_arg_concat = 1 if ($ctx =~ /\#\#/);
444708a2843eSJoe Perches
4448f74bd194SAndy Whitcroft			$dstat =~ s/^.\s*\#\s*define\s+$Ident(?:\([^\)]*\))?\s*//;
4449292f1a9bSAndy Whitcroft			$dstat =~ s/$;//g;
4450c45dcabdSAndy Whitcroft			$dstat =~ s/\\\n.//g;
4451c45dcabdSAndy Whitcroft			$dstat =~ s/^\s*//s;
4452c45dcabdSAndy Whitcroft			$dstat =~ s/\s*$//s;
4453c45dcabdSAndy Whitcroft
4454c45dcabdSAndy Whitcroft			# Flatten any parentheses and braces
4455bf30d6edSAndy Whitcroft			while ($dstat =~ s/\([^\(\)]*\)/1/ ||
4456bf30d6edSAndy Whitcroft			       $dstat =~ s/\{[^\{\}]*\}/1/ ||
4457c81769fdSAndy Whitcroft			       $dstat =~ s/\[[^\[\]]*\]/1/)
4458bf30d6edSAndy Whitcroft			{
4459c45dcabdSAndy Whitcroft			}
4460c45dcabdSAndy Whitcroft
4461e45bab8eSAndy Whitcroft			# Flatten any obvious string concatentation.
446233acb54aSJoe Perches			while ($dstat =~ s/($String)\s*$Ident/$1/ ||
446333acb54aSJoe Perches			       $dstat =~ s/$Ident\s*($String)/$1/)
4464e45bab8eSAndy Whitcroft			{
4465e45bab8eSAndy Whitcroft			}
4466e45bab8eSAndy Whitcroft
4467c45dcabdSAndy Whitcroft			my $exceptions = qr{
4468c45dcabdSAndy Whitcroft				$Declare|
4469c45dcabdSAndy Whitcroft				module_param_named|
4470a0a0a7a9SKees Cook				MODULE_PARM_DESC|
4471c45dcabdSAndy Whitcroft				DECLARE_PER_CPU|
4472c45dcabdSAndy Whitcroft				DEFINE_PER_CPU|
4473383099fdSAndy Whitcroft				__typeof__\(|
447422fd2d3eSStefani Seibold				union|
447522fd2d3eSStefani Seibold				struct|
4476ea71a0a0SAndy Whitcroft				\.$Ident\s*=\s*|
4477ea71a0a0SAndy Whitcroft				^\"|\"$
4478c45dcabdSAndy Whitcroft			}x;
44795eaa20b9SAndy Whitcroft			#print "REST<$rest> dstat<$dstat> ctx<$ctx>\n";
4480f74bd194SAndy Whitcroft			if ($dstat ne '' &&
4481f74bd194SAndy Whitcroft			    $dstat !~ /^(?:$Ident|-?$Constant),$/ &&			# 10, // foo(),
4482f74bd194SAndy Whitcroft			    $dstat !~ /^(?:$Ident|-?$Constant);$/ &&			# foo();
44833cc4b1c3SJoe Perches			    $dstat !~ /^[!~-]?(?:$Lval|$Constant)$/ &&		# 10 // foo() // !foo // ~foo // -foo // foo->bar // foo.bar->baz
4484356fd398SJoe Perches			    $dstat !~ /^'X'$/ && $dstat !~ /^'XX'$/ &&			# character constants
4485f74bd194SAndy Whitcroft			    $dstat !~ /$exceptions/ &&
4486f74bd194SAndy Whitcroft			    $dstat !~ /^\.$Ident\s*=/ &&				# .foo =
4487e942e2c3SJoe Perches			    $dstat !~ /^(?:\#\s*$Ident|\#\s*$Constant)\s*$/ &&		# stringification #foo
448872f115f9SAndy Whitcroft			    $dstat !~ /^do\s*$Constant\s*while\s*$Constant;?$/ &&	# do {...} while (...); // do {...} while (...)
4489f74bd194SAndy Whitcroft			    $dstat !~ /^for\s*$Constant$/ &&				# for (...)
4490f74bd194SAndy Whitcroft			    $dstat !~ /^for\s*$Constant\s+(?:$Ident|-?$Constant)$/ &&	# for (...) bar()
4491f74bd194SAndy Whitcroft			    $dstat !~ /^do\s*{/ &&					# do {...
4492f95a7e6aSJoe Perches			    $dstat !~ /^\({/ &&						# ({...
4493f95a7e6aSJoe Perches			    $ctx !~ /^.\s*#\s*define\s+TRACE_(?:SYSTEM|INCLUDE_FILE|INCLUDE_PATH)\b/)
4494c45dcabdSAndy Whitcroft			{
4495f74bd194SAndy Whitcroft				$ctx =~ s/\n*$//;
4496f74bd194SAndy Whitcroft				my $herectx = $here . "\n";
4497f74bd194SAndy Whitcroft				my $cnt = statement_rawlines($ctx);
4498f74bd194SAndy Whitcroft
4499f74bd194SAndy Whitcroft				for (my $n = 0; $n < $cnt; $n++) {
4500f74bd194SAndy Whitcroft					$herectx .= raw_line($linenr, $n) . "\n";
4501c45dcabdSAndy Whitcroft				}
4502c45dcabdSAndy Whitcroft
4503f74bd194SAndy Whitcroft				if ($dstat =~ /;/) {
4504f74bd194SAndy Whitcroft					ERROR("MULTISTATEMENT_MACRO_USE_DO_WHILE",
4505f74bd194SAndy Whitcroft					      "Macros with multiple statements should be enclosed in a do - while loop\n" . "$herectx");
4506f74bd194SAndy Whitcroft				} else {
4507000d1cc1SJoe Perches					ERROR("COMPLEX_MACRO",
4508388982b5SAndrew Morton					      "Macros with complex values should be enclosed in parentheses\n" . "$herectx");
4509d8aaf121SAndy Whitcroft				}
45100a920b5bSAndy Whitcroft			}
45115023d347SJoe Perches
451208a2843eSJoe Perches# check for macros with flow control, but without ## concatenation
451308a2843eSJoe Perches# ## concatenation is commonly a macro that defines a function so ignore those
451408a2843eSJoe Perches			if ($has_flow_statement && !$has_arg_concat) {
451508a2843eSJoe Perches				my $herectx = $here . "\n";
451608a2843eSJoe Perches				my $cnt = statement_rawlines($ctx);
451708a2843eSJoe Perches
451808a2843eSJoe Perches				for (my $n = 0; $n < $cnt; $n++) {
451908a2843eSJoe Perches					$herectx .= raw_line($linenr, $n) . "\n";
452008a2843eSJoe Perches				}
452108a2843eSJoe Perches				WARN("MACRO_WITH_FLOW_CONTROL",
452208a2843eSJoe Perches				     "Macros with flow control statements should be avoided\n" . "$herectx");
452308a2843eSJoe Perches			}
452408a2843eSJoe Perches
4525481eb486SJoe Perches# check for line continuations outside of #defines, preprocessor #, and asm
45265023d347SJoe Perches
45275023d347SJoe Perches		} else {
45285023d347SJoe Perches			if ($prevline !~ /^..*\\$/ &&
4529481eb486SJoe Perches			    $line !~ /^\+\s*\#.*\\$/ &&		# preprocessor
4530481eb486SJoe Perches			    $line !~ /^\+.*\b(__asm__|asm)\b.*\\$/ &&	# asm
45315023d347SJoe Perches			    $line =~ /^\+.*\\$/) {
45325023d347SJoe Perches				WARN("LINE_CONTINUATIONS",
45335023d347SJoe Perches				     "Avoid unnecessary line continuations\n" . $herecurr);
45345023d347SJoe Perches			}
4535653d4876SAndy Whitcroft		}
45360a920b5bSAndy Whitcroft
4537b13edf7fSJoe Perches# do {} while (0) macro tests:
4538b13edf7fSJoe Perches# single-statement macros do not need to be enclosed in do while (0) loop,
4539b13edf7fSJoe Perches# macro should not end with a semicolon
4540b13edf7fSJoe Perches		if ($^V && $^V ge 5.10.0 &&
4541b13edf7fSJoe Perches		    $realfile !~ m@/vmlinux.lds.h$@ &&
4542b13edf7fSJoe Perches		    $line =~ /^.\s*\#\s*define\s+$Ident(\()?/) {
4543b13edf7fSJoe Perches			my $ln = $linenr;
4544b13edf7fSJoe Perches			my $cnt = $realcnt;
4545b13edf7fSJoe Perches			my ($off, $dstat, $dcond, $rest);
4546b13edf7fSJoe Perches			my $ctx = '';
4547b13edf7fSJoe Perches			($dstat, $dcond, $ln, $cnt, $off) =
4548b13edf7fSJoe Perches				ctx_statement_block($linenr, $realcnt, 0);
4549b13edf7fSJoe Perches			$ctx = $dstat;
4550b13edf7fSJoe Perches
4551b13edf7fSJoe Perches			$dstat =~ s/\\\n.//g;
45521b36b201SJoe Perches			$dstat =~ s/$;/ /g;
4553b13edf7fSJoe Perches
4554b13edf7fSJoe Perches			if ($dstat =~ /^\+\s*#\s*define\s+$Ident\s*${balanced_parens}\s*do\s*{(.*)\s*}\s*while\s*\(\s*0\s*\)\s*([;\s]*)\s*$/) {
4555b13edf7fSJoe Perches				my $stmts = $2;
4556b13edf7fSJoe Perches				my $semis = $3;
4557b13edf7fSJoe Perches
4558b13edf7fSJoe Perches				$ctx =~ s/\n*$//;
4559b13edf7fSJoe Perches				my $cnt = statement_rawlines($ctx);
4560b13edf7fSJoe Perches				my $herectx = $here . "\n";
4561b13edf7fSJoe Perches
4562b13edf7fSJoe Perches				for (my $n = 0; $n < $cnt; $n++) {
4563b13edf7fSJoe Perches					$herectx .= raw_line($linenr, $n) . "\n";
4564b13edf7fSJoe Perches				}
4565b13edf7fSJoe Perches
4566ac8e97f8SJoe Perches				if (($stmts =~ tr/;/;/) == 1 &&
4567ac8e97f8SJoe Perches				    $stmts !~ /^\s*(if|while|for|switch)\b/) {
4568b13edf7fSJoe Perches					WARN("SINGLE_STATEMENT_DO_WHILE_MACRO",
4569b13edf7fSJoe Perches					     "Single statement macros should not use a do {} while (0) loop\n" . "$herectx");
4570b13edf7fSJoe Perches				}
4571b13edf7fSJoe Perches				if (defined $semis && $semis ne "") {
4572b13edf7fSJoe Perches					WARN("DO_WHILE_MACRO_WITH_TRAILING_SEMICOLON",
4573b13edf7fSJoe Perches					     "do {} while (0) macros should not be semicolon terminated\n" . "$herectx");
4574b13edf7fSJoe Perches				}
4575f5ef95b1SJoe Perches			} elsif ($dstat =~ /^\+\s*#\s*define\s+$Ident.*;\s*$/) {
4576f5ef95b1SJoe Perches				$ctx =~ s/\n*$//;
4577f5ef95b1SJoe Perches				my $cnt = statement_rawlines($ctx);
4578f5ef95b1SJoe Perches				my $herectx = $here . "\n";
4579f5ef95b1SJoe Perches
4580f5ef95b1SJoe Perches				for (my $n = 0; $n < $cnt; $n++) {
4581f5ef95b1SJoe Perches					$herectx .= raw_line($linenr, $n) . "\n";
4582f5ef95b1SJoe Perches				}
4583f5ef95b1SJoe Perches
4584f5ef95b1SJoe Perches				WARN("TRAILING_SEMICOLON",
4585f5ef95b1SJoe Perches				     "macros should not use a trailing semicolon\n" . "$herectx");
4586b13edf7fSJoe Perches			}
4587b13edf7fSJoe Perches		}
4588b13edf7fSJoe Perches
4589080ba929SMike Frysinger# make sure symbols are always wrapped with VMLINUX_SYMBOL() ...
4590080ba929SMike Frysinger# all assignments may have only one of the following with an assignment:
4591080ba929SMike Frysinger#	.
4592080ba929SMike Frysinger#	ALIGN(...)
4593080ba929SMike Frysinger#	VMLINUX_SYMBOL(...)
4594080ba929SMike Frysinger		if ($realfile eq 'vmlinux.lds.h' && $line =~ /(?:(?:^|\s)$Ident\s*=|=\s*$Ident(?:\s|$))/) {
4595000d1cc1SJoe Perches			WARN("MISSING_VMLINUX_SYMBOL",
4596000d1cc1SJoe Perches			     "vmlinux.lds.h needs VMLINUX_SYMBOL() around C-visible symbols\n" . $herecurr);
4597080ba929SMike Frysinger		}
4598080ba929SMike Frysinger
4599f0a594c1SAndy Whitcroft# check for redundant bracing round if etc
460013214adfSAndy Whitcroft		if ($line =~ /(^.*)\bif\b/ && $1 !~ /else\s*$/) {
460113214adfSAndy Whitcroft			my ($level, $endln, @chunks) =
4602cf655043SAndy Whitcroft				ctx_statement_full($linenr, $realcnt, 1);
460313214adfSAndy Whitcroft			#print "chunks<$#chunks> linenr<$linenr> endln<$endln> level<$level>\n";
4604cf655043SAndy Whitcroft			#print "APW: <<$chunks[1][0]>><<$chunks[1][1]>>\n";
4605cf655043SAndy Whitcroft			if ($#chunks > 0 && $level == 0) {
4606aad4f614SJoe Perches				my @allowed = ();
4607aad4f614SJoe Perches				my $allow = 0;
460813214adfSAndy Whitcroft				my $seen = 0;
4609773647a0SAndy Whitcroft				my $herectx = $here . "\n";
4610cf655043SAndy Whitcroft				my $ln = $linenr - 1;
461113214adfSAndy Whitcroft				for my $chunk (@chunks) {
461213214adfSAndy Whitcroft					my ($cond, $block) = @{$chunk};
461313214adfSAndy Whitcroft
4614773647a0SAndy Whitcroft					# If the condition carries leading newlines, then count those as offsets.
4615773647a0SAndy Whitcroft					my ($whitespace) = ($cond =~ /^((?:\s*\n[+-])*\s*)/s);
4616773647a0SAndy Whitcroft					my $offset = statement_rawlines($whitespace) - 1;
4617773647a0SAndy Whitcroft
4618aad4f614SJoe Perches					$allowed[$allow] = 0;
4619773647a0SAndy Whitcroft					#print "COND<$cond> whitespace<$whitespace> offset<$offset>\n";
4620773647a0SAndy Whitcroft
4621773647a0SAndy Whitcroft					# We have looked at and allowed this specific line.
4622773647a0SAndy Whitcroft					$suppress_ifbraces{$ln + $offset} = 1;
4623773647a0SAndy Whitcroft
4624773647a0SAndy Whitcroft					$herectx .= "$rawlines[$ln + $offset]\n[...]\n";
4625cf655043SAndy Whitcroft					$ln += statement_rawlines($block) - 1;
4626cf655043SAndy Whitcroft
4627773647a0SAndy Whitcroft					substr($block, 0, length($cond), '');
462813214adfSAndy Whitcroft
462913214adfSAndy Whitcroft					$seen++ if ($block =~ /^\s*{/);
463013214adfSAndy Whitcroft
4631aad4f614SJoe Perches					#print "cond<$cond> block<$block> allowed<$allowed[$allow]>\n";
4632cf655043SAndy Whitcroft					if (statement_lines($cond) > 1) {
4633cf655043SAndy Whitcroft						#print "APW: ALLOWED: cond<$cond>\n";
4634aad4f614SJoe Perches						$allowed[$allow] = 1;
463513214adfSAndy Whitcroft					}
463613214adfSAndy Whitcroft					if ($block =~/\b(?:if|for|while)\b/) {
4637cf655043SAndy Whitcroft						#print "APW: ALLOWED: block<$block>\n";
4638aad4f614SJoe Perches						$allowed[$allow] = 1;
463913214adfSAndy Whitcroft					}
4640cf655043SAndy Whitcroft					if (statement_block_size($block) > 1) {
4641cf655043SAndy Whitcroft						#print "APW: ALLOWED: lines block<$block>\n";
4642aad4f614SJoe Perches						$allowed[$allow] = 1;
464313214adfSAndy Whitcroft					}
4644aad4f614SJoe Perches					$allow++;
464513214adfSAndy Whitcroft				}
4646aad4f614SJoe Perches				if ($seen) {
4647aad4f614SJoe Perches					my $sum_allowed = 0;
4648aad4f614SJoe Perches					foreach (@allowed) {
4649aad4f614SJoe Perches						$sum_allowed += $_;
4650aad4f614SJoe Perches					}
4651aad4f614SJoe Perches					if ($sum_allowed == 0) {
4652000d1cc1SJoe Perches						WARN("BRACES",
4653000d1cc1SJoe Perches						     "braces {} are not necessary for any arm of this statement\n" . $herectx);
4654aad4f614SJoe Perches					} elsif ($sum_allowed != $allow &&
4655aad4f614SJoe Perches						 $seen != $allow) {
4656aad4f614SJoe Perches						CHK("BRACES",
4657aad4f614SJoe Perches						    "braces {} should be used on all arms of this statement\n" . $herectx);
4658aad4f614SJoe Perches					}
465913214adfSAndy Whitcroft				}
466013214adfSAndy Whitcroft			}
466113214adfSAndy Whitcroft		}
4662773647a0SAndy Whitcroft		if (!defined $suppress_ifbraces{$linenr - 1} &&
466313214adfSAndy Whitcroft					$line =~ /\b(if|while|for|else)\b/) {
4664cf655043SAndy Whitcroft			my $allowed = 0;
4665f0a594c1SAndy Whitcroft
4666cf655043SAndy Whitcroft			# Check the pre-context.
4667cf655043SAndy Whitcroft			if (substr($line, 0, $-[0]) =~ /(\}\s*)$/) {
4668cf655043SAndy Whitcroft				#print "APW: ALLOWED: pre<$1>\n";
4669cf655043SAndy Whitcroft				$allowed = 1;
4670f0a594c1SAndy Whitcroft			}
4671773647a0SAndy Whitcroft
4672773647a0SAndy Whitcroft			my ($level, $endln, @chunks) =
4673773647a0SAndy Whitcroft				ctx_statement_full($linenr, $realcnt, $-[0]);
4674773647a0SAndy Whitcroft
4675cf655043SAndy Whitcroft			# Check the condition.
4676cf655043SAndy Whitcroft			my ($cond, $block) = @{$chunks[0]};
4677773647a0SAndy Whitcroft			#print "CHECKING<$linenr> cond<$cond> block<$block>\n";
4678cf655043SAndy Whitcroft			if (defined $cond) {
4679773647a0SAndy Whitcroft				substr($block, 0, length($cond), '');
4680cf655043SAndy Whitcroft			}
4681cf655043SAndy Whitcroft			if (statement_lines($cond) > 1) {
4682cf655043SAndy Whitcroft				#print "APW: ALLOWED: cond<$cond>\n";
4683cf655043SAndy Whitcroft				$allowed = 1;
4684cf655043SAndy Whitcroft			}
4685cf655043SAndy Whitcroft			if ($block =~/\b(?:if|for|while)\b/) {
4686cf655043SAndy Whitcroft				#print "APW: ALLOWED: block<$block>\n";
4687cf655043SAndy Whitcroft				$allowed = 1;
4688cf655043SAndy Whitcroft			}
4689cf655043SAndy Whitcroft			if (statement_block_size($block) > 1) {
4690cf655043SAndy Whitcroft				#print "APW: ALLOWED: lines block<$block>\n";
4691cf655043SAndy Whitcroft				$allowed = 1;
4692cf655043SAndy Whitcroft			}
4693cf655043SAndy Whitcroft			# Check the post-context.
4694cf655043SAndy Whitcroft			if (defined $chunks[1]) {
4695cf655043SAndy Whitcroft				my ($cond, $block) = @{$chunks[1]};
4696cf655043SAndy Whitcroft				if (defined $cond) {
4697773647a0SAndy Whitcroft					substr($block, 0, length($cond), '');
4698cf655043SAndy Whitcroft				}
4699cf655043SAndy Whitcroft				if ($block =~ /^\s*\{/) {
4700cf655043SAndy Whitcroft					#print "APW: ALLOWED: chunk-1 block<$block>\n";
4701cf655043SAndy Whitcroft					$allowed = 1;
4702cf655043SAndy Whitcroft				}
4703cf655043SAndy Whitcroft			}
4704cf655043SAndy Whitcroft			if ($level == 0 && $block =~ /^\s*\{/ && !$allowed) {
470569932487SJustin P. Mattock				my $herectx = $here . "\n";
4706f055663cSAndy Whitcroft				my $cnt = statement_rawlines($block);
4707cf655043SAndy Whitcroft
4708f055663cSAndy Whitcroft				for (my $n = 0; $n < $cnt; $n++) {
470969932487SJustin P. Mattock					$herectx .= raw_line($linenr, $n) . "\n";
4710cf655043SAndy Whitcroft				}
4711cf655043SAndy Whitcroft
4712000d1cc1SJoe Perches				WARN("BRACES",
4713000d1cc1SJoe Perches				     "braces {} are not necessary for single statement blocks\n" . $herectx);
4714f0a594c1SAndy Whitcroft			}
4715f0a594c1SAndy Whitcroft		}
4716f0a594c1SAndy Whitcroft
47170979ae66SJoe Perches# check for unnecessary blank lines around braces
471877b9a53aSJoe Perches		if (($line =~ /^.\s*}\s*$/ && $prevrawline =~ /^.\s*$/)) {
4719f8e58219SJoe Perches			if (CHK("BRACES",
4720f8e58219SJoe Perches				"Blank lines aren't necessary before a close brace '}'\n" . $hereprev) &&
4721f8e58219SJoe Perches			    $fix && $prevrawline =~ /^\+/) {
4722f8e58219SJoe Perches				fix_delete_line($fixlinenr - 1, $prevrawline);
4723f8e58219SJoe Perches			}
47240979ae66SJoe Perches		}
472577b9a53aSJoe Perches		if (($rawline =~ /^.\s*$/ && $prevline =~ /^..*{\s*$/)) {
4726f8e58219SJoe Perches			if (CHK("BRACES",
4727f8e58219SJoe Perches				"Blank lines aren't necessary after an open brace '{'\n" . $hereprev) &&
4728f8e58219SJoe Perches			    $fix) {
4729f8e58219SJoe Perches				fix_delete_line($fixlinenr, $rawline);
4730f8e58219SJoe Perches			}
47310979ae66SJoe Perches		}
47320979ae66SJoe Perches
47334a0df2efSAndy Whitcroft# no volatiles please
47346c72ffaaSAndy Whitcroft		my $asm_volatile = qr{\b(__asm__|asm)\s+(__volatile__|volatile)\b};
47356c72ffaaSAndy Whitcroft		if ($line =~ /\bvolatile\b/ && $line !~ /$asm_volatile/) {
4736000d1cc1SJoe Perches			WARN("VOLATILE",
4737000d1cc1SJoe Perches			     "Use of volatile is usually wrong: see Documentation/volatile-considered-harmful.txt\n" . $herecurr);
47384a0df2efSAndy Whitcroft		}
47394a0df2efSAndy Whitcroft
47405e4f6ba5SJoe Perches# Check for user-visible strings broken across lines, which breaks the ability
47415e4f6ba5SJoe Perches# to grep for the string.  Make exceptions when the previous string ends in a
47425e4f6ba5SJoe Perches# newline (multiple lines in one string constant) or '\t', '\r', ';', or '{'
47435e4f6ba5SJoe Perches# (common in inline assembly) or is a octal \123 or hexadecimal \xaf value
474433acb54aSJoe Perches		if ($line =~ /^\+\s*$String/ &&
47455e4f6ba5SJoe Perches		    $prevline =~ /"\s*$/ &&
47465e4f6ba5SJoe Perches		    $prevrawline !~ /(?:\\(?:[ntr]|[0-7]{1,3}|x[0-9a-fA-F]{1,2})|;\s*|\{\s*)"\s*$/) {
47475e4f6ba5SJoe Perches			if (WARN("SPLIT_STRING",
47485e4f6ba5SJoe Perches				 "quoted string split across lines\n" . $hereprev) &&
47495e4f6ba5SJoe Perches				     $fix &&
47505e4f6ba5SJoe Perches				     $prevrawline =~ /^\+.*"\s*$/ &&
47515e4f6ba5SJoe Perches				     $last_coalesced_string_linenr != $linenr - 1) {
47525e4f6ba5SJoe Perches				my $extracted_string = get_quoted_string($line, $rawline);
47535e4f6ba5SJoe Perches				my $comma_close = "";
47545e4f6ba5SJoe Perches				if ($rawline =~ /\Q$extracted_string\E(\s*\)\s*;\s*$|\s*,\s*)/) {
47555e4f6ba5SJoe Perches					$comma_close = $1;
47565e4f6ba5SJoe Perches				}
47575e4f6ba5SJoe Perches
47585e4f6ba5SJoe Perches				fix_delete_line($fixlinenr - 1, $prevrawline);
47595e4f6ba5SJoe Perches				fix_delete_line($fixlinenr, $rawline);
47605e4f6ba5SJoe Perches				my $fixedline = $prevrawline;
47615e4f6ba5SJoe Perches				$fixedline =~ s/"\s*$//;
47625e4f6ba5SJoe Perches				$fixedline .= substr($extracted_string, 1) . trim($comma_close);
47635e4f6ba5SJoe Perches				fix_insert_line($fixlinenr - 1, $fixedline);
47645e4f6ba5SJoe Perches				$fixedline = $rawline;
47655e4f6ba5SJoe Perches				$fixedline =~ s/\Q$extracted_string\E\Q$comma_close\E//;
47665e4f6ba5SJoe Perches				if ($fixedline !~ /\+\s*$/) {
47675e4f6ba5SJoe Perches					fix_insert_line($fixlinenr, $fixedline);
47685e4f6ba5SJoe Perches				}
47695e4f6ba5SJoe Perches				$last_coalesced_string_linenr = $linenr;
47705e4f6ba5SJoe Perches			}
47715e4f6ba5SJoe Perches		}
47725e4f6ba5SJoe Perches
47735e4f6ba5SJoe Perches# check for missing a space in a string concatenation
47745e4f6ba5SJoe Perches		if ($prevrawline =~ /[^\\]\w"$/ && $rawline =~ /^\+[\t ]+"\w/) {
47755e4f6ba5SJoe Perches			WARN('MISSING_SPACE',
47765e4f6ba5SJoe Perches			     "break quoted strings at a space character\n" . $hereprev);
47775e4f6ba5SJoe Perches		}
47785e4f6ba5SJoe Perches
47795e4f6ba5SJoe Perches# check for spaces before a quoted newline
47805e4f6ba5SJoe Perches		if ($rawline =~ /^.*\".*\s\\n/) {
47815e4f6ba5SJoe Perches			if (WARN("QUOTED_WHITESPACE_BEFORE_NEWLINE",
47825e4f6ba5SJoe Perches				 "unnecessary whitespace before a quoted newline\n" . $herecurr) &&
47835e4f6ba5SJoe Perches			    $fix) {
47845e4f6ba5SJoe Perches				$fixed[$fixlinenr] =~ s/^(\+.*\".*)\s+\\n/$1\\n/;
47855e4f6ba5SJoe Perches			}
47865e4f6ba5SJoe Perches
47875e4f6ba5SJoe Perches		}
47885e4f6ba5SJoe Perches
4789f17dba4fSJoe Perches# concatenated string without spaces between elements
479033acb54aSJoe Perches		if ($line =~ /$String[A-Z_]/ || $line =~ /[A-Za-z0-9_]$String/) {
4791f17dba4fSJoe Perches			CHK("CONCATENATED_STRING",
4792f17dba4fSJoe Perches			    "Concatenated strings should use spaces between elements\n" . $herecurr);
4793f17dba4fSJoe Perches		}
4794f17dba4fSJoe Perches
479590ad30e5SJoe Perches# uncoalesced string fragments
479633acb54aSJoe Perches		if ($line =~ /$String\s*"/) {
479790ad30e5SJoe Perches			WARN("STRING_FRAGMENTS",
479890ad30e5SJoe Perches			     "Consecutive strings are generally better as a single string\n" . $herecurr);
479990ad30e5SJoe Perches		}
480090ad30e5SJoe Perches
48015e4f6ba5SJoe Perches# check for %L{u,d,i} in strings
48025e4f6ba5SJoe Perches		my $string;
48035e4f6ba5SJoe Perches		while ($line =~ /(?:^|")([X\t]*)(?:"|$)/g) {
48045e4f6ba5SJoe Perches			$string = substr($rawline, $-[1], $+[1] - $-[1]);
48055e4f6ba5SJoe Perches			$string =~ s/%%/__/g;
48065e4f6ba5SJoe Perches			if ($string =~ /(?<!%)%L[udi]/) {
48075e4f6ba5SJoe Perches				WARN("PRINTF_L",
48085e4f6ba5SJoe Perches				     "\%Ld/%Lu are not-standard C, use %lld/%llu\n" . $herecurr);
48095e4f6ba5SJoe Perches				last;
48105e4f6ba5SJoe Perches			}
48115e4f6ba5SJoe Perches		}
48125e4f6ba5SJoe Perches
48135e4f6ba5SJoe Perches# check for line continuations in quoted strings with odd counts of "
48145e4f6ba5SJoe Perches		if ($rawline =~ /\\$/ && $rawline =~ tr/"/"/ % 2) {
48155e4f6ba5SJoe Perches			WARN("LINE_CONTINUATIONS",
48165e4f6ba5SJoe Perches			     "Avoid line continuations in quoted strings\n" . $herecurr);
48175e4f6ba5SJoe Perches		}
48185e4f6ba5SJoe Perches
481900df344fSAndy Whitcroft# warn about #if 0
4820c45dcabdSAndy Whitcroft		if ($line =~ /^.\s*\#\s*if\s+0\b/) {
4821000d1cc1SJoe Perches			CHK("REDUNDANT_CODE",
4822000d1cc1SJoe Perches			    "if this code is redundant consider removing it\n" .
4823de7d4f0eSAndy Whitcroft				$herecurr);
48244a0df2efSAndy Whitcroft		}
48254a0df2efSAndy Whitcroft
482603df4b51SAndy Whitcroft# check for needless "if (<foo>) fn(<foo>)" uses
482703df4b51SAndy Whitcroft		if ($prevline =~ /\bif\s*\(\s*($Lval)\s*\)/) {
482803df4b51SAndy Whitcroft			my $expr = '\s*\(\s*' . quotemeta($1) . '\s*\)\s*;';
482903df4b51SAndy Whitcroft			if ($line =~ /\b(kfree|usb_free_urb|debugfs_remove(?:_recursive)?)$expr/) {
483003df4b51SAndy Whitcroft				WARN('NEEDLESS_IF',
483115160f90SFabio Estevam				     "$1(NULL) is safe and this check is probably not required\n" . $hereprev);
48324c432a8fSGreg Kroah-Hartman			}
48334c432a8fSGreg Kroah-Hartman		}
4834f0a594c1SAndy Whitcroft
4835ebfdc409SJoe Perches# check for unnecessary "Out of Memory" messages
4836ebfdc409SJoe Perches		if ($line =~ /^\+.*\b$logFunctions\s*\(/ &&
4837ebfdc409SJoe Perches		    $prevline =~ /^[ \+]\s*if\s*\(\s*(\!\s*|NULL\s*==\s*)?($Lval)(\s*==\s*NULL\s*)?\s*\)/ &&
4838ebfdc409SJoe Perches		    (defined $1 || defined $3) &&
4839ebfdc409SJoe Perches		    $linenr > 3) {
4840ebfdc409SJoe Perches			my $testval = $2;
4841ebfdc409SJoe Perches			my $testline = $lines[$linenr - 3];
4842ebfdc409SJoe Perches
4843ebfdc409SJoe Perches			my ($s, $c) = ctx_statement_block($linenr - 3, $realcnt, 0);
4844ebfdc409SJoe Perches#			print("line: <$line>\nprevline: <$prevline>\ns: <$s>\nc: <$c>\n\n\n");
4845ebfdc409SJoe Perches
4846ebfdc409SJoe Perches			if ($c =~ /(?:^|\n)[ \+]\s*(?:$Type\s*)?\Q$testval\E\s*=\s*(?:\([^\)]*\)\s*)?\s*(?:devm_)?(?:[kv][czm]alloc(?:_node|_array)?\b|kstrdup|(?:dev_)?alloc_skb)/) {
4847ebfdc409SJoe Perches				WARN("OOM_MESSAGE",
4848ebfdc409SJoe Perches				     "Possible unnecessary 'out of memory' message\n" . $hereprev);
4849ebfdc409SJoe Perches			}
4850ebfdc409SJoe Perches		}
4851ebfdc409SJoe Perches
4852f78d98f6SJoe Perches# check for logging functions with KERN_<LEVEL>
4853dcaf1123SPaolo Bonzini		if ($line !~ /printk(?:_ratelimited|_once)?\s*\(/ &&
4854f78d98f6SJoe Perches		    $line =~ /\b$logFunctions\s*\(.*\b(KERN_[A-Z]+)\b/) {
4855f78d98f6SJoe Perches			my $level = $1;
4856f78d98f6SJoe Perches			if (WARN("UNNECESSARY_KERN_LEVEL",
4857f78d98f6SJoe Perches				 "Possible unnecessary $level\n" . $herecurr) &&
4858f78d98f6SJoe Perches			    $fix) {
4859f78d98f6SJoe Perches				$fixed[$fixlinenr] =~ s/\s*$level\s*//;
4860f78d98f6SJoe Perches			}
4861f78d98f6SJoe Perches		}
4862f78d98f6SJoe Perches
4863abb08a53SJoe Perches# check for mask then right shift without a parentheses
4864abb08a53SJoe Perches		if ($^V && $^V ge 5.10.0 &&
4865abb08a53SJoe Perches		    $line =~ /$LvalOrFunc\s*\&\s*($LvalOrFunc)\s*>>/ &&
4866abb08a53SJoe Perches		    $4 !~ /^\&/) { # $LvalOrFunc may be &foo, ignore if so
4867abb08a53SJoe Perches			WARN("MASK_THEN_SHIFT",
4868abb08a53SJoe Perches			     "Possible precedence defect with mask then right shift - may need parentheses\n" . $herecurr);
4869abb08a53SJoe Perches		}
4870abb08a53SJoe Perches
4871b75ac618SJoe Perches# check for pointer comparisons to NULL
4872b75ac618SJoe Perches		if ($^V && $^V ge 5.10.0) {
4873b75ac618SJoe Perches			while ($line =~ /\b$LvalOrFunc\s*(==|\!=)\s*NULL\b/g) {
4874b75ac618SJoe Perches				my $val = $1;
4875b75ac618SJoe Perches				my $equal = "!";
4876b75ac618SJoe Perches				$equal = "" if ($4 eq "!=");
4877b75ac618SJoe Perches				if (CHK("COMPARISON_TO_NULL",
4878b75ac618SJoe Perches					"Comparison to NULL could be written \"${equal}${val}\"\n" . $herecurr) &&
4879b75ac618SJoe Perches					    $fix) {
4880b75ac618SJoe Perches					$fixed[$fixlinenr] =~ s/\b\Q$val\E\s*(?:==|\!=)\s*NULL\b/$equal$val/;
4881b75ac618SJoe Perches				}
4882b75ac618SJoe Perches			}
4883b75ac618SJoe Perches		}
4884b75ac618SJoe Perches
48858716de38SJoe Perches# check for bad placement of section $InitAttribute (e.g.: __initdata)
48868716de38SJoe Perches		if ($line =~ /(\b$InitAttribute\b)/) {
48878716de38SJoe Perches			my $attr = $1;
48888716de38SJoe Perches			if ($line =~ /^\+\s*static\s+(?:const\s+)?(?:$attr\s+)?($NonptrTypeWithAttr)\s+(?:$attr\s+)?($Ident(?:\[[^]]*\])?)\s*[=;]/) {
48898716de38SJoe Perches				my $ptr = $1;
48908716de38SJoe Perches				my $var = $2;
48918716de38SJoe Perches				if ((($ptr =~ /\b(union|struct)\s+$attr\b/ &&
48928716de38SJoe Perches				      ERROR("MISPLACED_INIT",
48938716de38SJoe Perches					    "$attr should be placed after $var\n" . $herecurr)) ||
48948716de38SJoe Perches				     ($ptr !~ /\b(union|struct)\s+$attr\b/ &&
48958716de38SJoe Perches				      WARN("MISPLACED_INIT",
48968716de38SJoe Perches					   "$attr should be placed after $var\n" . $herecurr))) &&
48978716de38SJoe Perches				    $fix) {
4898194f66fcSJoe Perches					$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;
48998716de38SJoe Perches				}
49008716de38SJoe Perches			}
49018716de38SJoe Perches		}
49028716de38SJoe Perches
4903e970b884SJoe Perches# check for $InitAttributeData (ie: __initdata) with const
4904e970b884SJoe Perches		if ($line =~ /\bconst\b/ && $line =~ /($InitAttributeData)/) {
4905e970b884SJoe Perches			my $attr = $1;
4906e970b884SJoe Perches			$attr =~ /($InitAttributePrefix)(.*)/;
4907e970b884SJoe Perches			my $attr_prefix = $1;
4908e970b884SJoe Perches			my $attr_type = $2;
4909e970b884SJoe Perches			if (ERROR("INIT_ATTRIBUTE",
4910e970b884SJoe Perches				  "Use of const init definition must use ${attr_prefix}initconst\n" . $herecurr) &&
4911e970b884SJoe Perches			    $fix) {
4912194f66fcSJoe Perches				$fixed[$fixlinenr] =~
4913e970b884SJoe Perches				    s/$InitAttributeData/${attr_prefix}initconst/;
4914e970b884SJoe Perches			}
4915e970b884SJoe Perches		}
4916e970b884SJoe Perches
4917e970b884SJoe Perches# check for $InitAttributeConst (ie: __initconst) without const
4918e970b884SJoe Perches		if ($line !~ /\bconst\b/ && $line =~ /($InitAttributeConst)/) {
4919e970b884SJoe Perches			my $attr = $1;
4920e970b884SJoe Perches			if (ERROR("INIT_ATTRIBUTE",
4921e970b884SJoe Perches				  "Use of $attr requires a separate use of const\n" . $herecurr) &&
4922e970b884SJoe Perches			    $fix) {
4923194f66fcSJoe Perches				my $lead = $fixed[$fixlinenr] =~
4924e970b884SJoe Perches				    /(^\+\s*(?:static\s+))/;
4925e970b884SJoe Perches				$lead = rtrim($1);
4926e970b884SJoe Perches				$lead = "$lead " if ($lead !~ /^\+$/);
4927e970b884SJoe Perches				$lead = "${lead}const ";
4928194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/(^\+\s*(?:static\s+))/$lead/;
4929e970b884SJoe Perches			}
4930e970b884SJoe Perches		}
4931e970b884SJoe Perches
4932c17893c7SJoe Perches# check for __read_mostly with const non-pointer (should just be const)
4933c17893c7SJoe Perches		if ($line =~ /\b__read_mostly\b/ &&
4934c17893c7SJoe Perches		    $line =~ /($Type)\s*$Ident/ && $1 !~ /\*\s*$/ && $1 =~ /\bconst\b/) {
4935c17893c7SJoe Perches			if (ERROR("CONST_READ_MOSTLY",
4936c17893c7SJoe Perches				  "Invalid use of __read_mostly with const type\n" . $herecurr) &&
4937c17893c7SJoe Perches			    $fix) {
4938c17893c7SJoe Perches				$fixed[$fixlinenr] =~ s/\s+__read_mostly\b//;
4939c17893c7SJoe Perches			}
4940c17893c7SJoe Perches		}
4941c17893c7SJoe Perches
4942fbdb8138SJoe Perches# don't use __constant_<foo> functions outside of include/uapi/
4943fbdb8138SJoe Perches		if ($realfile !~ m@^include/uapi/@ &&
4944fbdb8138SJoe Perches		    $line =~ /(__constant_(?:htons|ntohs|[bl]e(?:16|32|64)_to_cpu|cpu_to_[bl]e(?:16|32|64)))\s*\(/) {
4945fbdb8138SJoe Perches			my $constant_func = $1;
4946fbdb8138SJoe Perches			my $func = $constant_func;
4947fbdb8138SJoe Perches			$func =~ s/^__constant_//;
4948fbdb8138SJoe Perches			if (WARN("CONSTANT_CONVERSION",
4949fbdb8138SJoe Perches				 "$constant_func should be $func\n" . $herecurr) &&
4950fbdb8138SJoe Perches			    $fix) {
4951194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\b$constant_func\b/$func/g;
4952fbdb8138SJoe Perches			}
4953fbdb8138SJoe Perches		}
4954fbdb8138SJoe Perches
49551a15a250SPatrick Pannuto# prefer usleep_range over udelay
495637581c28SBruce Allan		if ($line =~ /\budelay\s*\(\s*(\d+)\s*\)/) {
495743c1d77cSJoe Perches			my $delay = $1;
49581a15a250SPatrick Pannuto			# ignore udelay's < 10, however
495943c1d77cSJoe Perches			if (! ($delay < 10) ) {
4960000d1cc1SJoe Perches				CHK("USLEEP_RANGE",
496143c1d77cSJoe Perches				    "usleep_range is preferred over udelay; see Documentation/timers/timers-howto.txt\n" . $herecurr);
496243c1d77cSJoe Perches			}
496343c1d77cSJoe Perches			if ($delay > 2000) {
496443c1d77cSJoe Perches				WARN("LONG_UDELAY",
496543c1d77cSJoe Perches				     "long udelay - prefer mdelay; see arch/arm/include/asm/delay.h\n" . $herecurr);
49661a15a250SPatrick Pannuto			}
49671a15a250SPatrick Pannuto		}
49681a15a250SPatrick Pannuto
496909ef8725SPatrick Pannuto# warn about unexpectedly long msleep's
497009ef8725SPatrick Pannuto		if ($line =~ /\bmsleep\s*\((\d+)\);/) {
497109ef8725SPatrick Pannuto			if ($1 < 20) {
4972000d1cc1SJoe Perches				WARN("MSLEEP",
497343c1d77cSJoe Perches				     "msleep < 20ms can sleep for up to 20ms; see Documentation/timers/timers-howto.txt\n" . $herecurr);
497409ef8725SPatrick Pannuto			}
497509ef8725SPatrick Pannuto		}
497609ef8725SPatrick Pannuto
497736ec1939SJoe Perches# check for comparisons of jiffies
497836ec1939SJoe Perches		if ($line =~ /\bjiffies\s*$Compare|$Compare\s*jiffies\b/) {
497936ec1939SJoe Perches			WARN("JIFFIES_COMPARISON",
498036ec1939SJoe Perches			     "Comparing jiffies is almost always wrong; prefer time_after, time_before and friends\n" . $herecurr);
498136ec1939SJoe Perches		}
498236ec1939SJoe Perches
49839d7a34a5SJoe Perches# check for comparisons of get_jiffies_64()
49849d7a34a5SJoe Perches		if ($line =~ /\bget_jiffies_64\s*\(\s*\)\s*$Compare|$Compare\s*get_jiffies_64\s*\(\s*\)/) {
49859d7a34a5SJoe Perches			WARN("JIFFIES_COMPARISON",
49869d7a34a5SJoe Perches			     "Comparing get_jiffies_64() is almost always wrong; prefer time_after64, time_before64 and friends\n" . $herecurr);
49879d7a34a5SJoe Perches		}
49889d7a34a5SJoe Perches
498900df344fSAndy Whitcroft# warn about #ifdefs in C files
4990c45dcabdSAndy Whitcroft#		if ($line =~ /^.\s*\#\s*if(|n)def/ && ($realfile =~ /\.c$/)) {
499100df344fSAndy Whitcroft#			print "#ifdef in C files should be avoided\n";
499200df344fSAndy Whitcroft#			print "$herecurr";
499300df344fSAndy Whitcroft#			$clean = 0;
499400df344fSAndy Whitcroft#		}
499500df344fSAndy Whitcroft
499622f2a2efSAndy Whitcroft# warn about spacing in #ifdefs
4997c45dcabdSAndy Whitcroft		if ($line =~ /^.\s*\#\s*(ifdef|ifndef|elif)\s\s+/) {
49983705ce5bSJoe Perches			if (ERROR("SPACING",
49993705ce5bSJoe Perches				  "exactly one space required after that #$1\n" . $herecurr) &&
50003705ce5bSJoe Perches			    $fix) {
5001194f66fcSJoe Perches				$fixed[$fixlinenr] =~
50023705ce5bSJoe Perches				    s/^(.\s*\#\s*(ifdef|ifndef|elif))\s{2,}/$1 /;
50033705ce5bSJoe Perches			}
50043705ce5bSJoe Perches
500522f2a2efSAndy Whitcroft		}
500622f2a2efSAndy Whitcroft
50074a0df2efSAndy Whitcroft# check for spinlock_t definitions without a comment.
5008171ae1a4SAndy Whitcroft		if ($line =~ /^.\s*(struct\s+mutex|spinlock_t)\s+\S+;/ ||
5009171ae1a4SAndy Whitcroft		    $line =~ /^.\s*(DEFINE_MUTEX)\s*\(/) {
50104a0df2efSAndy Whitcroft			my $which = $1;
50114a0df2efSAndy Whitcroft			if (!ctx_has_comment($first_line, $linenr)) {
5012000d1cc1SJoe Perches				CHK("UNCOMMENTED_DEFINITION",
5013000d1cc1SJoe Perches				    "$1 definition without comment\n" . $herecurr);
50144a0df2efSAndy Whitcroft			}
50154a0df2efSAndy Whitcroft		}
50164a0df2efSAndy Whitcroft# check for memory barriers without a comment.
50174a0df2efSAndy Whitcroft		if ($line =~ /\b(mb|rmb|wmb|read_barrier_depends|smp_mb|smp_rmb|smp_wmb|smp_read_barrier_depends)\(/) {
50184a0df2efSAndy Whitcroft			if (!ctx_has_comment($first_line, $linenr)) {
5019c1fd7bb9SJoe Perches				WARN("MEMORY_BARRIER",
5020000d1cc1SJoe Perches				     "memory barrier without comment\n" . $herecurr);
50214a0df2efSAndy Whitcroft			}
50224a0df2efSAndy Whitcroft		}
50233ad81779SPaul E. McKenney
5024cb426e99SJoe Perches# check for waitqueue_active without a comment.
5025cb426e99SJoe Perches		if ($line =~ /\bwaitqueue_active\s*\(/) {
5026cb426e99SJoe Perches			if (!ctx_has_comment($first_line, $linenr)) {
5027cb426e99SJoe Perches				WARN("WAITQUEUE_ACTIVE",
5028cb426e99SJoe Perches				     "waitqueue_active without comment\n" . $herecurr);
5029cb426e99SJoe Perches			}
5030cb426e99SJoe Perches		}
50313ad81779SPaul E. McKenney
50323ad81779SPaul E. McKenney# Check for expedited grace periods that interrupt non-idle non-nohz
50333ad81779SPaul E. McKenney# online CPUs.  These expedited can therefore degrade real-time response
50343ad81779SPaul E. McKenney# if used carelessly, and should be avoided where not absolutely
50353ad81779SPaul E. McKenney# needed.  It is always OK to use synchronize_rcu_expedited() and
50363ad81779SPaul E. McKenney# synchronize_sched_expedited() at boot time (before real-time applications
50373ad81779SPaul E. McKenney# start) and in error situations where real-time response is compromised in
50383ad81779SPaul E. McKenney# any case.  Note that synchronize_srcu_expedited() does -not- interrupt
50393ad81779SPaul E. McKenney# other CPUs, so don't warn on uses of synchronize_srcu_expedited().
50403ad81779SPaul E. McKenney# Of course, nothing comes for free, and srcu_read_lock() and
50413ad81779SPaul E. McKenney# srcu_read_unlock() do contain full memory barriers in payment for
50423ad81779SPaul E. McKenney# synchronize_srcu_expedited() non-interruption properties.
50433ad81779SPaul E. McKenney		if ($line =~ /\b(synchronize_rcu_expedited|synchronize_sched_expedited)\(/) {
50443ad81779SPaul E. McKenney			WARN("EXPEDITED_RCU_GRACE_PERIOD",
50453ad81779SPaul E. McKenney			     "expedited RCU grace periods should be avoided where they can degrade real-time response\n" . $herecurr);
50463ad81779SPaul E. McKenney
50473ad81779SPaul E. McKenney		}
50483ad81779SPaul E. McKenney
50494a0df2efSAndy Whitcroft# check of hardware specific defines
5050c45dcabdSAndy Whitcroft		if ($line =~ m@^.\s*\#\s*if.*\b(__i386__|__powerpc64__|__sun__|__s390x__)\b@ && $realfile !~ m@include/asm-@) {
5051000d1cc1SJoe Perches			CHK("ARCH_DEFINES",
5052000d1cc1SJoe Perches			    "architecture specific defines should be avoided\n" .  $herecurr);
50530a920b5bSAndy Whitcroft		}
5054653d4876SAndy Whitcroft
5055d4977c78STobias Klauser# Check that the storage class is at the beginning of a declaration
5056d4977c78STobias Klauser		if ($line =~ /\b$Storage\b/ && $line !~ /^.\s*$Storage\b/) {
5057000d1cc1SJoe Perches			WARN("STORAGE_CLASS",
5058000d1cc1SJoe Perches			     "storage class should be at the beginning of the declaration\n" . $herecurr)
5059d4977c78STobias Klauser		}
5060d4977c78STobias Klauser
5061de7d4f0eSAndy Whitcroft# check the location of the inline attribute, that it is between
5062de7d4f0eSAndy Whitcroft# storage class and type.
50639c0ca6f9SAndy Whitcroft		if ($line =~ /\b$Type\s+$Inline\b/ ||
50649c0ca6f9SAndy Whitcroft		    $line =~ /\b$Inline\s+$Storage\b/) {
5065000d1cc1SJoe Perches			ERROR("INLINE_LOCATION",
5066000d1cc1SJoe Perches			      "inline keyword should sit between storage class and type\n" . $herecurr);
5067de7d4f0eSAndy Whitcroft		}
5068de7d4f0eSAndy Whitcroft
50698905a67cSAndy Whitcroft# Check for __inline__ and __inline, prefer inline
50702b7ab453SJoe Perches		if ($realfile !~ m@\binclude/uapi/@ &&
50712b7ab453SJoe Perches		    $line =~ /\b(__inline__|__inline)\b/) {
5072d5e616fcSJoe Perches			if (WARN("INLINE",
5073d5e616fcSJoe Perches				 "plain inline is preferred over $1\n" . $herecurr) &&
5074d5e616fcSJoe Perches			    $fix) {
5075194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\b(__inline__|__inline)\b/inline/;
5076d5e616fcSJoe Perches
5077d5e616fcSJoe Perches			}
50788905a67cSAndy Whitcroft		}
50798905a67cSAndy Whitcroft
50803d130fd0SJoe Perches# Check for __attribute__ packed, prefer __packed
50812b7ab453SJoe Perches		if ($realfile !~ m@\binclude/uapi/@ &&
50822b7ab453SJoe Perches		    $line =~ /\b__attribute__\s*\(\s*\(.*\bpacked\b/) {
5083000d1cc1SJoe Perches			WARN("PREFER_PACKED",
5084000d1cc1SJoe Perches			     "__packed is preferred over __attribute__((packed))\n" . $herecurr);
50853d130fd0SJoe Perches		}
50863d130fd0SJoe Perches
508739b7e287SJoe Perches# Check for __attribute__ aligned, prefer __aligned
50882b7ab453SJoe Perches		if ($realfile !~ m@\binclude/uapi/@ &&
50892b7ab453SJoe Perches		    $line =~ /\b__attribute__\s*\(\s*\(.*aligned/) {
5090000d1cc1SJoe Perches			WARN("PREFER_ALIGNED",
5091000d1cc1SJoe Perches			     "__aligned(size) is preferred over __attribute__((aligned(size)))\n" . $herecurr);
509239b7e287SJoe Perches		}
509339b7e287SJoe Perches
50945f14d3bdSJoe Perches# Check for __attribute__ format(printf, prefer __printf
50952b7ab453SJoe Perches		if ($realfile !~ m@\binclude/uapi/@ &&
50962b7ab453SJoe Perches		    $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf/) {
5097d5e616fcSJoe Perches			if (WARN("PREFER_PRINTF",
5098d5e616fcSJoe Perches				 "__printf(string-index, first-to-check) is preferred over __attribute__((format(printf, string-index, first-to-check)))\n" . $herecurr) &&
5099d5e616fcSJoe Perches			    $fix) {
5100194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*printf\s*,\s*(.*)\)\s*\)\s*\)/"__printf(" . trim($1) . ")"/ex;
5101d5e616fcSJoe Perches
5102d5e616fcSJoe Perches			}
51035f14d3bdSJoe Perches		}
51045f14d3bdSJoe Perches
51056061d949SJoe Perches# Check for __attribute__ format(scanf, prefer __scanf
51062b7ab453SJoe Perches		if ($realfile !~ m@\binclude/uapi/@ &&
51072b7ab453SJoe Perches		    $line =~ /\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\b/) {
5108d5e616fcSJoe Perches			if (WARN("PREFER_SCANF",
5109d5e616fcSJoe Perches				 "__scanf(string-index, first-to-check) is preferred over __attribute__((format(scanf, string-index, first-to-check)))\n" . $herecurr) &&
5110d5e616fcSJoe Perches			    $fix) {
5111194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\b__attribute__\s*\(\s*\(\s*format\s*\(\s*scanf\s*,\s*(.*)\)\s*\)\s*\)/"__scanf(" . trim($1) . ")"/ex;
5112d5e616fcSJoe Perches			}
51136061d949SJoe Perches		}
51146061d949SJoe Perches
5115619a908aSJoe Perches# Check for __attribute__ weak, or __weak declarations (may have link issues)
5116619a908aSJoe Perches		if ($^V && $^V ge 5.10.0 &&
5117619a908aSJoe Perches		    $line =~ /(?:$Declare|$DeclareMisordered)\s*$Ident\s*$balanced_parens\s*(?:$Attribute)?\s*;/ &&
5118619a908aSJoe Perches		    ($line =~ /\b__attribute__\s*\(\s*\(.*\bweak\b/ ||
5119619a908aSJoe Perches		     $line =~ /\b__weak\b/)) {
5120619a908aSJoe Perches			ERROR("WEAK_DECLARATION",
5121619a908aSJoe Perches			      "Using weak declarations can have unintended link defects\n" . $herecurr);
5122619a908aSJoe Perches		}
5123619a908aSJoe Perches
5124e6176fa4SJoe Perches# check for c99 types like uint8_t used outside of uapi/
5125e6176fa4SJoe Perches		if ($realfile !~ m@\binclude/uapi/@ &&
5126e6176fa4SJoe Perches		    $line =~ /\b($Declare)\s*$Ident\s*[=;,\[]/) {
5127e6176fa4SJoe Perches			my $type = $1;
5128e6176fa4SJoe Perches			if ($type =~ /\b($typeC99Typedefs)\b/) {
5129e6176fa4SJoe Perches				$type = $1;
5130e6176fa4SJoe Perches				my $kernel_type = 'u';
5131e6176fa4SJoe Perches				$kernel_type = 's' if ($type =~ /^_*[si]/);
5132e6176fa4SJoe Perches				$type =~ /(\d+)/;
5133e6176fa4SJoe Perches				$kernel_type .= $1;
5134e6176fa4SJoe Perches				if (CHK("PREFER_KERNEL_TYPES",
5135e6176fa4SJoe Perches					"Prefer kernel type '$kernel_type' over '$type'\n" . $herecurr) &&
5136e6176fa4SJoe Perches				    $fix) {
5137e6176fa4SJoe Perches					$fixed[$fixlinenr] =~ s/\b$type\b/$kernel_type/;
5138e6176fa4SJoe Perches				}
5139e6176fa4SJoe Perches			}
5140e6176fa4SJoe Perches		}
5141e6176fa4SJoe Perches
51428f53a9b8SJoe Perches# check for sizeof(&)
51438f53a9b8SJoe Perches		if ($line =~ /\bsizeof\s*\(\s*\&/) {
5144000d1cc1SJoe Perches			WARN("SIZEOF_ADDRESS",
5145000d1cc1SJoe Perches			     "sizeof(& should be avoided\n" . $herecurr);
51468f53a9b8SJoe Perches		}
51478f53a9b8SJoe Perches
514866c80b60SJoe Perches# check for sizeof without parenthesis
514966c80b60SJoe Perches		if ($line =~ /\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/) {
5150d5e616fcSJoe Perches			if (WARN("SIZEOF_PARENTHESIS",
5151d5e616fcSJoe Perches				 "sizeof $1 should be sizeof($1)\n" . $herecurr) &&
5152d5e616fcSJoe Perches			    $fix) {
5153194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\bsizeof\s+((?:\*\s*|)$Lval|$Type(?:\s+$Lval|))/"sizeof(" . trim($1) . ")"/ex;
5154d5e616fcSJoe Perches			}
515566c80b60SJoe Perches		}
515666c80b60SJoe Perches
515788982feaSJoe Perches# check for struct spinlock declarations
515888982feaSJoe Perches		if ($line =~ /^.\s*\bstruct\s+spinlock\s+\w+\s*;/) {
515988982feaSJoe Perches			WARN("USE_SPINLOCK_T",
516088982feaSJoe Perches			     "struct spinlock should be spinlock_t\n" . $herecurr);
516188982feaSJoe Perches		}
516288982feaSJoe Perches
5163a6962d72SJoe Perches# check for seq_printf uses that could be seq_puts
516406668727SJoe Perches		if ($sline =~ /\bseq_printf\s*\(.*"\s*\)\s*;\s*$/) {
5165a6962d72SJoe Perches			my $fmt = get_quoted_string($line, $rawline);
5166caac1d5fSHeba Aamer			$fmt =~ s/%%//g;
5167caac1d5fSHeba Aamer			if ($fmt !~ /%/) {
5168d5e616fcSJoe Perches				if (WARN("PREFER_SEQ_PUTS",
5169d5e616fcSJoe Perches					 "Prefer seq_puts to seq_printf\n" . $herecurr) &&
5170d5e616fcSJoe Perches				    $fix) {
5171194f66fcSJoe Perches					$fixed[$fixlinenr] =~ s/\bseq_printf\b/seq_puts/;
5172d5e616fcSJoe Perches				}
5173a6962d72SJoe Perches			}
5174a6962d72SJoe Perches		}
5175a6962d72SJoe Perches
5176554e165cSAndy Whitcroft# Check for misused memsets
5177d1fe9c09SJoe Perches		if ($^V && $^V ge 5.10.0 &&
5178d1fe9c09SJoe Perches		    defined $stat &&
51799e20a853SMateusz Kulikowski		    $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*$FuncArg\s*\)/) {
5180554e165cSAndy Whitcroft
5181d7c76ba7SJoe Perches			my $ms_addr = $2;
5182d1fe9c09SJoe Perches			my $ms_val = $7;
5183d1fe9c09SJoe Perches			my $ms_size = $12;
5184d7c76ba7SJoe Perches
5185554e165cSAndy Whitcroft			if ($ms_size =~ /^(0x|)0$/i) {
5186554e165cSAndy Whitcroft				ERROR("MEMSET",
5187d7c76ba7SJoe Perches				      "memset to 0's uses 0 as the 2nd argument, not the 3rd\n" . "$here\n$stat\n");
5188554e165cSAndy Whitcroft			} elsif ($ms_size =~ /^(0x|)1$/i) {
5189554e165cSAndy Whitcroft				WARN("MEMSET",
5190d7c76ba7SJoe Perches				     "single byte memset is suspicious. Swapped 2nd/3rd argument?\n" . "$here\n$stat\n");
5191d7c76ba7SJoe Perches			}
5192d7c76ba7SJoe Perches		}
5193d7c76ba7SJoe Perches
519498a9bba5SJoe Perches# Check for memcpy(foo, bar, ETH_ALEN) that could be ether_addr_copy(foo, bar)
519598a9bba5SJoe Perches		if ($^V && $^V ge 5.10.0 &&
519610895d2cSMateusz Kulikowski		    defined $stat &&
519710895d2cSMateusz Kulikowski		    $stat =~ /^\+(?:.*?)\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) {
519898a9bba5SJoe Perches			if (WARN("PREFER_ETHER_ADDR_COPY",
519910895d2cSMateusz Kulikowski				 "Prefer ether_addr_copy() over memcpy() if the Ethernet addresses are __aligned(2)\n" . "$here\n$stat\n") &&
520098a9bba5SJoe Perches			    $fix) {
5201194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\bmemcpy\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/ether_addr_copy($2, $7)/;
520298a9bba5SJoe Perches			}
520398a9bba5SJoe Perches		}
520498a9bba5SJoe Perches
5205b6117d17SMateusz Kulikowski# Check for memcmp(foo, bar, ETH_ALEN) that could be ether_addr_equal*(foo, bar)
5206b6117d17SMateusz Kulikowski		if ($^V && $^V ge 5.10.0 &&
5207b6117d17SMateusz Kulikowski		    defined $stat &&
5208b6117d17SMateusz Kulikowski		    $stat =~ /^\+(?:.*?)\bmemcmp\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) {
5209b6117d17SMateusz Kulikowski			WARN("PREFER_ETHER_ADDR_EQUAL",
5210b6117d17SMateusz Kulikowski			     "Prefer ether_addr_equal() or ether_addr_equal_unaligned() over memcmp()\n" . "$here\n$stat\n")
5211b6117d17SMateusz Kulikowski		}
5212b6117d17SMateusz Kulikowski
52138617cd09SMateusz Kulikowski# check for memset(foo, 0x0, ETH_ALEN) that could be eth_zero_addr
52148617cd09SMateusz Kulikowski# check for memset(foo, 0xFF, ETH_ALEN) that could be eth_broadcast_addr
52158617cd09SMateusz Kulikowski		if ($^V && $^V ge 5.10.0 &&
52168617cd09SMateusz Kulikowski		    defined $stat &&
52178617cd09SMateusz Kulikowski		    $stat =~ /^\+(?:.*?)\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\,\s*ETH_ALEN\s*\)/) {
52188617cd09SMateusz Kulikowski
52198617cd09SMateusz Kulikowski			my $ms_val = $7;
52208617cd09SMateusz Kulikowski
52218617cd09SMateusz Kulikowski			if ($ms_val =~ /^(?:0x|)0+$/i) {
52228617cd09SMateusz Kulikowski				if (WARN("PREFER_ETH_ZERO_ADDR",
52238617cd09SMateusz Kulikowski					 "Prefer eth_zero_addr over memset()\n" . "$here\n$stat\n") &&
52248617cd09SMateusz Kulikowski				    $fix) {
52258617cd09SMateusz Kulikowski					$fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_zero_addr($2)/;
52268617cd09SMateusz Kulikowski				}
52278617cd09SMateusz Kulikowski			} elsif ($ms_val =~ /^(?:0xff|255)$/i) {
52288617cd09SMateusz Kulikowski				if (WARN("PREFER_ETH_BROADCAST_ADDR",
52298617cd09SMateusz Kulikowski					 "Prefer eth_broadcast_addr() over memset()\n" . "$here\n$stat\n") &&
52308617cd09SMateusz Kulikowski				    $fix) {
52318617cd09SMateusz Kulikowski					$fixed[$fixlinenr] =~ s/\bmemset\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*,\s*ETH_ALEN\s*\)/eth_broadcast_addr($2)/;
52328617cd09SMateusz Kulikowski				}
52338617cd09SMateusz Kulikowski			}
52348617cd09SMateusz Kulikowski		}
52358617cd09SMateusz Kulikowski
5236d7c76ba7SJoe Perches# typecasts on min/max could be min_t/max_t
5237d1fe9c09SJoe Perches		if ($^V && $^V ge 5.10.0 &&
5238d1fe9c09SJoe Perches		    defined $stat &&
5239d7c76ba7SJoe Perches		    $stat =~ /^\+(?:.*?)\b(min|max)\s*\(\s*$FuncArg\s*,\s*$FuncArg\s*\)/) {
5240d1fe9c09SJoe Perches			if (defined $2 || defined $7) {
5241d7c76ba7SJoe Perches				my $call = $1;
5242d7c76ba7SJoe Perches				my $cast1 = deparenthesize($2);
5243d7c76ba7SJoe Perches				my $arg1 = $3;
5244d1fe9c09SJoe Perches				my $cast2 = deparenthesize($7);
5245d1fe9c09SJoe Perches				my $arg2 = $8;
5246d7c76ba7SJoe Perches				my $cast;
5247d7c76ba7SJoe Perches
5248d1fe9c09SJoe Perches				if ($cast1 ne "" && $cast2 ne "" && $cast1 ne $cast2) {
5249d7c76ba7SJoe Perches					$cast = "$cast1 or $cast2";
5250d7c76ba7SJoe Perches				} elsif ($cast1 ne "") {
5251d7c76ba7SJoe Perches					$cast = $cast1;
5252d7c76ba7SJoe Perches				} else {
5253d7c76ba7SJoe Perches					$cast = $cast2;
5254d7c76ba7SJoe Perches				}
5255d7c76ba7SJoe Perches				WARN("MINMAX",
5256d7c76ba7SJoe Perches				     "$call() should probably be ${call}_t($cast, $arg1, $arg2)\n" . "$here\n$stat\n");
5257554e165cSAndy Whitcroft			}
5258554e165cSAndy Whitcroft		}
5259554e165cSAndy Whitcroft
52604a273195SJoe Perches# check usleep_range arguments
52614a273195SJoe Perches		if ($^V && $^V ge 5.10.0 &&
52624a273195SJoe Perches		    defined $stat &&
52634a273195SJoe Perches		    $stat =~ /^\+(?:.*?)\busleep_range\s*\(\s*($FuncArg)\s*,\s*($FuncArg)\s*\)/) {
52644a273195SJoe Perches			my $min = $1;
52654a273195SJoe Perches			my $max = $7;
52664a273195SJoe Perches			if ($min eq $max) {
52674a273195SJoe Perches				WARN("USLEEP_RANGE",
52684a273195SJoe Perches				     "usleep_range should not use min == max args; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n");
52694a273195SJoe Perches			} elsif ($min =~ /^\d+$/ && $max =~ /^\d+$/ &&
52704a273195SJoe Perches				 $min > $max) {
52714a273195SJoe Perches				WARN("USLEEP_RANGE",
52724a273195SJoe Perches				     "usleep_range args reversed, use min then max; see Documentation/timers/timers-howto.txt\n" . "$here\n$stat\n");
52734a273195SJoe Perches			}
52744a273195SJoe Perches		}
52754a273195SJoe Perches
5276823b794cSJoe Perches# check for naked sscanf
5277823b794cSJoe Perches		if ($^V && $^V ge 5.10.0 &&
5278823b794cSJoe Perches		    defined $stat &&
52796c8bd707SJoe Perches		    $line =~ /\bsscanf\b/ &&
5280823b794cSJoe Perches		    ($stat !~ /$Ident\s*=\s*sscanf\s*$balanced_parens/ &&
5281823b794cSJoe Perches		     $stat !~ /\bsscanf\s*$balanced_parens\s*(?:$Compare)/ &&
5282823b794cSJoe Perches		     $stat !~ /(?:$Compare)\s*\bsscanf\s*$balanced_parens/)) {
5283823b794cSJoe Perches			my $lc = $stat =~ tr@\n@@;
5284823b794cSJoe Perches			$lc = $lc + $linenr;
5285823b794cSJoe Perches			my $stat_real = raw_line($linenr, 0);
5286823b794cSJoe Perches		        for (my $count = $linenr + 1; $count <= $lc; $count++) {
5287823b794cSJoe Perches				$stat_real = $stat_real . "\n" . raw_line($count, 0);
5288823b794cSJoe Perches			}
5289823b794cSJoe Perches			WARN("NAKED_SSCANF",
5290823b794cSJoe Perches			     "unchecked sscanf return value\n" . "$here\n$stat_real\n");
5291823b794cSJoe Perches		}
5292823b794cSJoe Perches
5293afc819abSJoe Perches# check for simple sscanf that should be kstrto<foo>
5294afc819abSJoe Perches		if ($^V && $^V ge 5.10.0 &&
5295afc819abSJoe Perches		    defined $stat &&
5296afc819abSJoe Perches		    $line =~ /\bsscanf\b/) {
5297afc819abSJoe Perches			my $lc = $stat =~ tr@\n@@;
5298afc819abSJoe Perches			$lc = $lc + $linenr;
5299afc819abSJoe Perches			my $stat_real = raw_line($linenr, 0);
5300afc819abSJoe Perches		        for (my $count = $linenr + 1; $count <= $lc; $count++) {
5301afc819abSJoe Perches				$stat_real = $stat_real . "\n" . raw_line($count, 0);
5302afc819abSJoe Perches			}
5303afc819abSJoe Perches			if ($stat_real =~ /\bsscanf\b\s*\(\s*$FuncArg\s*,\s*("[^"]+")/) {
5304afc819abSJoe Perches				my $format = $6;
5305afc819abSJoe Perches				my $count = $format =~ tr@%@%@;
5306afc819abSJoe Perches				if ($count == 1 &&
5307afc819abSJoe Perches				    $format =~ /^"\%(?i:ll[udxi]|[udxi]ll|ll|[hl]h?[udxi]|[udxi][hl]h?|[hl]h?|[udxi])"$/) {
5308afc819abSJoe Perches					WARN("SSCANF_TO_KSTRTO",
5309afc819abSJoe Perches					     "Prefer kstrto<type> to single variable sscanf\n" . "$here\n$stat_real\n");
5310afc819abSJoe Perches				}
5311afc819abSJoe Perches			}
5312afc819abSJoe Perches		}
5313afc819abSJoe Perches
531470dc8a48SJoe Perches# check for new externs in .h files.
531570dc8a48SJoe Perches		if ($realfile =~ /\.h$/ &&
531670dc8a48SJoe Perches		    $line =~ /^\+\s*(extern\s+)$Type\s*$Ident\s*\(/s) {
5317d1d85780SJoe Perches			if (CHK("AVOID_EXTERNS",
531870dc8a48SJoe Perches				"extern prototypes should be avoided in .h files\n" . $herecurr) &&
531970dc8a48SJoe Perches			    $fix) {
5320194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/(.*)\bextern\b\s*(.*)/$1$2/;
532170dc8a48SJoe Perches			}
532270dc8a48SJoe Perches		}
532370dc8a48SJoe Perches
5324de7d4f0eSAndy Whitcroft# check for new externs in .c files.
5325171ae1a4SAndy Whitcroft		if ($realfile =~ /\.c$/ && defined $stat &&
5326c45dcabdSAndy Whitcroft		    $stat =~ /^.\s*(?:extern\s+)?$Type\s+($Ident)(\s*)\(/s)
5327171ae1a4SAndy Whitcroft		{
5328c45dcabdSAndy Whitcroft			my $function_name = $1;
5329c45dcabdSAndy Whitcroft			my $paren_space = $2;
5330171ae1a4SAndy Whitcroft
5331171ae1a4SAndy Whitcroft			my $s = $stat;
5332171ae1a4SAndy Whitcroft			if (defined $cond) {
5333171ae1a4SAndy Whitcroft				substr($s, 0, length($cond), '');
5334171ae1a4SAndy Whitcroft			}
5335c45dcabdSAndy Whitcroft			if ($s =~ /^\s*;/ &&
5336c45dcabdSAndy Whitcroft			    $function_name ne 'uninitialized_var')
5337c45dcabdSAndy Whitcroft			{
5338000d1cc1SJoe Perches				WARN("AVOID_EXTERNS",
5339000d1cc1SJoe Perches				     "externs should be avoided in .c files\n" .  $herecurr);
5340de7d4f0eSAndy Whitcroft			}
5341de7d4f0eSAndy Whitcroft
5342171ae1a4SAndy Whitcroft			if ($paren_space =~ /\n/) {
5343000d1cc1SJoe Perches				WARN("FUNCTION_ARGUMENTS",
5344000d1cc1SJoe Perches				     "arguments for function declarations should follow identifier\n" . $herecurr);
5345171ae1a4SAndy Whitcroft			}
53469c9ba34eSAndy Whitcroft
53479c9ba34eSAndy Whitcroft		} elsif ($realfile =~ /\.c$/ && defined $stat &&
53489c9ba34eSAndy Whitcroft		    $stat =~ /^.\s*extern\s+/)
53499c9ba34eSAndy Whitcroft		{
5350000d1cc1SJoe Perches			WARN("AVOID_EXTERNS",
5351000d1cc1SJoe Perches			     "externs should be avoided in .c files\n" .  $herecurr);
5352171ae1a4SAndy Whitcroft		}
5353171ae1a4SAndy Whitcroft
5354de7d4f0eSAndy Whitcroft# checks for new __setup's
5355de7d4f0eSAndy Whitcroft		if ($rawline =~ /\b__setup\("([^"]*)"/) {
5356de7d4f0eSAndy Whitcroft			my $name = $1;
5357de7d4f0eSAndy Whitcroft
5358de7d4f0eSAndy Whitcroft			if (!grep(/$name/, @setup_docs)) {
5359000d1cc1SJoe Perches				CHK("UNDOCUMENTED_SETUP",
5360000d1cc1SJoe Perches				    "__setup appears un-documented -- check Documentation/kernel-parameters.txt\n" . $herecurr);
5361de7d4f0eSAndy Whitcroft			}
5362653d4876SAndy Whitcroft		}
53639c0ca6f9SAndy Whitcroft
53649c0ca6f9SAndy Whitcroft# check for pointless casting of kmalloc return
5365caf2a54fSJoe Perches		if ($line =~ /\*\s*\)\s*[kv][czm]alloc(_node){0,1}\b/) {
5366000d1cc1SJoe Perches			WARN("UNNECESSARY_CASTS",
5367000d1cc1SJoe Perches			     "unnecessary cast may hide bugs, see http://c-faq.com/malloc/mallocnocast.html\n" . $herecurr);
53689c0ca6f9SAndy Whitcroft		}
536913214adfSAndy Whitcroft
5370a640d25cSJoe Perches# alloc style
5371a640d25cSJoe Perches# p = alloc(sizeof(struct foo), ...) should be p = alloc(sizeof(*p), ...)
5372a640d25cSJoe Perches		if ($^V && $^V ge 5.10.0 &&
5373a640d25cSJoe Perches		    $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*([kv][mz]alloc(?:_node)?)\s*\(\s*(sizeof\s*\(\s*struct\s+$Lval\s*\))/) {
5374a640d25cSJoe Perches			CHK("ALLOC_SIZEOF_STRUCT",
5375a640d25cSJoe Perches			    "Prefer $3(sizeof(*$1)...) over $3($4...)\n" . $herecurr);
5376a640d25cSJoe Perches		}
5377a640d25cSJoe Perches
537860a55369SJoe Perches# check for k[mz]alloc with multiplies that could be kmalloc_array/kcalloc
537960a55369SJoe Perches		if ($^V && $^V ge 5.10.0 &&
5380e367455aSJoe Perches		    $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*(k[mz]alloc)\s*\(\s*($FuncArg)\s*\*\s*($FuncArg)\s*,/) {
538160a55369SJoe Perches			my $oldfunc = $3;
538260a55369SJoe Perches			my $a1 = $4;
538360a55369SJoe Perches			my $a2 = $10;
538460a55369SJoe Perches			my $newfunc = "kmalloc_array";
538560a55369SJoe Perches			$newfunc = "kcalloc" if ($oldfunc eq "kzalloc");
538660a55369SJoe Perches			my $r1 = $a1;
538760a55369SJoe Perches			my $r2 = $a2;
538860a55369SJoe Perches			if ($a1 =~ /^sizeof\s*\S/) {
538960a55369SJoe Perches				$r1 = $a2;
539060a55369SJoe Perches				$r2 = $a1;
539160a55369SJoe Perches			}
5392e367455aSJoe Perches			if ($r1 !~ /^sizeof\b/ && $r2 =~ /^sizeof\s*\S/ &&
5393e367455aSJoe Perches			    !($r1 =~ /^$Constant$/ || $r1 =~ /^[A-Z_][A-Z0-9_]*$/)) {
5394e367455aSJoe Perches				if (WARN("ALLOC_WITH_MULTIPLY",
5395e367455aSJoe Perches					 "Prefer $newfunc over $oldfunc with multiply\n" . $herecurr) &&
5396e367455aSJoe Perches				    $fix) {
5397194f66fcSJoe Perches					$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;
539860a55369SJoe Perches
539960a55369SJoe Perches				}
540060a55369SJoe Perches			}
540160a55369SJoe Perches		}
540260a55369SJoe Perches
5403972fdea2SJoe Perches# check for krealloc arg reuse
5404972fdea2SJoe Perches		if ($^V && $^V ge 5.10.0 &&
5405972fdea2SJoe Perches		    $line =~ /\b($Lval)\s*\=\s*(?:$balanced_parens)?\s*krealloc\s*\(\s*\1\s*,/) {
5406972fdea2SJoe Perches			WARN("KREALLOC_ARG_REUSE",
5407972fdea2SJoe Perches			     "Reusing the krealloc arg is almost always a bug\n" . $herecurr);
5408972fdea2SJoe Perches		}
5409972fdea2SJoe Perches
54105ce59ae0SJoe Perches# check for alloc argument mismatch
54115ce59ae0SJoe Perches		if ($line =~ /\b(kcalloc|kmalloc_array)\s*\(\s*sizeof\b/) {
54125ce59ae0SJoe Perches			WARN("ALLOC_ARRAY_ARGS",
54135ce59ae0SJoe Perches			     "$1 uses number as first arg, sizeof is generally wrong\n" . $herecurr);
54145ce59ae0SJoe Perches		}
54155ce59ae0SJoe Perches
5416caf2a54fSJoe Perches# check for multiple semicolons
5417caf2a54fSJoe Perches		if ($line =~ /;\s*;\s*$/) {
5418d5e616fcSJoe Perches			if (WARN("ONE_SEMICOLON",
5419d5e616fcSJoe Perches				 "Statements terminations use 1 semicolon\n" . $herecurr) &&
5420d5e616fcSJoe Perches			    $fix) {
5421194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/(\s*;\s*){2,}$/;/g;
5422d5e616fcSJoe Perches			}
5423d1e2ad07SJoe Perches		}
5424d1e2ad07SJoe Perches
54250ab90191SJoe Perches# check for #defines like: 1 << <digit> that could be BIT(digit)
54260ab90191SJoe Perches		if ($line =~ /#\s*define\s+\w+\s+\(?\s*1\s*([ulUL]*)\s*\<\<\s*(?:\d+|$Ident)\s*\)?/) {
54270ab90191SJoe Perches			my $ull = "";
54280ab90191SJoe Perches			$ull = "_ULL" if (defined($1) && $1 =~ /ll/i);
54290ab90191SJoe Perches			if (CHK("BIT_MACRO",
54300ab90191SJoe Perches				"Prefer using the BIT$ull macro\n" . $herecurr) &&
54310ab90191SJoe Perches			    $fix) {
54320ab90191SJoe Perches				$fixed[$fixlinenr] =~ s/\(?\s*1\s*[ulUL]*\s*<<\s*(\d+|$Ident)\s*\)?/BIT${ull}($1)/;
54330ab90191SJoe Perches			}
54340ab90191SJoe Perches		}
54350ab90191SJoe Perches
5436e81f239bSJoe Perches# check for case / default statements not preceded by break/fallthrough/switch
5437c34c09a8SJoe Perches		if ($line =~ /^.\s*(?:case\s+(?:$Ident|$Constant)\s*|default):/) {
5438c34c09a8SJoe Perches			my $has_break = 0;
5439c34c09a8SJoe Perches			my $has_statement = 0;
5440c34c09a8SJoe Perches			my $count = 0;
5441c34c09a8SJoe Perches			my $prevline = $linenr;
5442e81f239bSJoe Perches			while ($prevline > 1 && ($file || $count < 3) && !$has_break) {
5443c34c09a8SJoe Perches				$prevline--;
5444c34c09a8SJoe Perches				my $rline = $rawlines[$prevline - 1];
5445c34c09a8SJoe Perches				my $fline = $lines[$prevline - 1];
5446c34c09a8SJoe Perches				last if ($fline =~ /^\@\@/);
5447c34c09a8SJoe Perches				next if ($fline =~ /^\-/);
5448c34c09a8SJoe Perches				next if ($fline =~ /^.(?:\s*(?:case\s+(?:$Ident|$Constant)[\s$;]*|default):[\s$;]*)*$/);
5449c34c09a8SJoe Perches				$has_break = 1 if ($rline =~ /fall[\s_-]*(through|thru)/i);
5450c34c09a8SJoe Perches				next if ($fline =~ /^.[\s$;]*$/);
5451c34c09a8SJoe Perches				$has_statement = 1;
5452c34c09a8SJoe Perches				$count++;
5453c34c09a8SJoe Perches				$has_break = 1 if ($fline =~ /\bswitch\b|\b(?:break\s*;[\s$;]*$|return\b|goto\b|continue\b)/);
5454c34c09a8SJoe Perches			}
5455c34c09a8SJoe Perches			if (!$has_break && $has_statement) {
5456c34c09a8SJoe Perches				WARN("MISSING_BREAK",
5457c34c09a8SJoe Perches				     "Possible switch case/default not preceeded by break or fallthrough comment\n" . $herecurr);
5458c34c09a8SJoe Perches			}
5459c34c09a8SJoe Perches		}
5460c34c09a8SJoe Perches
5461d1e2ad07SJoe Perches# check for switch/default statements without a break;
5462d1e2ad07SJoe Perches		if ($^V && $^V ge 5.10.0 &&
5463d1e2ad07SJoe Perches		    defined $stat &&
5464d1e2ad07SJoe Perches		    $stat =~ /^\+[$;\s]*(?:case[$;\s]+\w+[$;\s]*:[$;\s]*|)*[$;\s]*\bdefault[$;\s]*:[$;\s]*;/g) {
5465d1e2ad07SJoe Perches			my $ctx = '';
5466d1e2ad07SJoe Perches			my $herectx = $here . "\n";
5467d1e2ad07SJoe Perches			my $cnt = statement_rawlines($stat);
5468d1e2ad07SJoe Perches			for (my $n = 0; $n < $cnt; $n++) {
5469d1e2ad07SJoe Perches				$herectx .= raw_line($linenr, $n) . "\n";
5470d1e2ad07SJoe Perches			}
5471d1e2ad07SJoe Perches			WARN("DEFAULT_NO_BREAK",
5472d1e2ad07SJoe Perches			     "switch default: should use break\n" . $herectx);
5473caf2a54fSJoe Perches		}
5474caf2a54fSJoe Perches
547513214adfSAndy Whitcroft# check for gcc specific __FUNCTION__
5476d5e616fcSJoe Perches		if ($line =~ /\b__FUNCTION__\b/) {
5477d5e616fcSJoe Perches			if (WARN("USE_FUNC",
5478d5e616fcSJoe Perches				 "__func__ should be used instead of gcc specific __FUNCTION__\n"  . $herecurr) &&
5479d5e616fcSJoe Perches			    $fix) {
5480194f66fcSJoe Perches				$fixed[$fixlinenr] =~ s/\b__FUNCTION__\b/__func__/g;
5481d5e616fcSJoe Perches			}
548213214adfSAndy Whitcroft		}
5483773647a0SAndy Whitcroft
548462ec818fSJoe Perches# check for uses of __DATE__, __TIME__, __TIMESTAMP__
548562ec818fSJoe Perches		while ($line =~ /\b(__(?:DATE|TIME|TIMESTAMP)__)\b/g) {
548662ec818fSJoe Perches			ERROR("DATE_TIME",
548762ec818fSJoe Perches			      "Use of the '$1' macro makes the build non-deterministic\n" . $herecurr);
548862ec818fSJoe Perches		}
548962ec818fSJoe Perches
54902c92488aSJoe Perches# check for use of yield()
54912c92488aSJoe Perches		if ($line =~ /\byield\s*\(\s*\)/) {
54922c92488aSJoe Perches			WARN("YIELD",
54932c92488aSJoe Perches			     "Using yield() is generally wrong. See yield() kernel-doc (sched/core.c)\n"  . $herecurr);
54942c92488aSJoe Perches		}
54952c92488aSJoe Perches
5496179f8f40SJoe Perches# check for comparisons against true and false
5497179f8f40SJoe Perches		if ($line =~ /\+\s*(.*?)\b(true|false|$Lval)\s*(==|\!=)\s*(true|false|$Lval)\b(.*)$/i) {
5498179f8f40SJoe Perches			my $lead = $1;
5499179f8f40SJoe Perches			my $arg = $2;
5500179f8f40SJoe Perches			my $test = $3;
5501179f8f40SJoe Perches			my $otype = $4;
5502179f8f40SJoe Perches			my $trail = $5;
5503179f8f40SJoe Perches			my $op = "!";
5504179f8f40SJoe Perches
5505179f8f40SJoe Perches			($arg, $otype) = ($otype, $arg) if ($arg =~ /^(?:true|false)$/i);
5506179f8f40SJoe Perches
5507179f8f40SJoe Perches			my $type = lc($otype);
5508179f8f40SJoe Perches			if ($type =~ /^(?:true|false)$/) {
5509179f8f40SJoe Perches				if (("$test" eq "==" && "$type" eq "true") ||
5510179f8f40SJoe Perches				    ("$test" eq "!=" && "$type" eq "false")) {
5511179f8f40SJoe Perches					$op = "";
5512179f8f40SJoe Perches				}
5513179f8f40SJoe Perches
5514179f8f40SJoe Perches				CHK("BOOL_COMPARISON",
5515179f8f40SJoe Perches				    "Using comparison to $otype is error prone\n" . $herecurr);
5516179f8f40SJoe Perches
5517179f8f40SJoe Perches## maybe suggesting a correct construct would better
5518179f8f40SJoe Perches##				    "Using comparison to $otype is error prone.  Perhaps use '${lead}${op}${arg}${trail}'\n" . $herecurr);
5519179f8f40SJoe Perches
5520179f8f40SJoe Perches			}
5521179f8f40SJoe Perches		}
5522179f8f40SJoe Perches
55234882720bSThomas Gleixner# check for semaphores initialized locked
55244882720bSThomas Gleixner		if ($line =~ /^.\s*sema_init.+,\W?0\W?\)/) {
5525000d1cc1SJoe Perches			WARN("CONSIDER_COMPLETION",
5526000d1cc1SJoe Perches			     "consider using a completion\n" . $herecurr);
5527773647a0SAndy Whitcroft		}
55286712d858SJoe Perches
552967d0a075SJoe Perches# recommend kstrto* over simple_strto* and strict_strto*
553067d0a075SJoe Perches		if ($line =~ /\b((simple|strict)_(strto(l|ll|ul|ull)))\s*\(/) {
5531000d1cc1SJoe Perches			WARN("CONSIDER_KSTRTO",
553267d0a075SJoe Perches			     "$1 is obsolete, use k$3 instead\n" . $herecurr);
5533773647a0SAndy Whitcroft		}
55346712d858SJoe Perches
5535ae3ccc46SFabian Frederick# check for __initcall(), use device_initcall() explicitly or more appropriate function please
5536f3db6639SMichael Ellerman		if ($line =~ /^.\s*__initcall\s*\(/) {
5537000d1cc1SJoe Perches			WARN("USE_DEVICE_INITCALL",
5538ae3ccc46SFabian Frederick			     "please use device_initcall() or more appropriate function instead of __initcall() (see include/linux/init.h)\n" . $herecurr);
5539f3db6639SMichael Ellerman		}
55406712d858SJoe Perches
55410f3c5aabSJoe Perches# check for various structs that are normally const (ops, kgdb, device_tree)
55420f3c5aabSJoe Perches		my $const_structs = qr{
55430f3c5aabSJoe Perches				acpi_dock_ops|
554479404849SEmese Revfy				address_space_operations|
554579404849SEmese Revfy				backlight_ops|
554679404849SEmese Revfy				block_device_operations|
554779404849SEmese Revfy				dentry_operations|
554879404849SEmese Revfy				dev_pm_ops|
554979404849SEmese Revfy				dma_map_ops|
555079404849SEmese Revfy				extent_io_ops|
555179404849SEmese Revfy				file_lock_operations|
555279404849SEmese Revfy				file_operations|
555379404849SEmese Revfy				hv_ops|
555479404849SEmese Revfy				ide_dma_ops|
555579404849SEmese Revfy				intel_dvo_dev_ops|
555679404849SEmese Revfy				item_operations|
555779404849SEmese Revfy				iwl_ops|
555879404849SEmese Revfy				kgdb_arch|
555979404849SEmese Revfy				kgdb_io|
556079404849SEmese Revfy				kset_uevent_ops|
556179404849SEmese Revfy				lock_manager_operations|
556279404849SEmese Revfy				microcode_ops|
556379404849SEmese Revfy				mtrr_ops|
556479404849SEmese Revfy				neigh_ops|
556579404849SEmese Revfy				nlmsvc_binding|
55660f3c5aabSJoe Perches				of_device_id|
556779404849SEmese Revfy				pci_raw_ops|
556879404849SEmese Revfy				pipe_buf_operations|
556979404849SEmese Revfy				platform_hibernation_ops|
557079404849SEmese Revfy				platform_suspend_ops|
557179404849SEmese Revfy				proto_ops|
557279404849SEmese Revfy				rpc_pipe_ops|
557379404849SEmese Revfy				seq_operations|
557479404849SEmese Revfy				snd_ac97_build_ops|
557579404849SEmese Revfy				soc_pcmcia_socket_ops|
557679404849SEmese Revfy				stacktrace_ops|
557779404849SEmese Revfy				sysfs_ops|
557879404849SEmese Revfy				tty_operations|
55796d07d01bSJoe Perches				uart_ops|
558079404849SEmese Revfy				usb_mon_operations|
558179404849SEmese Revfy				wd_ops}x;
55826903ffb2SAndy Whitcroft		if ($line !~ /\bconst\b/ &&
55830f3c5aabSJoe Perches		    $line =~ /\bstruct\s+($const_structs)\b/) {
5584000d1cc1SJoe Perches			WARN("CONST_STRUCT",
5585000d1cc1SJoe Perches			     "struct $1 should normally be const\n" .
55866903ffb2SAndy Whitcroft				$herecurr);
55872b6db5cbSAndy Whitcroft		}
5588773647a0SAndy Whitcroft
5589773647a0SAndy Whitcroft# use of NR_CPUS is usually wrong
5590773647a0SAndy Whitcroft# ignore definitions of NR_CPUS and usage to define arrays as likely right
5591773647a0SAndy Whitcroft		if ($line =~ /\bNR_CPUS\b/ &&
5592c45dcabdSAndy Whitcroft		    $line !~ /^.\s*\s*#\s*if\b.*\bNR_CPUS\b/ &&
5593c45dcabdSAndy Whitcroft		    $line !~ /^.\s*\s*#\s*define\b.*\bNR_CPUS\b/ &&
5594171ae1a4SAndy Whitcroft		    $line !~ /^.\s*$Declare\s.*\[[^\]]*NR_CPUS[^\]]*\]/ &&
5595171ae1a4SAndy Whitcroft		    $line !~ /\[[^\]]*\.\.\.[^\]]*NR_CPUS[^\]]*\]/ &&
5596171ae1a4SAndy Whitcroft		    $line !~ /\[[^\]]*NR_CPUS[^\]]*\.\.\.[^\]]*\]/)
5597773647a0SAndy Whitcroft		{
5598000d1cc1SJoe Perches			WARN("NR_CPUS",
5599000d1cc1SJoe Perches			     "usage of NR_CPUS is often wrong - consider using cpu_possible(), num_possible_cpus(), for_each_possible_cpu(), etc\n" . $herecurr);
5600773647a0SAndy Whitcroft		}
56019c9ba34eSAndy Whitcroft
560252ea8506SJoe Perches# Use of __ARCH_HAS_<FOO> or ARCH_HAVE_<BAR> is wrong.
560352ea8506SJoe Perches		if ($line =~ /\+\s*#\s*define\s+((?:__)?ARCH_(?:HAS|HAVE)\w*)\b/) {
560452ea8506SJoe Perches			ERROR("DEFINE_ARCH_HAS",
560552ea8506SJoe Perches			      "#define of '$1' is wrong - use Kconfig variables or standard guards instead\n" . $herecurr);
560652ea8506SJoe Perches		}
560752ea8506SJoe Perches
5608acd9362cSJoe Perches# likely/unlikely comparisons similar to "(likely(foo) > 0)"
5609acd9362cSJoe Perches		if ($^V && $^V ge 5.10.0 &&
5610acd9362cSJoe Perches		    $line =~ /\b((?:un)?likely)\s*\(\s*$FuncArg\s*\)\s*$Compare/) {
5611acd9362cSJoe Perches			WARN("LIKELY_MISUSE",
5612acd9362cSJoe Perches			     "Using $1 should generally have parentheses around the comparison\n" . $herecurr);
5613acd9362cSJoe Perches		}
5614acd9362cSJoe Perches
5615691d77b6SAndy Whitcroft# whine mightly about in_atomic
5616691d77b6SAndy Whitcroft		if ($line =~ /\bin_atomic\s*\(/) {
5617691d77b6SAndy Whitcroft			if ($realfile =~ m@^drivers/@) {
5618000d1cc1SJoe Perches				ERROR("IN_ATOMIC",
5619000d1cc1SJoe Perches				      "do not use in_atomic in drivers\n" . $herecurr);
5620f4a87736SAndy Whitcroft			} elsif ($realfile !~ m@^kernel/@) {
5621000d1cc1SJoe Perches				WARN("IN_ATOMIC",
5622000d1cc1SJoe Perches				     "use of in_atomic() is incorrect outside core kernel code\n" . $herecurr);
5623691d77b6SAndy Whitcroft			}
5624691d77b6SAndy Whitcroft		}
56251704f47bSPeter Zijlstra
56261704f47bSPeter Zijlstra# check for lockdep_set_novalidate_class
56271704f47bSPeter Zijlstra		if ($line =~ /^.\s*lockdep_set_novalidate_class\s*\(/ ||
56281704f47bSPeter Zijlstra		    $line =~ /__lockdep_no_validate__\s*\)/ ) {
56291704f47bSPeter Zijlstra			if ($realfile !~ m@^kernel/lockdep@ &&
56301704f47bSPeter Zijlstra			    $realfile !~ m@^include/linux/lockdep@ &&
56311704f47bSPeter Zijlstra			    $realfile !~ m@^drivers/base/core@) {
5632000d1cc1SJoe Perches				ERROR("LOCKDEP",
5633000d1cc1SJoe Perches				      "lockdep_no_validate class is reserved for device->mutex.\n" . $herecurr);
56341704f47bSPeter Zijlstra			}
56351704f47bSPeter Zijlstra		}
563688f8831cSDave Jones
5637b392c64fSJoe Perches		if ($line =~ /debugfs_create_\w+.*\b$mode_perms_world_writable\b/ ||
5638b392c64fSJoe Perches		    $line =~ /DEVICE_ATTR.*\b$mode_perms_world_writable\b/) {
5639000d1cc1SJoe Perches			WARN("EXPORTED_WORLD_WRITABLE",
5640000d1cc1SJoe Perches			     "Exporting world writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr);
564188f8831cSDave Jones		}
56422435880fSJoe Perches
5643515a235eSJoe Perches# Mode permission misuses where it seems decimal should be octal
5644515a235eSJoe Perches# This uses a shortcut match to avoid unnecessary uses of a slow foreach loop
5645515a235eSJoe Perches		if ($^V && $^V ge 5.10.0 &&
5646515a235eSJoe Perches		    $line =~ /$mode_perms_search/) {
56472435880fSJoe Perches			foreach my $entry (@mode_permission_funcs) {
56482435880fSJoe Perches				my $func = $entry->[0];
56492435880fSJoe Perches				my $arg_pos = $entry->[1];
56502435880fSJoe Perches
56512435880fSJoe Perches				my $skip_args = "";
56522435880fSJoe Perches				if ($arg_pos > 1) {
56532435880fSJoe Perches					$arg_pos--;
56542435880fSJoe Perches					$skip_args = "(?:\\s*$FuncArg\\s*,\\s*){$arg_pos,$arg_pos}";
56552435880fSJoe Perches				}
56562435880fSJoe Perches				my $test = "\\b$func\\s*\\(${skip_args}([\\d]+)\\s*[,\\)]";
5657515a235eSJoe Perches				if ($line =~ /$test/) {
56582435880fSJoe Perches					my $val = $1;
56592435880fSJoe Perches					$val = $6 if ($skip_args ne "");
56602435880fSJoe Perches
56611727cc70SJoe Perches					if ($val !~ /^0$/ &&
56621727cc70SJoe Perches					    (($val =~ /^$Int$/ && $val !~ /^$Octal$/) ||
56631727cc70SJoe Perches					     length($val) ne 4)) {
56642435880fSJoe Perches						ERROR("NON_OCTAL_PERMISSIONS",
56651727cc70SJoe Perches						      "Use 4 digit octal (0777) not decimal permissions\n" . $herecurr);
5666c0a5c898SJoe Perches					} elsif ($val =~ /^$Octal$/ && (oct($val) & 02)) {
5667c0a5c898SJoe Perches						ERROR("EXPORTED_WORLD_WRITABLE",
5668c0a5c898SJoe Perches						      "Exporting writable files is usually an error. Consider more restrictive permissions.\n" . $herecurr);
56692435880fSJoe Perches					}
56702435880fSJoe Perches				}
56712435880fSJoe Perches			}
567213214adfSAndy Whitcroft		}
56735a6d20ceSBjorn Andersson
56745a6d20ceSBjorn Andersson# validate content of MODULE_LICENSE against list from include/linux/module.h
56755a6d20ceSBjorn Andersson		if ($line =~ /\bMODULE_LICENSE\s*\(\s*($String)\s*\)/) {
56765a6d20ceSBjorn Andersson			my $extracted_string = get_quoted_string($line, $rawline);
56775a6d20ceSBjorn Andersson			my $valid_licenses = qr{
56785a6d20ceSBjorn Andersson						GPL|
56795a6d20ceSBjorn Andersson						GPL\ v2|
56805a6d20ceSBjorn Andersson						GPL\ and\ additional\ rights|
56815a6d20ceSBjorn Andersson						Dual\ BSD/GPL|
56825a6d20ceSBjorn Andersson						Dual\ MIT/GPL|
56835a6d20ceSBjorn Andersson						Dual\ MPL/GPL|
56845a6d20ceSBjorn Andersson						Proprietary
56855a6d20ceSBjorn Andersson					}x;
56865a6d20ceSBjorn Andersson			if ($extracted_string !~ /^"(?:$valid_licenses)"$/x) {
56875a6d20ceSBjorn Andersson				WARN("MODULE_LICENSE",
56885a6d20ceSBjorn Andersson				     "unknown module license " . $extracted_string . "\n" . $herecurr);
56895a6d20ceSBjorn Andersson			}
56905a6d20ceSBjorn Andersson		}
5691515a235eSJoe Perches	}
569213214adfSAndy Whitcroft
569313214adfSAndy Whitcroft	# If we have no input at all, then there is nothing to report on
569413214adfSAndy Whitcroft	# so just keep quiet.
569513214adfSAndy Whitcroft	if ($#rawlines == -1) {
569613214adfSAndy Whitcroft		exit(0);
56970a920b5bSAndy Whitcroft	}
56980a920b5bSAndy Whitcroft
56998905a67cSAndy Whitcroft	# In mailback mode only produce a report in the negative, for
57008905a67cSAndy Whitcroft	# things that appear to be patches.
57018905a67cSAndy Whitcroft	if ($mailback && ($clean == 1 || !$is_patch)) {
57028905a67cSAndy Whitcroft		exit(0);
57038905a67cSAndy Whitcroft	}
57048905a67cSAndy Whitcroft
57058905a67cSAndy Whitcroft	# This is not a patch, and we are are in 'no-patch' mode so
57068905a67cSAndy Whitcroft	# just keep quiet.
57078905a67cSAndy Whitcroft	if (!$chk_patch && !$is_patch) {
57088905a67cSAndy Whitcroft		exit(0);
57098905a67cSAndy Whitcroft	}
57108905a67cSAndy Whitcroft
571106330fc4SJoe Perches	if (!$is_patch && $file !~ /cover-letter\.patch$/) {
5712000d1cc1SJoe Perches		ERROR("NOT_UNIFIED_DIFF",
5713000d1cc1SJoe Perches		      "Does not appear to be a unified-diff format patch\n");
57140a920b5bSAndy Whitcroft	}
571534d8815fSJoe Perches	if ($is_patch && $filename ne '-' && $chk_signoff && $signoff == 0) {
5716000d1cc1SJoe Perches		ERROR("MISSING_SIGN_OFF",
5717000d1cc1SJoe Perches		      "Missing Signed-off-by: line(s)\n");
57180a920b5bSAndy Whitcroft	}
57190a920b5bSAndy Whitcroft
5720f0a594c1SAndy Whitcroft	print report_dump();
572113214adfSAndy Whitcroft	if ($summary && !($clean == 1 && $quiet == 1)) {
572213214adfSAndy Whitcroft		print "$filename " if ($summary_file);
57236c72ffaaSAndy Whitcroft		print "total: $cnt_error errors, $cnt_warn warnings, " .
57246c72ffaaSAndy Whitcroft			(($check)? "$cnt_chk checks, " : "") .
57256c72ffaaSAndy Whitcroft			"$cnt_lines lines checked\n";
57266c72ffaaSAndy Whitcroft	}
57278905a67cSAndy Whitcroft
5728d2c0a235SAndy Whitcroft	if ($quiet == 0) {
5729d2c0a235SAndy Whitcroft		# If there were whitespace errors which cleanpatch can fix
5730d2c0a235SAndy Whitcroft		# then suggest that.
5731d2c0a235SAndy Whitcroft		if ($rpt_cleaners) {
5732b0781216SMike Frysinger			$rpt_cleaners = 0;
5733d8469f16SJoe Perches			print << "EOM"
5734d8469f16SJoe Perches
5735d8469f16SJoe PerchesNOTE: Whitespace errors detected.
5736d8469f16SJoe Perches      You may wish to use scripts/cleanpatch or scripts/cleanfile
5737d8469f16SJoe PerchesEOM
5738d2c0a235SAndy Whitcroft		}
5739d2c0a235SAndy Whitcroft	}
5740d2c0a235SAndy Whitcroft
5741d752fcc8SJoe Perches	if ($clean == 0 && $fix &&
5742d752fcc8SJoe Perches	    ("@rawlines" ne "@fixed" ||
5743d752fcc8SJoe Perches	     $#fixed_inserted >= 0 || $#fixed_deleted >= 0)) {
57449624b8d6SJoe Perches		my $newfile = $filename;
57459624b8d6SJoe Perches		$newfile .= ".EXPERIMENTAL-checkpatch-fixes" if (!$fix_inplace);
57463705ce5bSJoe Perches		my $linecount = 0;
57473705ce5bSJoe Perches		my $f;
57483705ce5bSJoe Perches
5749d752fcc8SJoe Perches		@fixed = fix_inserted_deleted_lines(\@fixed, \@fixed_inserted, \@fixed_deleted);
5750d752fcc8SJoe Perches
57513705ce5bSJoe Perches		open($f, '>', $newfile)
57523705ce5bSJoe Perches		    or die "$P: Can't open $newfile for write\n";
57533705ce5bSJoe Perches		foreach my $fixed_line (@fixed) {
57543705ce5bSJoe Perches			$linecount++;
57553705ce5bSJoe Perches			if ($file) {
57563705ce5bSJoe Perches				if ($linecount > 3) {
57573705ce5bSJoe Perches					$fixed_line =~ s/^\+//;
57583705ce5bSJoe Perches					print $f $fixed_line . "\n";
57593705ce5bSJoe Perches				}
57603705ce5bSJoe Perches			} else {
57613705ce5bSJoe Perches				print $f $fixed_line . "\n";
57623705ce5bSJoe Perches			}
57633705ce5bSJoe Perches		}
57643705ce5bSJoe Perches		close($f);
57653705ce5bSJoe Perches
57663705ce5bSJoe Perches		if (!$quiet) {
57673705ce5bSJoe Perches			print << "EOM";
5768d8469f16SJoe Perches
57693705ce5bSJoe PerchesWrote EXPERIMENTAL --fix correction(s) to '$newfile'
57703705ce5bSJoe Perches
57713705ce5bSJoe PerchesDo _NOT_ trust the results written to this file.
57723705ce5bSJoe PerchesDo _NOT_ submit these changes without inspecting them for correctness.
57733705ce5bSJoe Perches
57743705ce5bSJoe PerchesThis EXPERIMENTAL file is simply a convenience to help rewrite patches.
57753705ce5bSJoe PerchesNo warranties, expressed or implied...
57763705ce5bSJoe PerchesEOM
57773705ce5bSJoe Perches		}
57783705ce5bSJoe Perches	}
57793705ce5bSJoe Perches
5780d8469f16SJoe Perches	if ($quiet == 0) {
5781d8469f16SJoe Perches		print "\n";
5782d8469f16SJoe Perches		if ($clean == 1) {
5783d8469f16SJoe Perches			print "$vname has no obvious style problems and is ready for submission.\n";
5784d8469f16SJoe Perches		} else {
5785d8469f16SJoe Perches			print "$vname has style problems, please review.\n";
57860a920b5bSAndy Whitcroft		}
57870a920b5bSAndy Whitcroft	}
57880a920b5bSAndy Whitcroft	return $clean;
57890a920b5bSAndy Whitcroft}
5790