2007-06-01 00:46:48 -07:00
#!/usr/bin/perl -w
# (c) 2001, Dave Jones. <davej@codemonkey.org.uk> (the file handling bit)
2007-06-08 13:47:06 -07:00
# (c) 2005, Joel Schopp <jschopp@austin.ibm.com> (the ugly bit)
2007-06-01 00:46:48 -07:00
# (c) 2007, Andy Whitcroft <apw@uk.ibm.com> (new conditions, test suite, etc)
# Licensed under the terms of the GNU GPL License version 2
use strict ;
my $ P = $ 0 ;
2007-06-08 13:47:06 -07:00
$ P =~ s@.*/@@g ;
2007-06-01 00:46:48 -07:00
2007-07-15 23:37:22 -07:00
my $ V = '0.07' ;
2007-06-01 00:46:48 -07:00
use Getopt::Long qw( :config no_auto_abbrev ) ;
my $ quiet = 0 ;
my $ tree = 1 ;
my $ chk_signoff = 1 ;
my $ chk_patch = 1 ;
2007-06-23 17:16:34 -07:00
my $ tst_type = 0 ;
2007-06-01 00:46:48 -07:00
GetOptions (
'q|quiet' = > \ $ quiet ,
'tree!' = > \ $ tree ,
'signoff!' = > \ $ chk_signoff ,
'patch!' = > \ $ chk_patch ,
2007-06-23 17:16:34 -07:00
'test-type!' = > \ $ tst_type ,
2007-06-01 00:46:48 -07:00
) or exit ;
my $ exit = 0 ;
if ( $# ARGV < 0 ) {
2007-06-08 13:47:06 -07:00
print "usage: $P [options] patchfile\n" ;
2007-06-01 00:46:48 -07:00
print "version: $V\n" ;
print "options: -q => quiet\n" ;
print " --no-tree => run without a kernel tree\n" ;
exit ( 1 ) ;
}
if ( $ tree && ! top_of_kernel_tree ( ) ) {
print "Must be run from the top-level dir. of a kernel tree\n" ;
exit ( 2 ) ;
}
2007-06-08 13:46:39 -07:00
my @ dep_includes = ( ) ;
my @ dep_functions = ( ) ;
2007-06-01 00:46:48 -07:00
my $ removal = 'Documentation/feature-removal-schedule.txt' ;
if ( $ tree && - f $ removal ) {
open ( REMOVE , "<$removal" ) || die "$P: $removal: open failed - $!\n" ;
while ( <REMOVE> ) {
if ( /^Files:\s+(.*\S)/ ) {
for my $ file ( split ( /[, ]+/ , $ 1 ) ) {
if ( $ file =~ m @include/(.*)@ ) {
2007-06-08 13:46:39 -07:00
push ( @ dep_includes , $ 1 ) ;
2007-06-01 00:46:48 -07:00
}
}
2007-06-08 13:46:39 -07:00
} elsif ( /^Funcs:\s+(.*\S)/ ) {
for my $ func ( split ( /[, ]+/ , $ 1 ) ) {
push ( @ dep_functions , $ func ) ;
}
2007-06-01 00:46:48 -07:00
}
}
}
2007-06-08 13:47:06 -07:00
my @ rawlines = ( ) ;
2007-06-01 00:46:48 -07:00
while ( < > ) {
chomp ;
2007-06-08 13:47:06 -07:00
push ( @ rawlines , $ _ ) ;
2007-06-01 00:46:48 -07:00
if ( eof ( ARGV ) ) {
2007-06-08 13:47:06 -07:00
if ( ! process ( $ ARGV , @ rawlines ) ) {
2007-06-01 00:46:48 -07:00
$ exit = 1 ;
}
2007-06-08 13:47:06 -07:00
@ rawlines = ( ) ;
2007-06-01 00:46:48 -07:00
}
}
exit ( $ exit ) ;
sub top_of_kernel_tree {
if ( ( - f "COPYING" ) && ( - f "CREDITS" ) && ( - f "Kbuild" ) &&
( - f "MAINTAINERS" ) && ( - f "Makefile" ) && ( - f "README" ) &&
( - d "Documentation" ) && ( - d "arch" ) && ( - d "include" ) &&
( - d "drivers" ) && ( - d "fs" ) && ( - d "init" ) && ( - d "ipc" ) &&
( - d "kernel" ) && ( - d "lib" ) && ( - d "scripts" ) ) {
return 1 ;
}
return 0 ;
}
sub expand_tabs {
my ( $ str ) = @ _ ;
my $ res = '' ;
my $ n = 0 ;
for my $ c ( split ( // , $ str ) ) {
if ( $ c eq "\t" ) {
$ res . = ' ' ;
$ n + + ;
for ( ; ( $ n % 8 ) != 0 ; $ n + + ) {
$ res . = ' ' ;
}
next ;
}
$ res . = $ c ;
$ n + + ;
}
return $ res ;
}
2007-06-08 13:46:39 -07:00
sub line_stats {
my ( $ line ) = @ _ ;
# Drop the diff line leader and expand tabs
$ line =~ s/^.// ;
$ line = expand_tabs ( $ line ) ;
# Pick the indent from the front of the line.
my ( $ white ) = ( $ line =~ /^(\s*)/ ) ;
return ( length ( $ line ) , length ( $ white ) ) ;
}
2007-06-08 13:47:06 -07:00
sub sanitise_line {
my ( $ line ) = @ _ ;
my $ res = '' ;
my $ l = '' ;
my $ quote = '' ;
foreach my $ c ( split ( // , $ line ) ) {
if ( $ l ne "\\" && ( $ c eq "'" || $ c eq '"' ) ) {
if ( $ quote eq '' ) {
$ quote = $ c ;
$ res . = $ c ;
$ l = $ c ;
next ;
} elsif ( $ quote eq $ c ) {
$ quote = '' ;
}
}
if ( $ quote && $ c ne "\t" ) {
$ res . = "X" ;
} else {
$ res . = $ c ;
}
$ l = $ c ;
}
return $ res ;
}
2007-06-08 13:46:39 -07:00
sub ctx_block_get {
2007-06-23 17:16:34 -07:00
my ( $ linenr , $ remain , $ outer , $ open , $ close ) = @ _ ;
2007-06-08 13:46:39 -07:00
my $ line ;
my $ start = $ linenr - 1 ;
my $ blk = '' ;
my @ o ;
my @ c ;
my @ res = ( ) ;
2007-06-08 13:47:06 -07:00
for ( $ line = $ start ; $ remain > 0 ; $ line + + ) {
next if ( $ rawlines [ $ line ] =~ /^-/ ) ;
$ remain - - ;
$ blk . = $ rawlines [ $ line ] ;
2007-06-08 13:46:39 -07:00
2007-06-23 17:16:34 -07:00
@ o = ( $ blk =~ /$open/g ) ;
@ c = ( $ blk =~ /$close/g ) ;
2007-06-08 13:46:39 -07:00
if ( ! $ outer || ( scalar ( @ o ) - scalar ( @ c ) ) == 1 ) {
2007-06-08 13:47:06 -07:00
push ( @ res , $ rawlines [ $ line ] ) ;
2007-06-08 13:46:39 -07:00
}
last if ( scalar ( @ o ) == scalar ( @ c ) ) ;
}
return @ res ;
}
sub ctx_block_outer {
my ( $ linenr , $ remain ) = @ _ ;
2007-06-23 17:16:34 -07:00
return ctx_block_get ( $ linenr , $ remain , 1 , '\{' , '\}' ) ;
2007-06-08 13:46:39 -07:00
}
sub ctx_block {
my ( $ linenr , $ remain ) = @ _ ;
2007-06-23 17:16:34 -07:00
return ctx_block_get ( $ linenr , $ remain , 0 , '\{' , '\}' ) ;
}
sub ctx_statement {
my ( $ linenr , $ remain ) = @ _ ;
return ctx_block_get ( $ linenr , $ remain , 0 , '\(' , '\)' ) ;
2007-06-08 13:46:39 -07:00
}
sub ctx_locate_comment {
my ( $ first_line , $ end_line ) = @ _ ;
# Catch a comment on the end of the line itself.
2007-06-08 13:47:06 -07:00
my ( $ current_comment ) = ( $ rawlines [ $ end_line - 1 ] =~ m @.*(/\*.*\*/)\s*$@ ) ;
2007-06-08 13:46:39 -07:00
return $ current_comment if ( defined $ current_comment ) ;
# Look through the context and try and figure out if there is a
# comment.
my $ in_comment = 0 ;
$ current_comment = '' ;
for ( my $ linenr = $ first_line ; $ linenr < $ end_line ; $ linenr + + ) {
2007-06-08 13:47:06 -07:00
my $ line = $ rawlines [ $ linenr - 1 ] ;
#warn " $line\n";
2007-06-08 13:46:39 -07:00
if ( $ linenr == $ first_line and $ line =~ m @^.\s*\*@ ) {
$ in_comment = 1 ;
}
if ( $ line =~ m @/\*@ ) {
$ in_comment = 1 ;
}
if ( ! $ in_comment && $ current_comment ne '' ) {
$ current_comment = '' ;
}
$ current_comment . = $ line . "\n" if ( $ in_comment ) ;
if ( $ line =~ m @\*/@ ) {
$ in_comment = 0 ;
}
}
chomp ( $ current_comment ) ;
return ( $ current_comment ) ;
}
sub ctx_has_comment {
my ( $ first_line , $ end_line ) = @ _ ;
my $ cmt = ctx_locate_comment ( $ first_line , $ end_line ) ;
2007-06-08 13:47:06 -07:00
##print "LINE: $rawlines[$end_line - 1 ]\n";
2007-06-08 13:46:39 -07:00
##print "CMMT: $cmt\n";
return ( $ cmt ne '' ) ;
}
2007-06-01 00:46:48 -07:00
sub cat_vet {
my ( $ vet ) = @ _ ;
$ vet =~ s/\t/^I/ ;
$ vet =~ s/$/\$/ ;
return $ vet ;
}
2007-07-15 23:37:22 -07:00
sub ERROR {
print "ERROR: $_[0]\n" ;
our $ clean = 0 ;
}
sub WARN {
print "WARNING: $_[0]\n" ;
our $ clean = 0 ;
}
sub CHK {
print "CHECK: $_[0]\n" ;
our $ clean = 0 ;
}
2007-06-01 00:46:48 -07:00
sub process {
my $ filename = shift ;
my @ lines = @ _ ;
my $ linenr = 0 ;
my $ prevline = "" ;
my $ stashline = "" ;
2007-06-08 13:46:39 -07:00
my $ length ;
2007-06-01 00:46:48 -07:00
my $ indent ;
my $ previndent = 0 ;
my $ stashindent = 0 ;
2007-07-15 23:37:22 -07:00
our $ clean = 1 ;
2007-06-01 00:46:48 -07:00
my $ signoff = 0 ;
my $ is_patch = 0 ;
# Trace the real file/line as we go.
my $ realfile = '' ;
my $ realline = 0 ;
my $ realcnt = 0 ;
my $ here = '' ;
my $ in_comment = 0 ;
my $ first_line = 0 ;
2007-06-23 17:16:44 -07:00
my $ Ident = qr{ [A-Za-z \ d_]+ } ;
my $ Storage = qr{ extern|static } ;
my $ Sparse = qr{ __user|__kernel|__force|__iomem } ;
my $ NonptrType = qr{
\ b
( ? : const \ s + ) ?
( ? : unsigned \ s + ) ?
( ? :
void |
char |
short |
int |
long |
unsigned |
float |
double |
long \ s + int |
long \ s + long |
long \ s + long \ s + int |
2007-07-15 23:37:22 -07:00
u8 | u16 | u32 | u64 |
s8 | s16 | s32 | s64 |
2007-06-23 17:16:44 -07:00
struct \ s + $ Ident |
union \ s + $ Ident |
2007-07-15 23:37:22 -07:00
enum \ s + $ Ident |
2007-06-23 17:16:44 -07:00
$ { Ident } _t
)
( ? : \ s + $ Sparse ) *
\ b
} x ;
my $ Type = qr{
\ b $ NonptrType \ b
( ? : \ s * \ * + \ s * const | \ s * \ * + ) ?
} x ;
my $ Declare = qr{ (?:$Storage \ s+)?$Type } ;
my $ Attribute = qr{ __read_mostly|__init|__initdata } ;
2007-06-23 17:16:34 -07:00
2007-07-15 23:37:22 -07:00
# Pre-scan the patch looking for any __setup documentation.
my @ setup_docs = ( ) ;
my $ setup_docs = 0 ;
foreach my $ line ( @ lines ) {
if ( $ line =~ /^\+\+\+\s+(\S+)/ ) {
$ setup_docs = 0 ;
if ( $ 1 =~ m @Documentation/kernel-parameters.txt$@ ) {
$ setup_docs = 1 ;
}
next ;
}
if ( $ setup_docs && $ line =~ /^\+/ ) {
push ( @ setup_docs , $ line ) ;
}
}
2007-06-01 00:46:48 -07:00
foreach my $ line ( @ lines ) {
$ linenr + + ;
2007-06-23 17:16:34 -07:00
my $ rawline = $ line ;
2007-06-01 00:46:48 -07:00
#extract the filename as it passes
if ( $ line =~ /^\+\+\+\s+(\S+)/ ) {
$ realfile = $ 1 ;
2007-06-08 13:47:06 -07:00
$ realfile =~ s@^[^/]*/@@ ;
2007-06-01 00:46:48 -07:00
$ in_comment = 0 ;
next ;
}
#extract the line range in the file after the patch is applied
if ( $ line =~ /^\@\@ -\d+,\d+ \+(\d+)(,(\d+))? \@\@/ ) {
$ is_patch = 1 ;
2007-06-08 13:46:39 -07:00
$ first_line = $ linenr + 1 ;
2007-06-01 00:46:48 -07:00
$ in_comment = 0 ;
$ realline = $ 1 - 1 ;
if ( defined $ 2 ) {
$ realcnt = $ 3 + 1 ;
} else {
$ realcnt = 1 + 1 ;
}
next ;
}
2007-06-08 13:46:39 -07:00
# track the line number as we move through the hunk, note that
# new versions of GNU diff omit the leading space on completely
# blank context lines so we need to count that too.
if ( $ line =~ /^( |\+|$)/ ) {
2007-06-01 00:46:48 -07:00
$ realline + + ;
2007-06-23 17:16:44 -07:00
$ realcnt - - if ( $ realcnt != 0 ) ;
2007-06-01 00:46:48 -07:00
# track any sort of multi-line comment. Obviously if
# the added text or context do not include the whole
# comment we will not see it. Such is life.
#
# Guestimate if this is a continuing comment. If this
# is the start of a diff block and this line starts
# ' *' then it is very likely a comment.
2007-06-08 13:46:39 -07:00
if ( $ linenr == $ first_line and $ line =~ m @^.\s*\*@ ) {
2007-06-01 00:46:48 -07:00
$ in_comment = 1 ;
}
if ( $ line =~ m @/\*@ ) {
$ in_comment = 1 ;
}
if ( $ line =~ m @\*/@ ) {
$ in_comment = 0 ;
}
2007-06-08 13:46:39 -07:00
# Measure the line length and indent.
( $ length , $ indent ) = line_stats ( $ line ) ;
2007-06-01 00:46:48 -07:00
# Track the previous line.
( $ prevline , $ stashline ) = ( $ stashline , $ line ) ;
( $ previndent , $ stashindent ) = ( $ stashindent , $ indent ) ;
2007-06-23 17:16:44 -07:00
} elsif ( $ realcnt == 1 ) {
$ realcnt - - ;
2007-06-01 00:46:48 -07:00
}
#make up the handle for any error we report on this line
2007-06-08 13:47:03 -07:00
$ here = "#$linenr: " ;
$ here . = "FILE: $realfile:$realline:" if ( $ realcnt != 0 ) ;
2007-06-01 00:46:48 -07:00
2007-06-08 13:47:06 -07:00
my $ hereline = "$here\n$line\n" ;
2007-07-15 23:37:22 -07:00
my $ herecurr = "$here\n$line\n" ;
my $ hereprev = "$here\n$prevline\n$line\n" ;
2007-06-01 00:46:48 -07:00
#check the patch for a signoff:
2007-06-23 17:16:44 -07:00
if ( $ line =~ /^\s*signed-off-by:/i ) {
2007-06-08 13:46:39 -07:00
# This is a signoff, if ugly, so do not double report.
$ signoff + + ;
2007-06-01 00:46:48 -07:00
if ( ! ( $ line =~ /^\s*Signed-off-by:/ ) ) {
2007-07-15 23:37:22 -07:00
WARN ( "Signed-off-by: is the preferred form\n" .
$ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
if ( $ line =~ /^\s*signed-off-by:\S/i ) {
2007-07-15 23:37:22 -07:00
WARN ( "need space after Signed-off-by:\n" .
$ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
}
2007-06-08 13:47:06 -07:00
# Check for wrappage within a valid hunk of the file
if ( $ realcnt != 0 && $ line !~ m {^(?:\+|-| |$)} ) {
2007-07-15 23:37:22 -07:00
ERROR ( "patch seems to be corrupt (line wrapped?)\n" .
$ herecurr ) ;
}
# UTF-8 regex found at http://www.w3.org/International/questions/qa-forms-utf-8.en.php
if ( ( $ realfile =~ /^$/ || $ line =~ /^\+/ ) &&
! ( $ line =~ m / ^ (
[ \ x09 \ x0A \ x0D \ x20 - \ x7E ] # ASCII
| [ \ xC2 - \ xDF ] [ \ x80 - \ xBF ] # non-overlong 2-byte
| \ xE0 [ \ xA0 - \ xBF ] [ \ x80 - \ xBF ] # excluding overlongs
| [ \ xE1 - \ xEC \ xEE \ xEF ] [ \ x80 - \ xBF ] { 2 } # straight 3-byte
| \ xED [ \ x80 - \ x9F ] [ \ x80 - \ xBF ] # excluding surrogates
| \ xF0 [ \ x90 - \ xBF ] [ \ x80 - \ xBF ] { 2 } # planes 1-3
| [ \ xF1 - \ xF3 ] [ \ x80 - \ xBF ] { 3 } # planes 4-15
| \ xF4 [ \ x80 - \ x8F ] [ \ x80 - \ xBF ] { 2 } # plane 16
) * $ / x ) ) {
ERROR ( "Invalid UTF-8\n" . $ herecurr ) ;
2007-06-08 13:47:06 -07:00
}
#ignore lines being removed
if ( $ line =~ /^-/ ) { next ; }
2007-06-01 00:46:48 -07:00
2007-06-08 13:47:06 -07:00
# check we are in a valid source file if not then ignore this hunk
next if ( $ realfile !~ /\.(h|c|s|S|pl|sh)$/ ) ;
2007-06-01 00:46:48 -07:00
#trailing whitespace
2007-06-23 17:16:44 -07:00
if ( $ line =~ /^\+.*\S\s+$/ || $ line =~ /^\+\s+$/ ) {
2007-07-15 23:37:22 -07:00
my $ herevet = "$here\n" . cat_vet ( $ line ) . "\n" ;
ERROR ( "trailing whitespace\n" . $ herevet ) ;
2007-06-01 00:46:48 -07:00
}
#80 column limit
2007-06-08 13:47:06 -07:00
if ( $ line =~ /^\+/ && ! ( $ prevline =~ /\/\*\*/ ) && $ length > 80 ) {
2007-07-15 23:37:22 -07:00
WARN ( "line over 80 characters\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
# check we are in a valid source file *.[hc] if not then ignore this hunk
next if ( $ realfile !~ /\.[hc]$/ ) ;
# at the beginning of a line any tabs must come first and anything
# more than 8 must use tabs.
if ( $ line =~ /^\+\s* \t\s*\S/ or $ line =~ /^\+\s* \s*/ ) {
2007-07-15 23:37:22 -07:00
my $ herevet = "$here\n" . cat_vet ( $ line ) . "\n" ;
ERROR ( "use tabs not spaces\n" . $ herevet ) ;
2007-06-01 00:46:48 -07:00
}
#
# The rest of our checks refer specifically to C style
# only apply those _outside_ comments.
#
next if ( $ in_comment ) ;
2007-06-23 17:16:34 -07:00
# Remove comments from the line before processing.
2007-06-08 13:47:06 -07:00
$ line =~ s@/\*.*\*/@@g ;
$ line =~ s@/\*.*@@ ;
$ line =~ s@.*\*/@@ ;
2007-06-23 17:16:34 -07:00
# Standardise the strings and chars within the input to simplify matching.
$ line = sanitise_line ( $ line ) ;
#
# Checks which may be anchored in the context.
#
2007-06-08 13:47:06 -07:00
2007-06-23 17:16:34 -07:00
# Check for switch () and associated case and default
# statements should be at the same indent.
2007-06-08 13:47:06 -07:00
if ( $ line =~ /\bswitch\s*\(.*\)/ ) {
my $ err = '' ;
my $ sep = '' ;
my @ ctx = ctx_block_outer ( $ linenr , $ realcnt ) ;
shift ( @ ctx ) ;
for my $ ctx ( @ ctx ) {
my ( $ clen , $ cindent ) = line_stats ( $ ctx ) ;
if ( $ ctx =~ /^\+\s*(case\s+|default:)/ &&
$ indent != $ cindent ) {
$ err . = "$sep$ctx\n" ;
$ sep = '' ;
} else {
$ sep = "[...]\n" ;
}
}
if ( $ err ne '' ) {
2007-07-15 23:37:22 -07:00
ERROR ( "switch and case should be at the same indent\n$hereline\n$err\n" ) ;
}
}
# if/while/etc brace do not go on next line, unless defining a do while loop,
# or if that brace on the next line is for something else
if ( $ line =~ /\b(?:(if|while|for|switch)\s*\(|do\b|else\b)/ && $ line !~ /^.#/ ) {
my @ ctx = ctx_statement ( $ linenr , $ realcnt ) ;
my $ ctx_ln = $ linenr + $# ctx + 1 ;
my $ ctx_cnt = $ realcnt - $# ctx - 1 ;
my $ ctx = join ( "\n" , @ ctx ) ;
while ( $ ctx_cnt > 0 && $ lines [ $ ctx_ln - 1 ] =~ /^-/ ) {
$ ctx_ln + + ;
$ ctx_cnt - - ;
}
##warn "line<$line>\nctx<$ctx>\nnext<$lines[$ctx_ln - 1]>";
if ( $ ctx !~ /{\s*/ && $ ctx_cnt > 0 && $ lines [ $ ctx_ln - 1 ] =~ /^\+\s*{/ ) {
ERROR ( "That { should be on the previous line\n" .
"$here\n$ctx\n$lines[$ctx_ln - 1]" ) ;
2007-06-08 13:47:06 -07:00
}
}
#ignore lines not being added
if ( $ line =~ /^[^\+]/ ) { next ; }
2007-06-23 17:16:34 -07:00
# TEST: allow direct testing of the type matcher.
if ( $ tst_type && $ line =~ /^.$Declare$/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "TEST: is type $Declare\n" . $ herecurr ) ;
2007-06-23 17:16:34 -07:00
next ;
}
#
# Checks which are anchored on the added line.
#
# check for malformed paths in #include statements (uses RAW line)
if ( $ rawline =~ m {^.#\s*include\s+[<"](.*)[">]} ) {
my $ path = $ 1 ;
if ( $ path =~ m {//} ) {
2007-07-15 23:37:22 -07:00
ERROR ( "malformed #include filename\n" .
$ herecurr ) ;
2007-06-23 17:16:34 -07:00
}
# Sanitise this special form of string.
$ path = 'X' x length ( $ path ) ;
$ line =~ s{\<.*\>} {<$path>} ;
}
2007-06-08 13:47:06 -07:00
2007-06-01 00:46:48 -07:00
# no C99 // comments
2007-06-08 13:47:06 -07:00
if ( $ line =~ m {//} ) {
2007-07-15 23:37:22 -07:00
ERROR ( "do not use C99 // comments\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-08 13:47:06 -07:00
# Remove C99 comments.
2007-06-01 00:46:48 -07:00
$ line =~ s@//.*@@ ;
#EXPORT_SYMBOL should immediately follow its function closing }.
2007-06-23 17:16:34 -07:00
if ( ( $ line =~ /EXPORT_SYMBOL.*\((.*)\)/ ) ||
( $ line =~ /EXPORT_UNUSED_SYMBOL.*\((.*)\)/ ) ) {
my $ name = $ 1 ;
2007-06-01 00:46:48 -07:00
if ( ( $ prevline !~ /^}/ ) &&
( $ prevline !~ /^\+}/ ) &&
2007-06-23 17:16:34 -07:00
( $ prevline !~ /^ }/ ) &&
( $ prevline !~ /\s$name(?:\s+$Attribute)?\s*(?:;|=)/ ) ) {
2007-07-15 23:37:22 -07:00
WARN ( "EXPORT_SYMBOL(foo); should immediately follow its function/variable\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
}
2007-06-23 17:16:34 -07:00
# check for static initialisers.
2007-06-01 00:46:48 -07:00
if ( $ line =~ /\s*static\s.*=\s+(0|NULL);/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "do not initialise statics to 0 or NULL\n" .
$ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-23 17:16:34 -07:00
# check for new typedefs, only function parameters and sparse annotations
# make sense.
if ( $ line =~ /\btypedef\s/ &&
2007-07-15 23:37:22 -07:00
$ line !~ /\btypedef\s+$Type\s+\(\s*\*$Ident\s*\)\s*\(/ &&
2007-06-23 17:16:34 -07:00
$ line !~ /\b__bitwise(?:__|)\b/ ) {
2007-07-15 23:37:22 -07:00
WARN ( "do not add new typedefs\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
# * goes on variable not on type
2007-06-23 17:16:44 -07:00
if ( $ line =~ m {\($NonptrType(\*+)(?:\s+const)?\)} ) {
2007-07-15 23:37:22 -07:00
ERROR ( "\"(foo$1)\" should be \"(foo $1)\"\n" .
$ herecurr ) ;
2007-06-23 17:16:44 -07:00
} elsif ( $ line =~ m {\($NonptrType\s+(\*+)(?!\s+const)\s+\)} ) {
2007-07-15 23:37:22 -07:00
ERROR ( "\"(foo $1 )\" should be \"(foo $1)\"\n" .
$ herecurr ) ;
2007-06-23 17:16:44 -07:00
} elsif ( $ line =~ m {$NonptrType(\*+)(?:\s+const)?\s+[A-Za-z\d_]+} ) {
2007-07-15 23:37:22 -07:00
ERROR ( "\"foo$1 bar\" should be \"foo $1bar\"\n" .
$ herecurr ) ;
2007-06-23 17:16:44 -07:00
} elsif ( $ line =~ m {$NonptrType\s+(\*+)(?!\s+const)\s+[A-Za-z\d_]+} ) {
2007-07-15 23:37:22 -07:00
ERROR ( "\"foo $1 bar\" should be \"foo $1bar\"\n" .
$ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
# # no BUG() or BUG_ON()
# if ($line =~ /\b(BUG|BUG_ON)\b/) {
# print "Try to use WARN_ON & Recovery code rather than BUG() or BUG_ON()\n";
# print "$herecurr";
# $clean = 0;
# }
2007-06-08 13:47:06 -07:00
# printk should use KERN_* levels. Note that follow on printk's on the
# same line do not need a level, so we use the current block context
# to try and find and validate the current printk. In summary the current
# printk includes all preceeding printk's which have no newline on the end.
# we assume the first bad printk is the one to report.
2007-06-01 00:46:48 -07:00
if ( $ line =~ /\bprintk\((?!KERN_)/ ) {
2007-06-08 13:47:06 -07:00
my $ ok = 0 ;
for ( my $ ln = $ linenr - 1 ; $ ln >= $ first_line ; $ ln - - ) {
#print "CHECK<$lines[$ln - 1]\n";
# we have a preceeding printk if it ends
# with "\n" ignore it, else it is to blame
if ( $ lines [ $ ln - 1 ] =~ m {\bprintk\(} ) {
if ( $ rawlines [ $ ln - 1 ] !~ m {\\n"} ) {
$ ok = 1 ;
}
last ;
}
}
if ( $ ok == 0 ) {
2007-07-15 23:37:22 -07:00
WARN ( "printk() should include KERN_ facility level\n" . $ herecurr ) ;
2007-06-08 13:47:06 -07:00
}
2007-06-01 00:46:48 -07:00
}
2007-06-23 17:16:34 -07:00
# function brace can't be on same line, except for #defines of do while,
# or if closed on same line
2007-06-23 17:16:44 -07:00
if ( ( $ line =~ /$Type\s*[A-Za-z\d_]+\(.*\).* {/ ) and
2007-06-01 00:46:48 -07:00
! ( $ line =~ /\#define.*do\s{/ ) and ! ( $ line =~ /}/ ) ) {
2007-07-15 23:37:22 -07:00
ERROR ( "open brace '{' following function declarations go on the next line\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-23 17:16:34 -07:00
# Check operator spacing.
2007-06-08 13:46:39 -07:00
# Note we expand the line with the leading + as the real
# line will be displayed with the leading + and the tabs
# will therefore also expand that way.
2007-06-01 00:46:48 -07:00
my $ opline = $ line ;
2007-06-08 13:46:39 -07:00
$ opline = expand_tabs ( $ opline ) ;
2007-06-08 13:47:06 -07:00
$ opline =~ s/^./ / ;
2007-06-01 00:46:48 -07:00
if ( ! ( $ line =~ /\#\s*include/ ) ) {
my @ elements = split ( /(<<=|>>=|<=|>=|==|!=|\+=|-=|\*=|\/=|%=|\^=|\|=|&=|->|<<|>>|<|>|=|!|~|&&|\|\||,|\^|\+\+|--|;|&|\||\+|-|\*|\/\/|\/)/ , $ opline ) ;
2007-06-08 13:47:06 -07:00
my $ off = 0 ;
2007-06-01 00:46:48 -07:00
for ( my $ n = 0 ; $ n < $# elements ; $ n += 2 ) {
2007-06-08 13:46:39 -07:00
$ off += length ( $ elements [ $ n ] ) ;
my $ a = '' ;
$ a = 'V' if ( $ elements [ $ n ] ne '' ) ;
$ a = 'W' if ( $ elements [ $ n ] =~ /\s$/ ) ;
$ a = 'B' if ( $ elements [ $ n ] =~ /(\[|\()$/ ) ;
$ a = 'O' if ( $ elements [ $ n ] eq '' ) ;
$ a = 'E' if ( $ elements [ $ n ] eq '' && $ n == 0 ) ;
2007-06-01 00:46:48 -07:00
my $ op = $ elements [ $ n + 1 ] ;
2007-06-08 13:46:39 -07:00
my $ c = '' ;
2007-06-01 00:46:48 -07:00
if ( defined $ elements [ $ n + 2 ] ) {
2007-06-08 13:46:39 -07:00
$ c = 'V' if ( $ elements [ $ n + 2 ] ne '' ) ;
$ c = 'W' if ( $ elements [ $ n + 2 ] =~ /^\s/ ) ;
$ c = 'B' if ( $ elements [ $ n + 2 ] =~ /^(\)|\]|;)/ ) ;
$ c = 'O' if ( $ elements [ $ n + 2 ] eq '' ) ;
} else {
$ c = 'E' ;
2007-06-01 00:46:48 -07:00
}
2007-06-08 13:47:06 -07:00
# Pick up the preceeding and succeeding characters.
2007-07-15 23:37:22 -07:00
my $ ca = substr ( $ opline , 0 , $ off ) ;
2007-06-08 13:47:06 -07:00
my $ cc = '' ;
2007-06-23 17:16:34 -07:00
if ( length ( $ opline ) >= ( $ off + length ( $ elements [ $ n + 1 ] ) ) ) {
2007-06-23 17:16:44 -07:00
$ cc = substr ( $ opline , $ off + length ( $ elements [ $ n + 1 ] ) ) ;
2007-06-08 13:47:06 -07:00
}
2007-07-15 23:37:22 -07:00
my $ cb = "$ca$;$cc" ;
2007-06-08 13:47:06 -07:00
2007-06-08 13:46:39 -07:00
my $ ctx = "${a}x${c}" ;
my $ at = "(ctx:$ctx)" ;
my $ ptr = ( " " x $ off ) . "^" ;
2007-07-15 23:37:22 -07:00
my $ hereptr = "$hereline$ptr\n" ;
2007-06-01 00:46:48 -07:00
##print "<$s1:$op:$s2> <$elements[$n]:$elements[$n + 1]:$elements[$n + 2]>\n";
2007-06-23 17:16:44 -07:00
# ; should have either the end of line or a space or \ after it
if ( $ op eq ';' ) {
2007-07-15 23:37:22 -07:00
if ( $ ctx !~ /.x[WEB]/ && $ cc !~ /^\\/ &&
$ cc !~ /^;/ ) {
ERROR ( "need space after that '$op' $at\n" . $ hereptr ) ;
2007-06-23 17:16:44 -07:00
}
# // is a comment
} elsif ( $ op eq '//' ) {
2007-06-01 00:46:48 -07:00
# -> should have no spaces
} elsif ( $ op eq '->' ) {
2007-06-08 13:46:39 -07:00
if ( $ ctx =~ /Wx.|.xW/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "no spaces around that '$op' $at\n" . $ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
# , must have a space on the right.
} elsif ( $ op eq ',' ) {
2007-06-23 17:16:44 -07:00
if ( $ ctx !~ /.xW|.xE/ && $ cc !~ /^}/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "need space after that '$op' $at\n" . $ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
# unary ! and unary ~ are allowed no space on the right
} elsif ( $ op eq '!' or $ op eq '~' ) {
2007-06-08 13:46:39 -07:00
if ( $ ctx !~ /[WOEB]x./ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "need space before that '$op' $at\n" . $ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-08 13:46:39 -07:00
if ( $ ctx =~ /.xW/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "no space after that '$op' $at\n" . $ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
# unary ++ and unary -- are allowed no space on one side.
} elsif ( $ op eq '++' or $ op eq '--' ) {
2007-06-23 17:16:44 -07:00
if ( $ ctx !~ /[WOB]x[^W]/ && $ ctx !~ /[^W]x[WOBE]/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "need space one side of that '$op' $at\n" . $ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-23 17:16:44 -07:00
if ( $ ctx =~ /Wx./ && $ cc =~ /^;/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "no space before that '$op' $at\n" . $ hereptr ) ;
2007-06-23 17:16:34 -07:00
}
2007-06-01 00:46:48 -07:00
# & is both unary and binary
# unary:
# a &b
# binary (consistent spacing):
# a&b OK
# a & b OK
#
# boiling down to: if there is a space on the right then there
# should be one on the left.
#
# - is the same
#
2007-06-08 13:46:39 -07:00
} elsif ( $ op eq '&' or $ op eq '-' ) {
2007-06-08 13:47:06 -07:00
if ( $ ctx !~ /VxV|[EW]x[WE]|[EWB]x[VO]/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "need space before that '$op' $at\n" . $ hereptr ) ;
2007-06-08 13:46:39 -07:00
}
2007-06-08 13:47:06 -07:00
# * is the same as & only adding:
# type:
# (foo *)
# (foo **)
#
2007-06-08 13:46:39 -07:00
} elsif ( $ op eq '*' ) {
2007-07-15 23:37:22 -07:00
if ( $ ca !~ /$Type$/ && $ cb !~ /(\*$;|$;\*)/ &&
$ ctx !~ /VxV|[EW]x[WE]|[EWB]x[VO]|OxV|WxB|BxB/ ) {
ERROR ( "need space before that '$op' $at\n" . $ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
# << and >> may either have or not have spaces both sides
} elsif ( $ op eq '<<' or $ op eq '>>' or $ op eq '+' or $ op eq '/' or
$ op eq '^' or $ op eq '|' )
{
2007-06-08 13:46:39 -07:00
if ( $ ctx !~ /VxV|WxW|VxE|WxE/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "need consistent spacing around '$op' $at\n" .
$ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
# All the others need spaces both sides.
2007-06-08 13:46:39 -07:00
} elsif ( $ ctx !~ /[EW]x[WE]/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "need spaces around that '$op' $at\n" . $ hereptr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-08 13:46:39 -07:00
$ off += length ( $ elements [ $ n + 1 ] ) ;
2007-06-01 00:46:48 -07:00
}
}
#need space before brace following if, while, etc
2007-07-15 23:37:22 -07:00
if ( $ line =~ /\(.*\){/ || $ line =~ /do{/ ) {
ERROR ( "need a space before the open brace '{'\n" . $ herecurr ) ;
}
# closing brace should have a space following it when it has anything
# on the line
if ( $ line =~ /}(?!(?:,|;|\)))\S/ ) {
ERROR ( "need a space after that close brace '}'\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
#goto labels aren't indented, allow a single space however
2007-06-08 13:46:39 -07:00
if ( $ line =~ /^.\s+[A-Za-z\d_]+:(?![0-9]+)/ and
2007-06-01 00:46:48 -07:00
! ( $ line =~ /^. [A-Za-z\d_]+:/ ) and ! ( $ line =~ /^.\s+default:/ ) ) {
2007-07-15 23:37:22 -07:00
WARN ( "labels should not be indented\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
# Need a space before open parenthesis after if, while etc
2007-06-08 13:46:39 -07:00
if ( $ line =~ /\b(if|while|for|switch)\(/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "need a space before the open parenthesis '('\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
# Check for illegal assignment in if conditional.
2007-06-23 17:16:34 -07:00
if ( $ line =~ /\bif\s*\(.*[^<>!=]=[^=].*\)/ ) {
2007-06-08 13:47:06 -07:00
#next if ($line=~/\".*\Q$op\E.*\"/ or $line=~/\'\Q$op\E\'/);
2007-07-15 23:37:22 -07:00
ERROR ( "do not use assignment in if condition\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
# Check for }<nl>else {, these must be at the same
# indent level to be relevant to each other.
if ( $ prevline =~ /}\s*$/ and $ line =~ /^.\s*else\s*/ and
$ previndent == $ indent ) {
2007-07-15 23:37:22 -07:00
ERROR ( "else should follow close brace '}'\n" . $ hereprev ) ;
2007-06-01 00:46:48 -07:00
}
#studly caps, commented out until figure out how to distinguish between use of existing and adding new
# if (($line=~/[\w_][a-z\d]+[A-Z]/) and !($line=~/print/)) {
# print "No studly caps, use _\n";
# print "$herecurr";
# $clean = 0;
# }
#no spaces allowed after \ in define
if ( $ line =~ /\#define.*\\\s$/ ) {
2007-07-15 23:37:22 -07:00
WARN ( "Whitepspace after \\ makes next lines useless\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-23 17:16:34 -07:00
#warn if <asm/foo.h> is #included and <linux/foo.h> is available (uses RAW line)
if ( $ tree && $ rawline =~ m {^.\#\s*include\s*\<asm\/(.*)\.h\>} ) {
2007-06-01 00:46:48 -07:00
my $ checkfile = "include/linux/$1.h" ;
if ( - f $ checkfile ) {
2007-07-15 23:37:22 -07:00
CHK ( "Use #include <linux/$1.h> instead of <asm/$1.h>\n" .
$ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
}
2007-06-23 17:16:44 -07:00
# if and else should not have general statements after it
if ( $ line =~ /^.\s*(?:}\s*)?else\b(.*)/ &&
2007-07-15 23:37:22 -07:00
$ 1 !~ /^\s*(?:\sif|{|\\|$)/ ) {
ERROR ( "trailing statements should be on next line\n" . $ herecurr ) ;
2007-06-23 17:16:44 -07:00
}
2007-06-23 17:16:34 -07:00
# multi-statement macros should be enclosed in a do while loop, grab the
# first statement and ensure its the whole macro if its not enclosed
# in a known goot container
if ( ( $ prevline =~ /\#define.*\\/ ) and
! ( $ prevline =~ /do\s+{/ ) and ! ( $ prevline =~ /\(\{/ ) and
! ( $ line =~ /do.*{/ ) and ! ( $ line =~ /\(\{/ ) and
! ( $ line =~ /^.\s*$Declare\s/ ) ) {
# Grab the first statement, if that is the entire macro
# its ok. This may start either on the #define line
# or the one below.
2007-06-23 17:16:44 -07:00
my $ ln = $ linenr ;
my $ cnt = $ realcnt ;
2007-06-23 17:16:34 -07:00
2007-06-23 17:16:44 -07:00
# If the macro starts on the define line start there.
if ( $ prevline !~ m {^.#\s*define\s*$Ident(?:\([^\)]*\))?\s*\\\s*$} ) {
$ ln - - ;
$ cnt + + ;
}
2007-07-15 23:37:22 -07:00
my @ ctx = ctx_statement ( $ ln , $ cnt ) ;
my $ ctx_ln = $ ln + $# ctx + 1 ;
my $ ctx = join ( "\n" , @ ctx ) ;
# Pull in any empty extension lines.
while ( $ ctx =~ /\\$/ &&
$ lines [ $ ctx_ln - 1 ] =~ /^.\s*(?:\\)?$/ ) {
$ ctx . = $ lines [ $ ctx_ln - 1 ] ;
$ ctx_ln + + ;
}
2007-06-23 17:16:44 -07:00
if ( $ ctx =~ /\\$/ ) {
if ( $ ctx =~ /;/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "Macros with multiple statements should be enclosed in a do - while loop\n" . "$here\n$ctx\n" ) ;
2007-06-23 17:16:44 -07:00
} else {
2007-07-15 23:37:22 -07:00
ERROR ( "Macros with complex values should be enclosed in parenthesis\n" . "$here\n$ctx\n" ) ;
2007-06-23 17:16:44 -07:00
}
2007-06-23 17:16:34 -07:00
}
2007-06-01 00:46:48 -07:00
}
2007-06-23 17:16:34 -07:00
# don't include deprecated include files (uses RAW line)
2007-06-08 13:46:39 -07:00
for my $ inc ( @ dep_includes ) {
2007-06-23 17:16:34 -07:00
if ( $ rawline =~ m @\#\s*include\s*\<$inc>@ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "Don't use <$inc>: see Documentation/feature-removal-schedule.txt\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
}
2007-06-08 13:46:39 -07:00
# don't use deprecated functions
for my $ func ( @ dep_functions ) {
2007-06-08 13:47:06 -07:00
if ( $ line =~ /\b$func\b/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "Don't use $func(): see Documentation/feature-removal-schedule.txt\n" . $ herecurr ) ;
2007-06-08 13:46:39 -07:00
}
}
# no volatiles please
2007-06-08 13:47:06 -07:00
if ( $ line =~ /\bvolatile\b/ && $ line !~ /\basm\s+volatile\b/ ) {
2007-07-15 23:37:22 -07:00
WARN ( "Use of volatile is usually wrong: see Documentation/volatile-considered-harmful.txt\n" . $ herecurr ) ;
2007-06-08 13:46:39 -07:00
}
2007-06-08 13:47:06 -07:00
# warn about #if 0
if ( $ line =~ /^.#\s*if\s+0\b/ ) {
2007-07-15 23:37:22 -07:00
CHK ( "if this code is redundant consider removing it\n" .
$ herecurr ) ;
2007-06-08 13:46:39 -07:00
}
2007-06-08 13:47:06 -07:00
# warn about #ifdefs in C files
# if ($line =~ /^.#\s*if(|n)def/ && ($realfile =~ /\.c$/)) {
# print "#ifdef in C files should be avoided\n";
# print "$herecurr";
# $clean = 0;
# }
2007-06-08 13:46:39 -07:00
# check for spinlock_t definitions without a comment.
if ( $ line =~ /^.\s*(struct\s+mutex|spinlock_t)\s+\S+;/ ) {
my $ which = $ 1 ;
if ( ! ctx_has_comment ( $ first_line , $ linenr ) ) {
2007-07-15 23:37:22 -07:00
CHK ( "$1 definition without comment\n" . $ herecurr ) ;
2007-06-08 13:46:39 -07:00
}
}
# check for memory barriers without a comment.
if ( $ line =~ /\b(mb|rmb|wmb|read_barrier_depends|smp_mb|smp_rmb|smp_wmb|smp_read_barrier_depends)\(/ ) {
if ( ! ctx_has_comment ( $ first_line , $ linenr ) ) {
2007-07-15 23:37:22 -07:00
CHK ( "memory barrier without comment\n" . $ herecurr ) ;
2007-06-08 13:46:39 -07:00
}
}
# check of hardware specific defines
if ( $ line =~ m @^.#\s*if.*\b(__i386__|__powerpc64__|__sun__|__s390x__)\b@ ) {
2007-07-15 23:37:22 -07:00
CHK ( "architecture specific defines should be avoided\n" . $ herecurr ) ;
2007-06-01 00:46:48 -07:00
}
2007-06-23 17:16:34 -07:00
2007-07-15 23:37:22 -07:00
# check the location of the inline attribute, that it is between
# storage class and type.
2007-06-23 17:16:34 -07:00
if ( $ line =~ /$Type\s+(?:inline|__always_inline)\b/ ||
$ line =~ /\b(?:inline|always_inline)\s+$Storage/ ) {
2007-07-15 23:37:22 -07:00
ERROR ( "inline keyword should sit between storage class and type\n" . $ herecurr ) ;
}
# check for new externs in .c files.
if ( $ line =~ /^.\s*extern\s/ && ( $ realfile =~ /\.c$/ ) ) {
WARN ( "externs should be avoided in .c files\n" . $ herecurr ) ;
}
# checks for new __setup's
if ( $ rawline =~ /\b__setup\("([^"]*)"/ ) {
my $ name = $ 1 ;
if ( ! grep ( /$name/ , @ setup_docs ) ) {
CHK ( "__setup appears un-documented -- check Documentation/kernel-parameters.txt\n" . $ herecurr ) ;
}
2007-06-23 17:16:34 -07:00
}
2007-06-01 00:46:48 -07:00
}
if ( $ chk_patch && ! $ is_patch ) {
2007-07-15 23:37:22 -07:00
ERROR ( "Does not appear to be a unified-diff format patch\n" ) ;
2007-06-01 00:46:48 -07:00
}
if ( $ is_patch && $ chk_signoff && $ signoff == 0 ) {
2007-07-15 23:37:22 -07:00
ERROR ( "Missing Signed-off-by: line(s)\n" ) ;
2007-06-01 00:46:48 -07:00
}
if ( $ clean == 1 && $ quiet == 0 ) {
print "Your patch has no obvious style problems and is ready for submission.\n"
}
if ( $ clean == 0 && $ quiet == 0 ) {
print "Your patch has style problems, please review. If any of these errors\n" ;
print "are false positives report them to the maintainer, see\n" ;
print "CHECKPATCH in MAINTAINERS.\n" ;
}
return $ clean ;
}