Skip to content

Commit 2f85e94

Browse files
albankurtiakpm00
authored andcommitted
checkpatch: add warning for pr_* and dev_* macros without a trailing newline
Add a new check in scripts/checkpatch.pl to detect usage of pr_(level) and dev_(level) macros (for both C and Rust) when the string literal does not end with '\n'. Missing trailing newlines can lead to incomplete log lines that do not appear properly in dmesg or in console output. To show an example of this working after applying the patch we can run the script on the commit that likely motivated this need/issue: ./scripts/checkpatch.pl --strict -g "f431c5c581fa1" Also, the patch is able to handle correctly if there is a printing call without a newline which then has a newline printed via pr_cont for both Rust and C alike. If there is no newline printed and the patch ends or there is another pr_* call before a newline with pr_cont is printed it will show a warning. Not implemented for dev_cont because it is not clear to me if that is used at all. One false warning that will be generated due to this change is in case we have a patch that modifies a `pr_* call without a newline` which has a pr_cont with a newline following it. In this case there will be a warning but because the patch does not include the following pr_cont it will warn there is nothing creating a newline. I have modified the warning to be softer due to this known problem. I have tested with comments, whitespace, differen orders of pr_* calls and pr_cont and the only case that I suspect to be a problem is the one outlined above. Link: https://lkml.kernel.org/r/[email protected] Signed-off-by: Alban Kurti <[email protected]> Suggested-by: Miguel Ojeda <[email protected]> Closes: Rust-for-Linux#1140 Cc: Alex Gaynor <[email protected]> Cc: Alice Ryhl <[email protected]> Cc: Andreas Hindborg <[email protected]> Cc: Andy Whitcroft <[email protected]> Cc: Benno Lossin <[email protected]> Cc: Björn Roy Baron <[email protected]> Cc: Boqun Feng <[email protected]> Cc: Dwaipayan Ray <[email protected]> Cc: Gary Guo <[email protected]> Cc: Joe Perches <[email protected]> Cc: Lukas Bulwahn <[email protected]> Cc: Trevor Gross <[email protected]> Cc: Charalampos Mitrodimas <[email protected]> Signed-off-by: Andrew Morton <[email protected]>
1 parent f05cfcf commit 2f85e94

File tree

1 file changed

+96
-0
lines changed

1 file changed

+96
-0
lines changed

scripts/checkpatch.pl

+96
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,8 @@
7777

7878
my %maybe_linker_symbol; # for externs in c exceptions, when seen in *vmlinux.lds.h
7979

80+
my $pending_log = undef;
81+
8082
sub help {
8183
my ($exitcode) = @_;
8284

@@ -3878,6 +3880,91 @@ sub process {
38783880
}
38793881
}
38803882

3883+
# check for pr_* and dev_* logs without a newline for C and Rust files to avoid missing log messages
3884+
my $pr_cont_pattern = qr{
3885+
\b
3886+
pr_cont!?
3887+
\s*
3888+
\(
3889+
\s*
3890+
"([^"]*)"
3891+
[^)]*
3892+
\)
3893+
}x;
3894+
my $log_macro_pattern = qr{
3895+
\b
3896+
(
3897+
pr_(?:emerg|alert|crit|err|warn|notice|info|debug)
3898+
| dev_(?:emerg|alert|crit|err|warn|notice|info|dbg)
3899+
)
3900+
(!?)
3901+
\s*
3902+
\(
3903+
\s*
3904+
"([^"]*)"
3905+
}x;
3906+
3907+
if ($realfile =~ /\.(?:c|h|rs)$/) {
3908+
if ($rawline =~ /^\+/) {
3909+
my $cleanline = $rawline;
3910+
$cleanline =~ s/^[+\s]+//;
3911+
$cleanline =~ s/\r?$//;
3912+
$cleanline =~ s{/\*.*?\*/}{}g;
3913+
$cleanline =~ s{//.*}{}g;
3914+
3915+
if ($pending_log) {
3916+
if ($cleanline =~ /$pr_cont_pattern/) {
3917+
my $cont_string_arg = $1;
3918+
if ($cont_string_arg =~ /\\n$/) {
3919+
$pending_log = undef;
3920+
}
3921+
} elsif ($cleanline =~ /$log_macro_pattern/) {
3922+
WARN($pending_log->{lang} . "_LOG_NO_NEWLINE",
3923+
"Possible usage of $pending_log->{macro_call} without a trailing newline.\n" .
3924+
$pending_log->{herecurr});
3925+
3926+
$pending_log = undef;
3927+
3928+
my $macro_call = $1;
3929+
my $maybe_excl = $2;
3930+
my $string_arg = $3;
3931+
$string_arg =~ s/\s+$//;
3932+
3933+
if ($realfile =~ /\.rs$/ && $maybe_excl ne '!') {
3934+
return;
3935+
}
3936+
3937+
if ($string_arg !~ /\\n$/ && $string_arg !~ /\n$/) {
3938+
$pending_log = {
3939+
macro_call => $macro_call,
3940+
herecurr => $herecurr,
3941+
lang => ($realfile =~ /\.rs$/) ? "Rust" : "C",
3942+
};
3943+
}
3944+
}
3945+
} else {
3946+
if ($cleanline =~ /$log_macro_pattern/) {
3947+
my $macro_call = $1;
3948+
my $maybe_excl = $2;
3949+
my $string_arg = $3;
3950+
$string_arg =~ s/\s+$//;
3951+
3952+
if ($realfile =~ /\.rs$/ && $maybe_excl ne '!') {
3953+
return;
3954+
}
3955+
3956+
if ($string_arg !~ /\\n$/ && $string_arg !~ /\n$/) {
3957+
$pending_log = {
3958+
macro_call => $macro_call,
3959+
herecurr => $herecurr,
3960+
lang => ($realfile =~ /\.rs$/) ? "Rust" : "C",
3961+
};
3962+
}
3963+
}
3964+
}
3965+
}
3966+
}
3967+
38813968
# check for .L prefix local symbols in .S files
38823969
if ($realfile =~ /\.S$/ &&
38833970
$line =~ /^\+\s*(?:[A-Z]+_)?SYM_[A-Z]+_(?:START|END)(?:_[A-Z_]+)?\s*\(\s*\.L/) {
@@ -7670,6 +7757,15 @@ sub process {
76707757
}
76717758
}
76727759

7760+
# pending log means a pr_* without an ending newline has not
7761+
# been followed by a pr_cont call with a newline at the end
7762+
if ($pending_log) {
7763+
WARN($pending_log->{lang} . "_LOG_NO_NEWLINE",
7764+
"Usage of $pending_log->{macro_call} without a trailing newline.\n" .
7765+
$pending_log->{herecurr});
7766+
$pending_log = undef;
7767+
}
7768+
76737769
# If we have no input at all, then there is nothing to report on
76747770
# so just keep quiet.
76757771
if ($#rawlines == -1) {

0 commit comments

Comments
 (0)