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