2 # SPDX-License-Identifier: GPL-2.0
4 # (c) 2007, Joe Perches <joe@perches.com>
5 # created from checkpatch.pl
7 # Print selected MAINTAINERS information for
8 # the files modified in a patch or for a file
10 # usage: perl scripts/get_maintainer.pl [OPTIONS] <patch>
11 # perl scripts/get_maintainer.pl [OPTIONS] -f <file>
19 use Getopt
::Long
qw(:config no_auto_abbrev);
23 my $cur_path = fastgetcwd
() . '/';
26 my $email_usename = 1;
27 my $email_maintainer = 1;
28 my $email_reviewer = 1;
30 my $email_moderated_list = 1;
31 my $email_subscriber_list = 0;
32 my $email_git_penguin_chiefs = 0;
34 my $email_git_all_signature_types = 0;
35 my $email_git_blame = 0;
36 my $email_git_blame_signatures = 1;
37 my $email_git_fallback = 1;
38 my $email_git_min_signatures = 1;
39 my $email_git_max_maintainers = 5;
40 my $email_git_min_percent = 5;
41 my $email_git_since = "1-year-ago";
42 my $email_hg_since = "-365";
44 my $email_remove_duplicates = 1;
45 my $email_use_mailmap = 1;
46 my $output_multiline = 1;
47 my $output_separator = ", ";
49 my $output_rolestats = 1;
50 my $output_section_maxlen = 50;
60 my $from_filename = 0;
61 my $pattern_depth = 0;
62 my $self_test = undef;
65 my $find_maintainer_files = 0;
71 my %commit_author_hash;
72 my %commit_signer_hash;
74 my @penguin_chief = ();
75 push(@penguin_chief, "Linus Torvalds:torvalds\@linux-foundation.org");
76 #Andrew wants in on most everything - 2009/01/14
77 #push(@penguin_chief, "Andrew Morton:akpm\@linux-foundation.org");
79 my @penguin_chief_names = ();
80 foreach my $chief (@penguin_chief) {
81 if ($chief =~ m/^(.*):(.*)/) {
84 push(@penguin_chief_names, $chief_name);
87 my $penguin_chiefs = "\(" . join("|", @penguin_chief_names) . "\)";
89 # Signature types of people who are either
90 # a) responsible for the code in question, or
91 # b) familiar enough with it to give relevant feedback
92 my @signature_tags = ();
93 push(@signature_tags, "Signed-off-by:");
94 push(@signature_tags, "Reviewed-by:");
95 push(@signature_tags, "Acked-by:");
97 my $signature_pattern = "\(" . join("|", @signature_tags) . "\)";
99 # rfc822 email address - preloaded methods go here.
100 my $rfc822_lwsp = "(?:(?:\\r\\n)?[ \\t])";
101 my $rfc822_char = '[\\000-\\377]';
103 # VCS command support: class-like functions and strings
108 "execute_cmd" => \
&git_execute_cmd
,
109 "available" => '(which("git") ne "") && (-e ".git")',
110 "find_signers_cmd" =>
111 "git log --no-color --follow --since=\$email_git_since " .
112 '--numstat --no-merges ' .
113 '--format="GitCommit: %H%n' .
114 'GitAuthor: %an <%ae>%n' .
119 "find_commit_signers_cmd" =>
120 "git log --no-color " .
122 '--format="GitCommit: %H%n' .
123 'GitAuthor: %an <%ae>%n' .
128 "find_commit_author_cmd" =>
129 "git log --no-color " .
131 '--format="GitCommit: %H%n' .
132 'GitAuthor: %an <%ae>%n' .
134 'GitSubject: %s%n"' .
136 "blame_range_cmd" => "git blame -l -L \$diff_start,+\$diff_length \$file",
137 "blame_file_cmd" => "git blame -l \$file",
138 "commit_pattern" => "^GitCommit: ([0-9a-f]{40,40})",
139 "blame_commit_pattern" => "^([0-9a-f]+) ",
140 "author_pattern" => "^GitAuthor: (.*)",
141 "subject_pattern" => "^GitSubject: (.*)",
142 "stat_pattern" => "^(\\d+)\\t(\\d+)\\t\$file\$",
143 "file_exists_cmd" => "git ls-files \$file",
144 "list_files_cmd" => "git ls-files \$file",
148 "execute_cmd" => \
&hg_execute_cmd
,
149 "available" => '(which("hg") ne "") && (-d ".hg")',
150 "find_signers_cmd" =>
151 "hg log --date=\$email_hg_since " .
152 "--template='HgCommit: {node}\\n" .
153 "HgAuthor: {author}\\n" .
154 "HgSubject: {desc}\\n'" .
156 "find_commit_signers_cmd" =>
158 "--template='HgSubject: {desc}\\n'" .
160 "find_commit_author_cmd" =>
162 "--template='HgCommit: {node}\\n" .
163 "HgAuthor: {author}\\n" .
164 "HgSubject: {desc|firstline}\\n'" .
166 "blame_range_cmd" => "", # not supported
167 "blame_file_cmd" => "hg blame -n \$file",
168 "commit_pattern" => "^HgCommit: ([0-9a-f]{40,40})",
169 "blame_commit_pattern" => "^([ 0-9a-f]+):",
170 "author_pattern" => "^HgAuthor: (.*)",
171 "subject_pattern" => "^HgSubject: (.*)",
172 "stat_pattern" => "^(\\d+)\t(\\d+)\t\$file\$",
173 "file_exists_cmd" => "hg files \$file",
174 "list_files_cmd" => "hg manifest -R \$file",
177 my $conf = which_conf
(".get_maintainer.conf");
180 open(my $conffile, '<', "$conf")
181 or warn "$P: Can't find a readable .get_maintainer.conf file $!\n";
183 while (<$conffile>) {
186 $line =~ s/\s*\n?$//g;
190 next if ($line =~ m/^\s*#/);
191 next if ($line =~ m/^\s*$/);
193 my @words = split(" ", $line);
194 foreach my $word (@words) {
195 last if ($word =~ m/^#/);
196 push (@conf_args, $word);
200 unshift(@ARGV, @conf_args) if @conf_args;
203 my @ignore_emails = ();
204 my $ignore_file = which_conf
(".get_maintainer.ignore");
205 if (-f
$ignore_file) {
206 open(my $ignore, '<', "$ignore_file")
207 or warn "$P: Can't find a readable .get_maintainer.ignore file $!\n";
211 $line =~ s/\s*\n?$//;
216 next if ($line =~ m/^\s*$/);
217 if (rfc822_valid
($line)) {
218 push(@ignore_emails, $line);
226 if ($_ =~ /^-{1,2}self-test(?:=|$)/) {
227 die "$P: using --self-test does not allow any other option or argument\n";
234 'git!' => \
$email_git,
235 'git-all-signature-types!' => \
$email_git_all_signature_types,
236 'git-blame!' => \
$email_git_blame,
237 'git-blame-signatures!' => \
$email_git_blame_signatures,
238 'git-fallback!' => \
$email_git_fallback,
239 'git-chief-penguins!' => \
$email_git_penguin_chiefs,
240 'git-min-signatures=i' => \
$email_git_min_signatures,
241 'git-max-maintainers=i' => \
$email_git_max_maintainers,
242 'git-min-percent=i' => \
$email_git_min_percent,
243 'git-since=s' => \
$email_git_since,
244 'hg-since=s' => \
$email_hg_since,
245 'i|interactive!' => \
$interactive,
246 'remove-duplicates!' => \
$email_remove_duplicates,
247 'mailmap!' => \
$email_use_mailmap,
248 'm!' => \
$email_maintainer,
249 'r!' => \
$email_reviewer,
250 'n!' => \
$email_usename,
251 'l!' => \
$email_list,
252 'moderated!' => \
$email_moderated_list,
253 's!' => \
$email_subscriber_list,
254 'multiline!' => \
$output_multiline,
255 'roles!' => \
$output_roles,
256 'rolestats!' => \
$output_rolestats,
257 'separator=s' => \
$output_separator,
258 'subsystem!' => \
$subsystem,
259 'status!' => \
$status,
263 'letters=s' => \
$letters,
264 'pattern-depth=i' => \
$pattern_depth,
265 'k|keywords!' => \
$keywords,
266 'sections!' => \
$sections,
267 'fe|file-emails!' => \
$file_emails,
268 'f|file' => \
$from_filename,
269 'find-maintainer-files' => \
$find_maintainer_files,
270 'mpath|maintainer-path=s' => \
$maintainer_path,
271 'self-test:s' => \
$self_test,
272 'v|version' => \
$version,
273 'h|help|usage' => \
$help,
275 die "$P: invalid argument - use --help if necessary\n";
284 print("${P} ${V}\n");
288 if (defined $self_test) {
289 read_all_maintainer_files
();
294 if (-t STDIN
&& !@ARGV) {
295 # We're talking to a terminal, but have no command line arguments.
296 die "$P: missing patchfile or -f file - use --help if necessary\n";
299 $output_multiline = 0 if ($output_separator ne ", ");
300 $output_rolestats = 1 if ($interactive);
301 $output_roles = 1 if ($output_rolestats);
303 if ($sections || $letters ne "") {
314 my $selections = $email + $scm + $status + $subsystem + $web;
315 if ($selections == 0) {
316 die "$P: Missing required option: email, scm, status, subsystem or web\n";
321 ($email_maintainer + $email_reviewer +
322 $email_list + $email_subscriber_list +
323 $email_git + $email_git_penguin_chiefs + $email_git_blame) == 0) {
324 die "$P: Please select at least 1 email option\n";
327 if ($tree && !top_of_kernel_tree
($lk_path)) {
328 die "$P: The current directory does not appear to be "
329 . "a linux kernel source tree.\n";
332 ## Read MAINTAINERS for type/value pairs
337 my @self_test_info = ();
339 sub read_maintainer_file
{
342 open (my $maint, '<', "$file")
343 or die "$P: Can't open MAINTAINERS file '$file': $!\n";
349 if ($line =~ m/^([A-Z]):\s*(.*)/) {
353 ##Filename pattern matching
354 if ($type eq "F" || $type eq "X") {
355 $value =~ s@\
.@
\\\
.@g; ##Convert . to \.
356 $value =~ s/\*/\.\*/g; ##Convert * to .*
357 $value =~ s/\?/\./g; ##Convert ? to .
358 ##if pattern is a directory and it lacks a trailing slash, add one
360 $value =~ s@
([^/])$@$1/@
;
362 } elsif ($type eq "K") {
363 $keyword_hash{@typevalue} = $value;
365 push(@typevalue, "$type:$value");
366 } elsif (!(/^\s*$/ || /^\s*\#/)) {
367 push(@typevalue, $line);
369 if (defined $self_test) {
370 push(@self_test_info, {file
=>$file, linenr
=>$i, line
=>$line});
377 sub find_is_maintainer_file
{
379 return if ($file !~ m@
/MAINTAINERS
$@
);
380 $file = $File::Find
::name
;
381 return if (! -f
$file);
382 push(@mfiles, $file);
385 sub find_ignore_git
{
386 return grep { $_ !~ /^\.git$/; } @_;
389 read_all_maintainer_files
();
391 sub read_all_maintainer_files
{
392 my $path = "${lk_path}MAINTAINERS";
393 if (defined $maintainer_path) {
394 $path = $maintainer_path;
395 # Perl Cookbook tilde expansion if necessary
396 $path =~ s@
^~([^/]*)@
$1 ?
(getpwnam($1))[7] : ( $ENV{HOME
} || $ENV{LOGDIR
} || (getpwuid($<))[7])@ex;
400 $path .= '/' if ($path !~ m@
/$@
);
401 if ($find_maintainer_files) {
402 find
( { wanted
=> \
&find_is_maintainer_file
,
403 preprocess
=> \
&find_ignore_git
,
407 opendir(DIR
, "$path") or die $!;
408 my @files = readdir(DIR
);
410 foreach my $file (@files) {
411 push(@mfiles, "$path$file") if ($file !~ /^\./);
414 } elsif (-f
"$path") {
415 push(@mfiles, "$path");
417 die "$P: MAINTAINER file not found '$path'\n";
419 die "$P: No MAINTAINER files found in '$path'\n" if (scalar(@mfiles) == 0);
420 foreach my $file (@mfiles) {
421 read_maintainer_file
("$file");
426 # Read mail address map
439 return if (!$email_use_mailmap || !(-f
"${lk_path}.mailmap"));
441 open(my $mailmap_file, '<', "${lk_path}.mailmap")
442 or warn "$P: Can't open .mailmap: $!\n";
444 while (<$mailmap_file>) {
445 s/#.*$//; #strip comments
446 s/^\s+|\s+$//g; #trim
448 next if (/^\s*$/); #skip empty lines
449 #entries have one of the following formats:
452 # name1 <mail1> <mail2>
453 # name1 <mail1> name2 <mail2>
454 # (see man git-shortlog)
456 if (/^([^<]+)<([^>]+)>$/) {
460 $real_name =~ s/\s+$//;
461 ($real_name, $address) = parse_email
("$real_name <$address>");
462 $mailmap->{names
}->{$address} = $real_name;
464 } elsif (/^<([^>]+)>\s*<([^>]+)>$/) {
465 my $real_address = $1;
466 my $wrong_address = $2;
468 $mailmap->{addresses
}->{$wrong_address} = $real_address;
470 } elsif (/^(.+)<([^>]+)>\s*<([^>]+)>$/) {
472 my $real_address = $2;
473 my $wrong_address = $3;
475 $real_name =~ s/\s+$//;
476 ($real_name, $real_address) =
477 parse_email
("$real_name <$real_address>");
478 $mailmap->{names
}->{$wrong_address} = $real_name;
479 $mailmap->{addresses
}->{$wrong_address} = $real_address;
481 } elsif (/^(.+)<([^>]+)>\s*(.+)\s*<([^>]+)>$/) {
483 my $real_address = $2;
485 my $wrong_address = $4;
487 $real_name =~ s/\s+$//;
488 ($real_name, $real_address) =
489 parse_email
("$real_name <$real_address>");
491 $wrong_name =~ s/\s+$//;
492 ($wrong_name, $wrong_address) =
493 parse_email
("$wrong_name <$wrong_address>");
495 my $wrong_email = format_email
($wrong_name, $wrong_address, 1);
496 $mailmap->{names
}->{$wrong_email} = $real_name;
497 $mailmap->{addresses
}->{$wrong_email} = $real_address;
500 close($mailmap_file);
503 ## use the filenames on the command line or find the filenames in the patchfiles
507 my @keyword_tvi = ();
508 my @file_emails = ();
511 push(@ARGV, "&STDIN");
514 foreach my $file (@ARGV) {
515 if ($file ne "&STDIN") {
516 ##if $file is a directory and it lacks a trailing slash, add one
518 $file =~ s@
([^/])$@$1/@
;
519 } elsif (!(-f
$file)) {
520 die "$P: file '${file}' not found\n";
523 if ($from_filename || ($file ne "&STDIN" && vcs_file_exists
($file))) {
524 $file =~ s/^\Q${cur_path}\E//; #strip any absolute path
525 $file =~ s/^\Q${lk_path}\E//; #or the path to the lk tree
527 if ($file ne "MAINTAINERS" && -f
$file && ($keywords || $file_emails)) {
528 open(my $f, '<', $file)
529 or die "$P: Can't open $file: $!\n";
530 my $text = do { local($/) ; <$f> };
533 foreach my $line (keys %keyword_hash) {
534 if ($text =~ m/$keyword_hash{$line}/x) {
535 push(@keyword_tvi, $line);
540 my @poss_addr = $text =~ m
$[A
-Za
-zÀ
-ÿ
\"\' \
,\
.\
+-]*\s
*[\
,]*\s
*[\
(\
<\
{]{0,1}[A
-Za
-z0
-9_\
.\
+-]+\@
[A
-Za
-z0
-9\
.-]+\
.[A
-Za
-z0
-9]+[\
)\
>\
}]{0,1}$g;
541 push(@file_emails, clean_file_emails
(@poss_addr));
545 my $file_cnt = @files;
548 open(my $patch, "< $file")
549 or die "$P: Can't open $file: $!\n";
551 # We can check arbitrary information before the patch
552 # like the commit message, mail headers, etc...
553 # This allows us to match arbitrary keywords against any part
554 # of a git format-patch generated file (subject tags, etc...)
556 my $patch_prefix = ""; #Parsing the intro
560 if (m/^ mode change [0-7]+ => [0-7]+ (\S+)\s*$/) {
562 push(@files, $filename);
563 } elsif (m/^rename (?:from|to) (\S+)\s*$/) {
565 push(@files, $filename);
566 } elsif (m/^diff --git a\/(\S
+) b\
/(\S+)\s*$/) {
569 push(@files, $filename1);
570 push(@files, $filename2);
571 } elsif (m/^\+\+\+\s+(\S+)/ or m/^---\s+(\S+)/) {
573 $filename =~ s@
^[^/]*/@@
;
575 $lastfile = $filename;
576 push(@files, $filename);
577 $patch_prefix = "^[+-].*"; #Now parsing the actual patch
578 } elsif (m/^\@\@ -(\d+),(\d+)/) {
579 if ($email_git_blame) {
580 push(@range, "$lastfile:$1:$2");
582 } elsif ($keywords) {
583 foreach my $line (keys %keyword_hash) {
584 if ($patch_line =~ m/${patch_prefix}$keyword_hash{$line}/x) {
585 push(@keyword_tvi, $line);
592 if ($file_cnt == @files) {
593 warn "$P: file '${file}' doesn't appear to be a patch. "
594 . "Add -f to options?\n";
596 @files = sort_and_uniq
(@files);
600 @file_emails = uniq
(@file_emails);
603 my %email_hash_address;
611 my %deduplicate_name_hash = ();
612 my %deduplicate_address_hash = ();
614 my @maintainers = get_maintainers
();
617 @maintainers = merge_email
(@maintainers);
618 output
(@maintainers);
627 @status = uniq
(@status);
632 @subsystem = uniq
(@subsystem);
647 my @section_headers = ();
650 @lsfiles = vcs_list_files
($lk_path);
652 for my $x (@self_test_info) {
655 ## Section header duplication and missing section content
656 if (($self_test eq "" || $self_test =~ /\bsections\b/) &&
657 $x->{line
} =~ /^\S[^:]/ &&
658 defined $self_test_info[$index] &&
659 $self_test_info[$index]->{line
} =~ /^([A-Z]):\s*\S/) {
664 if (grep(m@
^\Q
$x->{line
}\E@
, @section_headers)) {
665 print("$x->{file}:$x->{linenr}: warning: duplicate section header\t$x->{line}\n");
667 push(@section_headers, $x->{line
});
669 my $nextline = $index;
670 while (defined $self_test_info[$nextline] &&
671 $self_test_info[$nextline]->{line
} =~ /^([A-Z]):\s*(\S.*)/) {
677 } elsif ($type eq "F" || $type eq "N") {
679 } elsif ($type eq "M" || $type eq "R" || $type eq "L") {
684 if (!$has_ML && $status !~ /orphan|obsolete/i) {
685 print("$x->{file}:$x->{linenr}: warning: section without email address\t$x->{line}\n");
688 print("$x->{file}:$x->{linenr}: warning: section without status \t$x->{line}\n");
691 print("$x->{file}:$x->{linenr}: warning: section without file pattern\t$x->{line}\n");
695 next if ($x->{line
} !~ /^([A-Z]):\s*(.*)/);
700 ## Filename pattern matching
701 if (($type eq "F" || $type eq "X") &&
702 ($self_test eq "" || $self_test =~ /\bpatterns\b/)) {
703 $value =~ s@\
.@
\\\
.@g; ##Convert . to \.
704 $value =~ s/\*/\.\*/g; ##Convert * to .*
705 $value =~ s/\?/\./g; ##Convert ? to .
706 ##if pattern is a directory and it lacks a trailing slash, add one
708 $value =~ s@
([^/])$@$1/@
;
710 if (!grep(m@
^$value@
, @lsfiles)) {
711 print("$x->{file}:$x->{linenr}: warning: no file matches\t$x->{line}\n");
715 } elsif (($type eq "W" || $type eq "Q" || $type eq "B") &&
716 $value =~ /^https?:/ &&
717 ($self_test eq "" || $self_test =~ /\blinks\b/)) {
718 next if (grep(m@
^\Q
$value\E
$@
, @good_links));
720 if (grep(m@
^\Q
$value\E
$@
, @bad_links)) {
723 my $output = `wget --spider -q --no-check-certificate --timeout 10 --tries 1 $value`;
725 push(@good_links, $value);
727 push(@bad_links, $value);
732 print("$x->{file}:$x->{linenr}: warning: possible bad link\t$x->{line}\n");
736 } elsif ($type eq "T" &&
737 ($self_test eq "" || $self_test =~ /\bscm\b/)) {
738 next if (grep(m@
^\Q
$value\E
$@
, @good_links));
740 if (grep(m@
^\Q
$value\E
$@
, @bad_links)) {
742 } elsif ($value !~ /^(?:git|quilt|hg)\s+\S/) {
743 print("$x->{file}:$x->{linenr}: warning: malformed entry\t$x->{line}\n");
744 } elsif ($value =~ /^git\s+(\S+)(\s+([^\(]+\S+))?/) {
748 my $output = `git ls-remote --exit-code -h "$url" $branch > /dev/null 2>&1`;
750 push(@good_links, $value);
752 push(@bad_links, $value);
755 } elsif ($value =~ /^(?:quilt|hg)\s+(https?:\S+)/) {
757 my $output = `wget --spider -q --no-check-certificate --timeout 10 --tries 1 $url`;
759 push(@good_links, $value);
761 push(@bad_links, $value);
766 print("$x->{file}:$x->{linenr}: warning: possible bad link\t$x->{line}\n");
772 sub ignore_email_address
{
775 foreach my $ignore (@ignore_emails) {
776 return 1 if ($ignore eq $address);
782 sub range_is_maintained
{
783 my ($start, $end) = @_;
785 for (my $i = $start; $i < $end; $i++) {
786 my $line = $typevalue[$i];
787 if ($line =~ m/^([A-Z]):\s*(.*)/) {
791 if ($value =~ /(maintain|support)/i) {
800 sub range_has_maintainer
{
801 my ($start, $end) = @_;
803 for (my $i = $start; $i < $end; $i++) {
804 my $line = $typevalue[$i];
805 if ($line =~ m/^([A-Z]):\s*(.*)/) {
816 sub get_maintainers
{
817 %email_hash_name = ();
818 %email_hash_address = ();
819 %commit_author_hash = ();
820 %commit_signer_hash = ();
828 %deduplicate_name_hash = ();
829 %deduplicate_address_hash = ();
830 if ($email_git_all_signature_types) {
831 $signature_pattern = "(.+?)[Bb][Yy]:";
833 $signature_pattern = "\(" . join("|", @signature_tags) . "\)";
836 # Find responsible parties
838 my %exact_pattern_match_hash = ();
840 foreach my $file (@files) {
843 my $tvi = find_first_section
();
844 while ($tvi < @typevalue) {
845 my $start = find_starting_index
($tvi);
846 my $end = find_ending_index
($tvi);
850 #Do not match excluded file patterns
852 for ($i = $start; $i < $end; $i++) {
853 my $line = $typevalue[$i];
854 if ($line =~ m/^([A-Z]):\s*(.*)/) {
858 if (file_match_pattern
($file, $value)) {
867 for ($i = $start; $i < $end; $i++) {
868 my $line = $typevalue[$i];
869 if ($line =~ m/^([A-Z]):\s*(.*)/) {
873 if (file_match_pattern
($file, $value)) {
874 my $value_pd = ($value =~ tr@
/@@
);
875 my $file_pd = ($file =~ tr@
/@@
);
876 $value_pd++ if (substr($value,-1,1) ne "/");
877 $value_pd = -1 if ($value =~ /^\.\*/);
878 if ($value_pd >= $file_pd &&
879 range_is_maintained
($start, $end) &&
880 range_has_maintainer
($start, $end)) {
881 $exact_pattern_match_hash{$file} = 1;
883 if ($pattern_depth == 0 ||
884 (($file_pd - $value_pd) < $pattern_depth)) {
885 $hash{$tvi} = $value_pd;
888 } elsif ($type eq 'N') {
889 if ($file =~ m/$value/x) {
899 foreach my $line (sort {$hash{$b} <=> $hash{$a}} keys %hash) {
900 add_categories
($line);
903 my $start = find_starting_index
($line);
904 my $end = find_ending_index
($line);
905 for ($i = $start; $i < $end; $i++) {
906 my $line = $typevalue[$i];
907 if ($line =~ /^[FX]:/) { ##Restore file patterns
908 $line =~ s/([^\\])\.([^\*])/$1\?$2/g;
909 $line =~ s/([^\\])\.$/$1\?/g; ##Convert . back to ?
910 $line =~ s/\\\./\./g; ##Convert \. to .
911 $line =~ s/\.\*/\*/g; ##Convert .* to *
913 my $count = $line =~ s/^([A-Z]):/$1:\t/g;
914 if ($letters eq "" || (!$count || $letters =~ /$1/i)) {
924 @keyword_tvi = sort_and_uniq
(@keyword_tvi);
925 foreach my $line (@keyword_tvi) {
926 add_categories
($line);
930 foreach my $email (@email_to, @list_to) {
931 $email->[0] = deduplicate_email
($email->[0]);
934 foreach my $file (@files) {
936 ($email_git || ($email_git_fallback &&
937 !$exact_pattern_match_hash{$file}))) {
938 vcs_file_signoffs
($file);
940 if ($email && $email_git_blame) {
941 vcs_file_blame
($file);
946 foreach my $chief (@penguin_chief) {
947 if ($chief =~ m/^(.*):(.*)/) {
950 $email_address = format_email
($1, $2, $email_usename);
951 if ($email_git_penguin_chiefs) {
952 push(@email_to, [$email_address, 'chief penguin']);
954 @email_to = grep($_->[0] !~ /${email_address}/, @email_to);
959 foreach my $email (@file_emails) {
960 my ($name, $address) = parse_email
($email);
962 my $tmp_email = format_email
($name, $address, $email_usename);
963 push_email_address
($tmp_email, '');
964 add_role
($tmp_email, 'in file');
969 if ($email || $email_list) {
971 @to = (@to, @email_to);
974 @to = (@to, @list_to);
979 @to = interactive_get_maintainers
(\
@to);
985 sub file_match_pattern
{
986 my ($file, $pattern) = @_;
987 if (substr($pattern, -1) eq "/") {
988 if ($file =~ m@
^$pattern@
) {
992 if ($file =~ m@
^$pattern@
) {
993 my $s1 = ($file =~ tr@
/@@
);
994 my $s2 = ($pattern =~ tr@
/@@
);
1005 usage: $P [options] patchfile
1006 $P [options] -f file|directory
1009 MAINTAINER field selection options:
1010 --email => print email address(es) if any
1011 --git => include recent git \*-by: signers
1012 --git-all-signature-types => include signers regardless of signature type
1013 or use only ${signature_pattern} signers (default: $email_git_all_signature_types)
1014 --git-fallback => use git when no exact MAINTAINERS pattern (default: $email_git_fallback)
1015 --git-chief-penguins => include ${penguin_chiefs}
1016 --git-min-signatures => number of signatures required (default: $email_git_min_signatures)
1017 --git-max-maintainers => maximum maintainers to add (default: $email_git_max_maintainers)
1018 --git-min-percent => minimum percentage of commits required (default: $email_git_min_percent)
1019 --git-blame => use git blame to find modified commits for patch or file
1020 --git-blame-signatures => when used with --git-blame, also include all commit signers
1021 --git-since => git history to use (default: $email_git_since)
1022 --hg-since => hg history to use (default: $email_hg_since)
1023 --interactive => display a menu (mostly useful if used with the --git option)
1024 --m => include maintainer(s) if any
1025 --r => include reviewer(s) if any
1026 --n => include name 'Full Name <addr\@domain.tld>'
1027 --l => include list(s) if any
1028 --moderated => include moderated lists(s) if any (default: true)
1029 --s => include subscriber only list(s) if any (default: false)
1030 --remove-duplicates => minimize duplicate email names/addresses
1031 --roles => show roles (status:subsystem, git-signer, list, etc...)
1032 --rolestats => show roles and statistics (commits/total_commits, %)
1033 --file-emails => add email addresses found in -f file (default: 0 (off))
1034 --scm => print SCM tree(s) if any
1035 --status => print status if any
1036 --subsystem => print subsystem name if any
1037 --web => print website(s) if any
1039 Output type options:
1040 --separator [, ] => separator for multiple entries on 1 line
1041 using --separator also sets --nomultiline if --separator is not [, ]
1042 --multiline => print 1 entry per line
1045 --pattern-depth => Number of pattern directory traversals (default: 0 (all))
1046 --keywords => scan patch for keywords (default: $keywords)
1047 --sections => print all of the subsystem sections with pattern matches
1048 --letters => print all matching 'letter' types from all matching sections
1049 --mailmap => use .mailmap file (default: $email_use_mailmap)
1050 --no-tree => run without a kernel tree
1051 --self-test => show potential issues with MAINTAINERS file content
1052 --version => show version
1053 --help => show this help information
1056 [--email --tree --nogit --git-fallback --m --r --n --l --multiline
1057 --pattern-depth=0 --remove-duplicates --rolestats]
1060 Using "-f directory" may give unexpected results:
1061 Used with "--git", git signators for _all_ files in and below
1062 directory are examined as git recurses directories.
1063 Any specified X: (exclude) pattern matches are _not_ ignored.
1064 Used with "--nogit", directory is used as a pattern match,
1065 no individual file within the directory or subdirectory
1067 Used with "--git-blame", does not iterate all files in directory
1068 Using "--git-blame" is slow and may add old committers and authors
1069 that are no longer active maintainers to the output.
1070 Using "--roles" or "--rolestats" with git send-email --cc-cmd or any
1071 other automated tools that expect only ["name"] <email address>
1072 may not work because of additional output after <email address>.
1073 Using "--rolestats" and "--git-blame" shows the #/total=% commits,
1074 not the percentage of the entire file authored. # of commits is
1075 not a good measure of amount of code authored. 1 major commit may
1076 contain a thousand lines, 5 trivial commits may modify a single line.
1077 If git is not installed, but mercurial (hg) is installed and an .hg
1078 repository exists, the following options apply to mercurial:
1080 --git-min-signatures, --git-max-maintainers, --git-min-percent, and
1082 Use --hg-since not --git-since to control date selection
1083 File ".get_maintainer.conf", if it exists in the linux kernel source root
1084 directory, can change whatever get_maintainer defaults are desired.
1085 Entries in this file can be any command line argument.
1086 This file is prepended to any additional command line arguments.
1087 Multiple lines and # comments are allowed.
1088 Most options have both positive and negative forms.
1089 The negative forms for --<foo> are --no<foo> and --no-<foo>.
1094 sub top_of_kernel_tree
{
1097 if ($lk_path ne "" && substr($lk_path,length($lk_path)-1,1) ne "/") {
1100 if ( (-f
"${lk_path}COPYING")
1101 && (-f
"${lk_path}CREDITS")
1102 && (-f
"${lk_path}Kbuild")
1103 && (-e
"${lk_path}MAINTAINERS")
1104 && (-f
"${lk_path}Makefile")
1105 && (-f
"${lk_path}README")
1106 && (-d
"${lk_path}Documentation")
1107 && (-d
"${lk_path}arch")
1108 && (-d
"${lk_path}include")
1109 && (-d
"${lk_path}drivers")
1110 && (-d
"${lk_path}fs")
1111 && (-d
"${lk_path}init")
1112 && (-d
"${lk_path}ipc")
1113 && (-d
"${lk_path}kernel")
1114 && (-d
"${lk_path}lib")
1115 && (-d
"${lk_path}scripts")) {
1122 my ($formatted_email) = @_;
1127 if ($formatted_email =~ /^([^<]+)<(.+\@.*)>.*$/) {
1130 } elsif ($formatted_email =~ /^\s*<(.+\@\S*)>.*$/) {
1132 } elsif ($formatted_email =~ /^(.+\@\S*).*$/) {
1136 $name =~ s/^\s+|\s+$//g;
1137 $name =~ s/^\"|\"$//g;
1138 $address =~ s/^\s+|\s+$//g;
1140 if ($name =~ /[^\w \-]/i) { ##has "must quote" chars
1141 $name =~ s/(?<!\\)"/\\"/g; ##escape quotes
1142 $name = "\"$name\"";
1145 return ($name, $address);
1149 my ($name, $address, $usename) = @_;
1151 my $formatted_email;
1153 $name =~ s/^\s+|\s+$//g;
1154 $name =~ s/^\"|\"$//g;
1155 $address =~ s/^\s+|\s+$//g;
1157 if ($name =~ /[^\w \-]/i) { ##has "must quote" chars
1158 $name =~ s/(?<!\\)"/\\"/g; ##escape quotes
1159 $name = "\"$name\"";
1163 if ("$name" eq "") {
1164 $formatted_email = "$address";
1166 $formatted_email = "$name <$address>";
1169 $formatted_email = $address;
1172 return $formatted_email;
1175 sub find_first_section
{
1178 while ($index < @typevalue) {
1179 my $tv = $typevalue[$index];
1180 if (($tv =~ m/^([A-Z]):\s*(.*)/)) {
1189 sub find_starting_index
{
1192 while ($index > 0) {
1193 my $tv = $typevalue[$index];
1194 if (!($tv =~ m/^([A-Z]):\s*(.*)/)) {
1203 sub find_ending_index
{
1206 while ($index < @typevalue) {
1207 my $tv = $typevalue[$index];
1208 if (!($tv =~ m/^([A-Z]):\s*(.*)/)) {
1217 sub get_subsystem_name
{
1220 my $start = find_starting_index
($index);
1222 my $subsystem = $typevalue[$start];
1223 if ($output_section_maxlen && length($subsystem) > $output_section_maxlen) {
1224 $subsystem = substr($subsystem, 0, $output_section_maxlen - 3);
1225 $subsystem =~ s/\s*$//;
1226 $subsystem = $subsystem . "...";
1231 sub get_maintainer_role
{
1235 my $start = find_starting_index
($index);
1236 my $end = find_ending_index
($index);
1238 my $role = "unknown";
1239 my $subsystem = get_subsystem_name
($index);
1241 for ($i = $start + 1; $i < $end; $i++) {
1242 my $tv = $typevalue[$i];
1243 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
1246 if ($ptype eq "S") {
1253 if ($role eq "supported") {
1254 $role = "supporter";
1255 } elsif ($role eq "maintained") {
1256 $role = "maintainer";
1257 } elsif ($role eq "odd fixes") {
1258 $role = "odd fixer";
1259 } elsif ($role eq "orphan") {
1260 $role = "orphan minder";
1261 } elsif ($role eq "obsolete") {
1262 $role = "obsolete minder";
1263 } elsif ($role eq "buried alive in reporters") {
1264 $role = "chief penguin";
1267 return $role . ":" . $subsystem;
1273 my $subsystem = get_subsystem_name
($index);
1275 if ($subsystem eq "THE REST") {
1282 sub add_categories
{
1286 my $start = find_starting_index
($index);
1287 my $end = find_ending_index
($index);
1289 push(@subsystem, $typevalue[$start]);
1291 for ($i = $start + 1; $i < $end; $i++) {
1292 my $tv = $typevalue[$i];
1293 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
1296 if ($ptype eq "L") {
1297 my $list_address = $pvalue;
1298 my $list_additional = "";
1299 my $list_role = get_list_role
($i);
1301 if ($list_role ne "") {
1302 $list_role = ":" . $list_role;
1304 if ($list_address =~ m/([^\s]+)\s+(.*)$/) {
1306 $list_additional = $2;
1308 if ($list_additional =~ m/subscribers-only/) {
1309 if ($email_subscriber_list) {
1310 if (!$hash_list_to{lc($list_address)}) {
1311 $hash_list_to{lc($list_address)} = 1;
1312 push(@list_to, [$list_address,
1313 "subscriber list${list_role}"]);
1318 if (!$hash_list_to{lc($list_address)}) {
1319 if ($list_additional =~ m/moderated/) {
1320 if ($email_moderated_list) {
1321 $hash_list_to{lc($list_address)} = 1;
1322 push(@list_to, [$list_address,
1323 "moderated list${list_role}"]);
1326 $hash_list_to{lc($list_address)} = 1;
1327 push(@list_to, [$list_address,
1328 "open list${list_role}"]);
1333 } elsif ($ptype eq "M") {
1334 my ($name, $address) = parse_email
($pvalue);
1337 my $tv = $typevalue[$i - 1];
1338 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
1341 $pvalue = format_email
($name, $address, $email_usename);
1346 if ($email_maintainer) {
1347 my $role = get_maintainer_role
($i);
1348 push_email_addresses
($pvalue, $role);
1350 } elsif ($ptype eq "R") {
1351 my ($name, $address) = parse_email
($pvalue);
1354 my $tv = $typevalue[$i - 1];
1355 if ($tv =~ m/^([A-Z]):\s*(.*)/) {
1358 $pvalue = format_email
($name, $address, $email_usename);
1363 if ($email_reviewer) {
1364 my $subsystem = get_subsystem_name
($i);
1365 push_email_addresses
($pvalue, "reviewer:$subsystem");
1367 } elsif ($ptype eq "T") {
1368 push(@scm, $pvalue);
1369 } elsif ($ptype eq "W") {
1370 push(@web, $pvalue);
1371 } elsif ($ptype eq "S") {
1372 push(@status, $pvalue);
1379 my ($name, $address) = @_;
1381 return 1 if (($name eq "") && ($address eq ""));
1382 return 1 if (($name ne "") && exists($email_hash_name{lc($name)}));
1383 return 1 if (($address ne "") && exists($email_hash_address{lc($address)}));
1388 sub push_email_address
{
1389 my ($line, $role) = @_;
1391 my ($name, $address) = parse_email
($line);
1393 if ($address eq "") {
1397 if (!$email_remove_duplicates) {
1398 push(@email_to, [format_email
($name, $address, $email_usename), $role]);
1399 } elsif (!email_inuse
($name, $address)) {
1400 push(@email_to, [format_email
($name, $address, $email_usename), $role]);
1401 $email_hash_name{lc($name)}++ if ($name ne "");
1402 $email_hash_address{lc($address)}++;
1408 sub push_email_addresses
{
1409 my ($address, $role) = @_;
1411 my @address_list = ();
1413 if (rfc822_valid
($address)) {
1414 push_email_address
($address, $role);
1415 } elsif (@address_list = rfc822_validlist
($address)) {
1416 my $array_count = shift(@address_list);
1417 while (my $entry = shift(@address_list)) {
1418 push_email_address
($entry, $role);
1421 if (!push_email_address
($address, $role)) {
1422 warn("Invalid MAINTAINERS address: '" . $address . "'\n");
1428 my ($line, $role) = @_;
1430 my ($name, $address) = parse_email
($line);
1431 my $email = format_email
($name, $address, $email_usename);
1433 foreach my $entry (@email_to) {
1434 if ($email_remove_duplicates) {
1435 my ($entry_name, $entry_address) = parse_email
($entry->[0]);
1436 if (($name eq $entry_name || $address eq $entry_address)
1437 && ($role eq "" || !($entry->[1] =~ m/$role/))
1439 if ($entry->[1] eq "") {
1440 $entry->[1] = "$role";
1442 $entry->[1] = "$entry->[1],$role";
1446 if ($email eq $entry->[0]
1447 && ($role eq "" || !($entry->[1] =~ m/$role/))
1449 if ($entry->[1] eq "") {
1450 $entry->[1] = "$role";
1452 $entry->[1] = "$entry->[1],$role";
1462 foreach my $path (split(/:/, $ENV{PATH
})) {
1463 if (-e
"$path/$bin") {
1464 return "$path/$bin";
1474 foreach my $path (split(/:/, ".:$ENV{HOME}:.scripts")) {
1475 if (-e
"$path/$conf") {
1476 return "$path/$conf";
1486 my ($name, $address) = parse_email
($line);
1487 my $email = format_email
($name, $address, 1);
1488 my $real_name = $name;
1489 my $real_address = $address;
1491 if (exists $mailmap->{names
}->{$email} ||
1492 exists $mailmap->{addresses
}->{$email}) {
1493 if (exists $mailmap->{names
}->{$email}) {
1494 $real_name = $mailmap->{names
}->{$email};
1496 if (exists $mailmap->{addresses
}->{$email}) {
1497 $real_address = $mailmap->{addresses
}->{$email};
1500 if (exists $mailmap->{names
}->{$address}) {
1501 $real_name = $mailmap->{names
}->{$address};
1503 if (exists $mailmap->{addresses
}->{$address}) {
1504 $real_address = $mailmap->{addresses
}->{$address};
1507 return format_email
($real_name, $real_address, 1);
1511 my (@addresses) = @_;
1513 my @mapped_emails = ();
1514 foreach my $line (@addresses) {
1515 push(@mapped_emails, mailmap_email
($line));
1517 merge_by_realname
(@mapped_emails) if ($email_use_mailmap);
1518 return @mapped_emails;
1521 sub merge_by_realname
{
1525 foreach my $email (@emails) {
1526 my ($name, $address) = parse_email
($email);
1527 if (exists $address_map{$name}) {
1528 $address = $address_map{$name};
1529 $email = format_email
($name, $address, 1);
1531 $address_map{$name} = $address;
1536 sub git_execute_cmd
{
1540 my $output = `$cmd`;
1541 $output =~ s/^\s*//gm;
1542 @lines = split("\n", $output);
1547 sub hg_execute_cmd
{
1551 my $output = `$cmd`;
1552 @lines = split("\n", $output);
1557 sub extract_formatted_signatures
{
1558 my (@signature_lines) = @_;
1560 my @type = @signature_lines;
1562 s/\s*(.*):.*/$1/ for (@type);
1565 s/\s*.*:\s*(.+)\s*/$1/ for (@signature_lines);
1567 ## Reformat email addresses (with names) to avoid badly written signatures
1569 foreach my $signer (@signature_lines) {
1570 $signer = deduplicate_email
($signer);
1573 return (\
@type, \
@signature_lines);
1576 sub vcs_find_signers
{
1577 my ($cmd, $file) = @_;
1580 my @signatures = ();
1584 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
1586 my $pattern = $VCS_cmds{"commit_pattern"};
1587 my $author_pattern = $VCS_cmds{"author_pattern"};
1588 my $stat_pattern = $VCS_cmds{"stat_pattern"};
1590 $stat_pattern =~ s/(\$\w+)/$1/eeg; #interpolate $stat_pattern
1592 $commits = grep(/$pattern/, @lines); # of commits
1594 @authors = grep(/$author_pattern/, @lines);
1595 @signatures = grep(/^[ \t]*${signature_pattern}.*\@.*$/, @lines);
1596 @stats = grep(/$stat_pattern/, @lines);
1598 # print("stats: <@stats>\n");
1600 return (0, \
@signatures, \
@authors, \
@stats) if !@signatures;
1602 save_commits_by_author
(@lines) if ($interactive);
1603 save_commits_by_signer
(@lines) if ($interactive);
1605 if (!$email_git_penguin_chiefs) {
1606 @signatures = grep(!/${penguin_chiefs}/i, @signatures);
1609 my ($author_ref, $authors_ref) = extract_formatted_signatures
(@authors);
1610 my ($types_ref, $signers_ref) = extract_formatted_signatures
(@signatures);
1612 return ($commits, $signers_ref, $authors_ref, \
@stats);
1615 sub vcs_find_author
{
1619 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
1621 if (!$email_git_penguin_chiefs) {
1622 @lines = grep(!/${penguin_chiefs}/i, @lines);
1625 return @lines if !@lines;
1628 foreach my $line (@lines) {
1629 if ($line =~ m/$VCS_cmds{"author_pattern"}/) {
1631 my ($name, $address) = parse_email
($author);
1632 $author = format_email
($name, $address, 1);
1633 push(@authors, $author);
1637 save_commits_by_author
(@lines) if ($interactive);
1638 save_commits_by_signer
(@lines) if ($interactive);
1643 sub vcs_save_commits
{
1648 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
1650 foreach my $line (@lines) {
1651 if ($line =~ m/$VCS_cmds{"blame_commit_pattern"}/) {
1664 return @commits if (!(-f
$file));
1666 if (@range && $VCS_cmds{"blame_range_cmd"} eq "") {
1667 my @all_commits = ();
1669 $cmd = $VCS_cmds{"blame_file_cmd"};
1670 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
1671 @all_commits = vcs_save_commits
($cmd);
1673 foreach my $file_range_diff (@range) {
1674 next if (!($file_range_diff =~ m/(.+):(.+):(.+)/));
1676 my $diff_start = $2;
1677 my $diff_length = $3;
1678 next if ("$file" ne "$diff_file");
1679 for (my $i = $diff_start; $i < $diff_start + $diff_length; $i++) {
1680 push(@commits, $all_commits[$i]);
1684 foreach my $file_range_diff (@range) {
1685 next if (!($file_range_diff =~ m/(.+):(.+):(.+)/));
1687 my $diff_start = $2;
1688 my $diff_length = $3;
1689 next if ("$file" ne "$diff_file");
1690 $cmd = $VCS_cmds{"blame_range_cmd"};
1691 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
1692 push(@commits, vcs_save_commits
($cmd));
1695 $cmd = $VCS_cmds{"blame_file_cmd"};
1696 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
1697 @commits = vcs_save_commits
($cmd);
1700 foreach my $commit (@commits) {
1701 $commit =~ s/^\^//g;
1707 my $printed_novcs = 0;
1709 %VCS_cmds = %VCS_cmds_git;
1710 return 1 if eval $VCS_cmds{"available"};
1711 %VCS_cmds = %VCS_cmds_hg;
1712 return 2 if eval $VCS_cmds{"available"};
1714 if (!$printed_novcs) {
1715 warn("$P: No supported VCS found. Add --nogit to options?\n");
1716 warn("Using a git repository produces better results.\n");
1717 warn("Try Linus Torvalds' latest git repository using:\n");
1718 warn("git clone git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git\n");
1726 return $vcs_used == 1;
1730 return $vcs_used == 2;
1733 sub interactive_get_maintainers
{
1734 my ($list_ref) = @_;
1735 my @list = @
$list_ref;
1744 foreach my $entry (@list) {
1745 $maintained = 1 if ($entry->[1] =~ /^(maintainer|supporter)/i);
1746 $selected{$count} = 1;
1747 $authored{$count} = 0;
1748 $signed{$count} = 0;
1754 my $print_options = 0;
1759 printf STDERR
"\n%1s %2s %-65s",
1760 "*", "#", "email/list and role:stats";
1762 ($email_git_fallback && !$maintained) ||
1764 print STDERR
"auth sign";
1767 foreach my $entry (@list) {
1768 my $email = $entry->[0];
1769 my $role = $entry->[1];
1771 $sel = "*" if ($selected{$count});
1772 my $commit_author = $commit_author_hash{$email};
1773 my $commit_signer = $commit_signer_hash{$email};
1776 $authored++ for (@
{$commit_author});
1777 $signed++ for (@
{$commit_signer});
1778 printf STDERR
"%1s %2d %-65s", $sel, $count + 1, $email;
1779 printf STDERR
"%4d %4d", $authored, $signed
1780 if ($authored > 0 || $signed > 0);
1781 printf STDERR
"\n %s\n", $role;
1782 if ($authored{$count}) {
1783 my $commit_author = $commit_author_hash{$email};
1784 foreach my $ref (@
{$commit_author}) {
1785 print STDERR
" Author: @{$ref}[1]\n";
1788 if ($signed{$count}) {
1789 my $commit_signer = $commit_signer_hash{$email};
1790 foreach my $ref (@
{$commit_signer}) {
1791 print STDERR
" @{$ref}[2]: @{$ref}[1]\n";
1798 my $date_ref = \
$email_git_since;
1799 $date_ref = \
$email_hg_since if (vcs_is_hg
());
1800 if ($print_options) {
1805 Version Control options:
1806 g use git history [$email_git]
1807 gf use git-fallback [$email_git_fallback]
1808 b use git blame [$email_git_blame]
1809 bs use blame signatures [$email_git_blame_signatures]
1810 c# minimum commits [$email_git_min_signatures]
1811 %# min percent [$email_git_min_percent]
1812 d# history to use [$$date_ref]
1813 x# max maintainers [$email_git_max_maintainers]
1814 t all signature types [$email_git_all_signature_types]
1815 m use .mailmap [$email_use_mailmap]
1822 tm toggle maintainers
1823 tg toggle git entries
1824 tl toggle open list entries
1825 ts toggle subscriber list entries
1826 f emails in file [$file_emails]
1827 k keywords in file [$keywords]
1828 r remove duplicates [$email_remove_duplicates]
1829 p# pattern match depth [$pattern_depth]
1833 "\n#(toggle), A#(author), S#(signed) *(all), ^(none), O(options), Y(approve): ";
1835 my $input = <STDIN
>;
1840 my @wish = split(/[, ]+/, $input);
1841 foreach my $nr (@wish) {
1843 my $sel = substr($nr, 0, 1);
1844 my $str = substr($nr, 1);
1846 $val = $1 if $str =~ /^(\d+)$/;
1851 $output_rolestats = 0;
1854 } elsif ($nr =~ /^\d+$/ && $nr > 0 && $nr <= $count) {
1855 $selected{$nr - 1} = !$selected{$nr - 1};
1856 } elsif ($sel eq "*" || $sel eq '^') {
1858 $toggle = 1 if ($sel eq '*');
1859 for (my $i = 0; $i < $count; $i++) {
1860 $selected{$i} = $toggle;
1862 } elsif ($sel eq "0") {
1863 for (my $i = 0; $i < $count; $i++) {
1864 $selected{$i} = !$selected{$i};
1866 } elsif ($sel eq "t") {
1867 if (lc($str) eq "m") {
1868 for (my $i = 0; $i < $count; $i++) {
1869 $selected{$i} = !$selected{$i}
1870 if ($list[$i]->[1] =~ /^(maintainer|supporter)/i);
1872 } elsif (lc($str) eq "g") {
1873 for (my $i = 0; $i < $count; $i++) {
1874 $selected{$i} = !$selected{$i}
1875 if ($list[$i]->[1] =~ /^(author|commit|signer)/i);
1877 } elsif (lc($str) eq "l") {
1878 for (my $i = 0; $i < $count; $i++) {
1879 $selected{$i} = !$selected{$i}
1880 if ($list[$i]->[1] =~ /^(open list)/i);
1882 } elsif (lc($str) eq "s") {
1883 for (my $i = 0; $i < $count; $i++) {
1884 $selected{$i} = !$selected{$i}
1885 if ($list[$i]->[1] =~ /^(subscriber list)/i);
1888 } elsif ($sel eq "a") {
1889 if ($val > 0 && $val <= $count) {
1890 $authored{$val - 1} = !$authored{$val - 1};
1891 } elsif ($str eq '*' || $str eq '^') {
1893 $toggle = 1 if ($str eq '*');
1894 for (my $i = 0; $i < $count; $i++) {
1895 $authored{$i} = $toggle;
1898 } elsif ($sel eq "s") {
1899 if ($val > 0 && $val <= $count) {
1900 $signed{$val - 1} = !$signed{$val - 1};
1901 } elsif ($str eq '*' || $str eq '^') {
1903 $toggle = 1 if ($str eq '*');
1904 for (my $i = 0; $i < $count; $i++) {
1905 $signed{$i} = $toggle;
1908 } elsif ($sel eq "o") {
1911 } elsif ($sel eq "g") {
1913 bool_invert
(\
$email_git_fallback);
1915 bool_invert
(\
$email_git);
1918 } elsif ($sel eq "b") {
1920 bool_invert
(\
$email_git_blame_signatures);
1922 bool_invert
(\
$email_git_blame);
1925 } elsif ($sel eq "c") {
1927 $email_git_min_signatures = $val;
1930 } elsif ($sel eq "x") {
1932 $email_git_max_maintainers = $val;
1935 } elsif ($sel eq "%") {
1936 if ($str ne "" && $val >= 0) {
1937 $email_git_min_percent = $val;
1940 } elsif ($sel eq "d") {
1942 $email_git_since = $str;
1943 } elsif (vcs_is_hg
()) {
1944 $email_hg_since = $str;
1947 } elsif ($sel eq "t") {
1948 bool_invert
(\
$email_git_all_signature_types);
1950 } elsif ($sel eq "f") {
1951 bool_invert
(\
$file_emails);
1953 } elsif ($sel eq "r") {
1954 bool_invert
(\
$email_remove_duplicates);
1956 } elsif ($sel eq "m") {
1957 bool_invert
(\
$email_use_mailmap);
1960 } elsif ($sel eq "k") {
1961 bool_invert
(\
$keywords);
1963 } elsif ($sel eq "p") {
1964 if ($str ne "" && $val >= 0) {
1965 $pattern_depth = $val;
1968 } elsif ($sel eq "h" || $sel eq "?") {
1971 Interactive mode allows you to select the various maintainers, submitters,
1972 commit signers and mailing lists that could be CC'd on a patch.
1974 Any *'d entry is selected.
1976 If you have git or hg installed, you can choose to summarize the commit
1977 history of files in the patch. Also, each line of the current file can
1978 be matched to its commit author and that commits signers with blame.
1980 Various knobs exist to control the length of time for active commit
1981 tracking, the maximum number of commit authors and signers to add,
1984 Enter selections at the prompt until you are satisfied that the selected
1985 maintainers are appropriate. You may enter multiple selections separated
1986 by either commas or spaces.
1990 print STDERR
"invalid option: '$nr'\n";
1995 print STDERR
"git-blame can be very slow, please have patience..."
1996 if ($email_git_blame);
1997 goto &get_maintainers
;
2001 #drop not selected entries
2003 my @new_emailto = ();
2004 foreach my $entry (@list) {
2005 if ($selected{$count}) {
2006 push(@new_emailto, $list[$count]);
2010 return @new_emailto;
2014 my ($bool_ref) = @_;
2023 sub deduplicate_email
{
2027 my ($name, $address) = parse_email
($email);
2028 $email = format_email
($name, $address, 1);
2029 $email = mailmap_email
($email);
2031 return $email if (!$email_remove_duplicates);
2033 ($name, $address) = parse_email
($email);
2035 if ($name ne "" && $deduplicate_name_hash{lc($name)}) {
2036 $name = $deduplicate_name_hash{lc($name)}->[0];
2037 $address = $deduplicate_name_hash{lc($name)}->[1];
2039 } elsif ($deduplicate_address_hash{lc($address)}) {
2040 $name = $deduplicate_address_hash{lc($address)}->[0];
2041 $address = $deduplicate_address_hash{lc($address)}->[1];
2045 $deduplicate_name_hash{lc($name)} = [ $name, $address ];
2046 $deduplicate_address_hash{lc($address)} = [ $name, $address ];
2048 $email = format_email
($name, $address, 1);
2049 $email = mailmap_email
($email);
2053 sub save_commits_by_author
{
2060 foreach my $line (@lines) {
2061 if ($line =~ m/$VCS_cmds{"author_pattern"}/) {
2063 $author = deduplicate_email
($author);
2064 push(@authors, $author);
2066 push(@commits, $1) if ($line =~ m/$VCS_cmds{"commit_pattern"}/);
2067 push(@subjects, $1) if ($line =~ m/$VCS_cmds{"subject_pattern"}/);
2070 for (my $i = 0; $i < @authors; $i++) {
2072 foreach my $ref(@
{$commit_author_hash{$authors[$i]}}) {
2073 if (@
{$ref}[0] eq $commits[$i] &&
2074 @
{$ref}[1] eq $subjects[$i]) {
2080 push(@
{$commit_author_hash{$authors[$i]}},
2081 [ ($commits[$i], $subjects[$i]) ]);
2086 sub save_commits_by_signer
{
2092 foreach my $line (@lines) {
2093 $commit = $1 if ($line =~ m/$VCS_cmds{"commit_pattern"}/);
2094 $subject = $1 if ($line =~ m/$VCS_cmds{"subject_pattern"}/);
2095 if ($line =~ /^[ \t]*${signature_pattern}.*\@.*$/) {
2096 my @signatures = ($line);
2097 my ($types_ref, $signers_ref) = extract_formatted_signatures
(@signatures);
2098 my @types = @
$types_ref;
2099 my @signers = @
$signers_ref;
2101 my $type = $types[0];
2102 my $signer = $signers[0];
2104 $signer = deduplicate_email
($signer);
2107 foreach my $ref(@
{$commit_signer_hash{$signer}}) {
2108 if (@
{$ref}[0] eq $commit &&
2109 @
{$ref}[1] eq $subject &&
2110 @
{$ref}[2] eq $type) {
2116 push(@
{$commit_signer_hash{$signer}},
2117 [ ($commit, $subject, $type) ]);
2124 my ($role, $divisor, @lines) = @_;
2129 return if (@lines <= 0);
2131 if ($divisor <= 0) {
2132 warn("Bad divisor in " . (caller(0))[3] . ": $divisor\n");
2136 @lines = mailmap
(@lines);
2138 return if (@lines <= 0);
2140 @lines = sort(@lines);
2143 $hash{$_}++ for @lines;
2146 foreach my $line (sort {$hash{$b} <=> $hash{$a}} keys %hash) {
2147 my $sign_offs = $hash{$line};
2148 my $percent = $sign_offs * 100 / $divisor;
2150 $percent = 100 if ($percent > 100);
2151 next if (ignore_email_address
($line));
2153 last if ($sign_offs < $email_git_min_signatures ||
2154 $count > $email_git_max_maintainers ||
2155 $percent < $email_git_min_percent);
2156 push_email_address
($line, '');
2157 if ($output_rolestats) {
2158 my $fmt_percent = sprintf("%.0f", $percent);
2159 add_role
($line, "$role:$sign_offs/$divisor=$fmt_percent%");
2161 add_role
($line, $role);
2166 sub vcs_file_signoffs
{
2177 $vcs_used = vcs_exists
();
2178 return if (!$vcs_used);
2180 my $cmd = $VCS_cmds{"find_signers_cmd"};
2181 $cmd =~ s/(\$\w+)/$1/eeg; # interpolate $cmd
2183 ($commits, $signers_ref, $authors_ref, $stats_ref) = vcs_find_signers
($cmd, $file);
2185 @signers = @
{$signers_ref} if defined $signers_ref;
2186 @authors = @
{$authors_ref} if defined $authors_ref;
2187 @stats = @
{$stats_ref} if defined $stats_ref;
2189 # print("commits: <$commits>\nsigners:<@signers>\nauthors: <@authors>\nstats: <@stats>\n");
2191 foreach my $signer (@signers) {
2192 $signer = deduplicate_email
($signer);
2195 vcs_assign
("commit_signer", $commits, @signers);
2196 vcs_assign
("authored", $commits, @authors);
2197 if ($#authors == $#stats) {
2198 my $stat_pattern = $VCS_cmds{"stat_pattern"};
2199 $stat_pattern =~ s/(\$\w+)/$1/eeg; #interpolate $stat_pattern
2203 for (my $i = 0; $i <= $#stats; $i++) {
2204 if ($stats[$i] =~ /$stat_pattern/) {
2209 my @tmp_authors = uniq
(@authors);
2210 foreach my $author (@tmp_authors) {
2211 $author = deduplicate_email
($author);
2213 @tmp_authors = uniq
(@tmp_authors);
2214 my @list_added = ();
2215 my @list_deleted = ();
2216 foreach my $author (@tmp_authors) {
2218 my $auth_deleted = 0;
2219 for (my $i = 0; $i <= $#stats; $i++) {
2220 if ($author eq deduplicate_email
($authors[$i]) &&
2221 $stats[$i] =~ /$stat_pattern/) {
2223 $auth_deleted += $2;
2226 for (my $i = 0; $i < $auth_added; $i++) {
2227 push(@list_added, $author);
2229 for (my $i = 0; $i < $auth_deleted; $i++) {
2230 push(@list_deleted, $author);
2233 vcs_assign
("added_lines", $added, @list_added);
2234 vcs_assign
("removed_lines", $deleted, @list_deleted);
2238 sub vcs_file_blame
{
2242 my @all_commits = ();
2247 $vcs_used = vcs_exists
();
2248 return if (!$vcs_used);
2250 @all_commits = vcs_blame
($file);
2251 @commits = uniq
(@all_commits);
2252 $total_commits = @commits;
2253 $total_lines = @all_commits;
2255 if ($email_git_blame_signatures) {
2258 my $commit_authors_ref;
2259 my $commit_signers_ref;
2261 my @commit_authors = ();
2262 my @commit_signers = ();
2263 my $commit = join(" -r ", @commits);
2266 $cmd = $VCS_cmds{"find_commit_signers_cmd"};
2267 $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd
2269 ($commit_count, $commit_signers_ref, $commit_authors_ref, $stats_ref) = vcs_find_signers
($cmd, $file);
2270 @commit_authors = @
{$commit_authors_ref} if defined $commit_authors_ref;
2271 @commit_signers = @
{$commit_signers_ref} if defined $commit_signers_ref;
2273 push(@signers, @commit_signers);
2275 foreach my $commit (@commits) {
2277 my $commit_authors_ref;
2278 my $commit_signers_ref;
2280 my @commit_authors = ();
2281 my @commit_signers = ();
2284 $cmd = $VCS_cmds{"find_commit_signers_cmd"};
2285 $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd
2287 ($commit_count, $commit_signers_ref, $commit_authors_ref, $stats_ref) = vcs_find_signers
($cmd, $file);
2288 @commit_authors = @
{$commit_authors_ref} if defined $commit_authors_ref;
2289 @commit_signers = @
{$commit_signers_ref} if defined $commit_signers_ref;
2291 push(@signers, @commit_signers);
2296 if ($from_filename) {
2297 if ($output_rolestats) {
2299 if (vcs_is_hg
()) {{ # Double brace for last exit
2301 my @commit_signers = ();
2302 @commits = uniq
(@commits);
2303 @commits = sort(@commits);
2304 my $commit = join(" -r ", @commits);
2307 $cmd = $VCS_cmds{"find_commit_author_cmd"};
2308 $cmd =~ s/(\$\w+)/$1/eeg; #substitute variables in $cmd
2312 @lines = &{$VCS_cmds{"execute_cmd"}}($cmd);
2314 if (!$email_git_penguin_chiefs) {
2315 @lines = grep(!/${penguin_chiefs}/i, @lines);
2321 foreach my $line (@lines) {
2322 if ($line =~ m/$VCS_cmds{"author_pattern"}/) {
2324 $author = deduplicate_email
($author);
2325 push(@authors, $author);
2329 save_commits_by_author
(@lines) if ($interactive);
2330 save_commits_by_signer
(@lines) if ($interactive);
2332 push(@signers, @authors);
2335 foreach my $commit (@commits) {
2337 my $cmd = $VCS_cmds{"find_commit_author_cmd"};
2338 $cmd =~ s/(\$\w+)/$1/eeg; #interpolate $cmd
2339 my @author = vcs_find_author
($cmd);
2342 my $formatted_author = deduplicate_email
($author[0]);
2344 my $count = grep(/$commit/, @all_commits);
2345 for ($i = 0; $i < $count ; $i++) {
2346 push(@blame_signers, $formatted_author);
2350 if (@blame_signers) {
2351 vcs_assign
("authored lines", $total_lines, @blame_signers);
2354 foreach my $signer (@signers) {
2355 $signer = deduplicate_email
($signer);
2357 vcs_assign
("commits", $total_commits, @signers);
2359 foreach my $signer (@signers) {
2360 $signer = deduplicate_email
($signer);
2362 vcs_assign
("modified commits", $total_commits, @signers);
2366 sub vcs_file_exists
{
2371 my $vcs_used = vcs_exists
();
2372 return 0 if (!$vcs_used);
2374 my $cmd = $VCS_cmds{"file_exists_cmd"};
2375 $cmd =~ s/(\$\w+)/$1/eeg; # interpolate $cmd
2377 $exists = &{$VCS_cmds{"execute_cmd"}}($cmd);
2379 return 0 if ($?
!= 0);
2384 sub vcs_list_files
{
2389 my $vcs_used = vcs_exists
();
2390 return 0 if (!$vcs_used);
2392 my $cmd = $VCS_cmds{"list_files_cmd"};
2393 $cmd =~ s/(\$\w+)/$1/eeg; # interpolate $cmd
2394 @lsfiles = &{$VCS_cmds{"execute_cmd"}}($cmd);
2396 return () if ($?
!= 0);
2405 @parms = grep(!$saw{$_}++, @parms);
2413 @parms = sort @parms;
2414 @parms = grep(!$saw{$_}++, @parms);
2418 sub clean_file_emails
{
2419 my (@file_emails) = @_;
2420 my @fmt_emails = ();
2422 foreach my $email (@file_emails) {
2423 $email =~ s/[\(\<\{]{0,1}([A-Za-z0-9_\.\+-]+\@[A-Za-z0-9\.-]+)[\)\>\}]{0,1}/\<$1\>/g;
2424 my ($name, $address) = parse_email
($email);
2425 if ($name eq '"[,\.]"') {
2429 my @nw = split(/[^A-Za-zÀ-ÿ\'\,\.\+-]/, $name);
2431 my $first = $nw[@nw - 3];
2432 my $middle = $nw[@nw - 2];
2433 my $last = $nw[@nw - 1];
2435 if (((length($first) == 1 && $first =~ m/[A-Za-z]/) ||
2436 (length($first) == 2 && substr($first, -1) eq ".")) ||
2437 (length($middle) == 1 ||
2438 (length($middle) == 2 && substr($middle, -1) eq "."))) {
2439 $name = "$first $middle $last";
2441 $name = "$middle $last";
2445 if (substr($name, -1) =~ /[,\.]/) {
2446 $name = substr($name, 0, length($name) - 1);
2447 } elsif (substr($name, -2) =~ /[,\.]"/) {
2448 $name = substr($name, 0, length($name) - 2) . '"';
2451 if (substr($name, 0, 1) =~ /[,\.]/) {
2452 $name = substr($name, 1, length($name) - 1);
2453 } elsif (substr($name, 0, 2) =~ /"[,\.]/) {
2454 $name = '"' . substr($name, 2, length($name) - 2);
2457 my $fmt_email = format_email
($name, $address, $email_usename);
2458 push(@fmt_emails, $fmt_email);
2468 my ($address, $role) = @
$_;
2469 if (!$saw{$address}) {
2470 if ($output_roles) {
2471 push(@lines, "$address ($role)");
2473 push(@lines, $address);
2485 if ($output_multiline) {
2486 foreach my $line (@parms) {
2490 print(join($output_separator, @parms));
2498 # Basic lexical tokens are specials, domain_literal, quoted_string, atom, and
2499 # comment. We must allow for rfc822_lwsp (or comments) after each of these.
2500 # This regexp will only work on addresses which have had comments stripped
2501 # and replaced with rfc822_lwsp.
2503 my $specials = '()<>@,;:\\\\".\\[\\]';
2504 my $controls = '\\000-\\037\\177';
2506 my $dtext = "[^\\[\\]\\r\\\\]";
2507 my $domain_literal = "\\[(?:$dtext|\\\\.)*\\]$rfc822_lwsp*";
2509 my $quoted_string = "\"(?:[^\\\"\\r\\\\]|\\\\.|$rfc822_lwsp)*\"$rfc822_lwsp*";
2511 # Use zero-width assertion to spot the limit of an atom. A simple
2512 # $rfc822_lwsp* causes the regexp engine to hang occasionally.
2513 my $atom = "[^$specials $controls]+(?:$rfc822_lwsp+|\\Z|(?=[\\[\"$specials]))";
2514 my $word = "(?:$atom|$quoted_string)";
2515 my $localpart = "$word(?:\\.$rfc822_lwsp*$word)*";
2517 my $sub_domain = "(?:$atom|$domain_literal)";
2518 my $domain = "$sub_domain(?:\\.$rfc822_lwsp*$sub_domain)*";
2520 my $addr_spec = "$localpart\@$rfc822_lwsp*$domain";
2522 my $phrase = "$word*";
2523 my $route = "(?:\@$domain(?:,\@$rfc822_lwsp*$domain)*:$rfc822_lwsp*)";
2524 my $route_addr = "\\<$rfc822_lwsp*$route?$addr_spec\\>$rfc822_lwsp*";
2525 my $mailbox = "(?:$addr_spec|$phrase$route_addr)";
2527 my $group = "$phrase:$rfc822_lwsp*(?:$mailbox(?:,\\s*$mailbox)*)?;\\s*";
2528 my $address = "(?:$mailbox|$group)";
2530 return "$rfc822_lwsp*$address";
2533 sub rfc822_strip_comments
{
2535 # Recursively remove comments, and replace with a single space. The simpler
2536 # regexps in the Email Addressing FAQ are imperfect - they will miss escaped
2537 # chars in atoms, for example.
2539 while ($s =~ s
/^((?
:[^"\\]|\\.)*
2540 (?:"(?
:[^"\\]|\\.)*"(?
:[^"\\]|\\.)*)*)
2541 \((?:[^()\\]|\\.)*\)/$1 /osx) {}
2545 # valid: returns true if the parameter is an RFC822 valid address
2548 my $s = rfc822_strip_comments(shift);
2551 $rfc822re = make_rfc822re();
2554 return $s =~ m/^$rfc822re$/so && $s =~ m/^$rfc822_char*$/;
2557 # validlist: In scalar context, returns true if the parameter is an RFC822
2558 # valid list of addresses.
2560 # In list context, returns an empty list on failure (an invalid
2561 # address was found); otherwise a list whose first element is the
2562 # number of addresses found and whose remaining elements are the
2563 # addresses. This is needed to disambiguate failure (invalid)
2564 # from success with no addresses found, because an empty string is
2567 sub rfc822_validlist {
2568 my $s = rfc822_strip_comments(shift);
2571 $rfc822re = make_rfc822re();
2573 # * null list items are valid according to the RFC
2574 # * the '1' business is to aid in distinguishing failure from no results
2577 if ($s =~ m/^(?:$rfc822re)?(?:,(?:$rfc822re)?)*$/so &&
2578 $s =~ m/^$rfc822_char*$/) {
2579 while ($s =~ m/(?:^|,$rfc822_lwsp*)($rfc822re)/gos) {
2582 return wantarray ? (scalar(@r), @r) : 1;
2584 return wantarray ? () : 0;