Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 1 | #!/usr/bin/perl -w |
| 2 | # |
| 3 | # Program: userloc.pl |
| 4 | # |
| 5 | # Synopsis: This program uses "cvs annotate" to get a summary of how many lines |
| 6 | # of code the various developres are responsible for. It takes one |
| 7 | # argument, the directory to process. If the argument is not specified |
| 8 | # then the cwd is used. The directory must be an LLVM tree checked out |
| 9 | # from cvs. |
| 10 | # |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 11 | # Syntax: userloc.pl [-tag=tag|-html... <directory>... |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 12 | # |
| 13 | # Options: |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 14 | # -tag=tag |
| 15 | # Use "tag" to select the revision (as per cvs -r option) |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 16 | # -filedetails |
| 17 | # Report details about lines of code in each file for each user |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 18 | # -html |
| 19 | # Generate HTML output instead of text output |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 20 | # Directories: |
| 21 | # The directories passed after the options should be relative paths to |
| 22 | # directories of interest from the top of the llvm source tree, e.g. "lib" |
| 23 | # or "include", etc. |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 24 | |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 25 | die "Usage userloc.pl [-tag=tag|-html] <directories>..." |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 26 | if ($#ARGV < 0); |
| 27 | |
| 28 | my $tag = ""; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 29 | my $html = 0; |
Reid Spencer | 092ae90 | 2006-08-11 18:36:55 +0000 | [diff] [blame] | 30 | my $debug = 0; |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 31 | my $filedetails = ""; |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 32 | while ( defined($ARGV[0]) && substr($ARGV[0],0,1) eq '-' ) |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 33 | { |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 34 | if ($ARGV[0] =~ /-tag=.*/) { |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 35 | $tag = $ARGV[0]; |
| 36 | $tag =~ s#-tag=(.*)#$1#; |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 37 | } elsif ($ARGV[0] =~ /-filedetails/) { |
| 38 | $filedetails = 1; |
Reid Spencer | 092ae90 | 2006-08-11 18:36:55 +0000 | [diff] [blame] | 39 | } elsif ($ARGV[0] eq "-html") { |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 40 | $html = 1; |
Reid Spencer | 092ae90 | 2006-08-11 18:36:55 +0000 | [diff] [blame] | 41 | } elsif ($ARGV[0] eq "-debug") { |
| 42 | $debug = 1; |
| 43 | } else { |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 44 | die "Invalid option: $ARGV[0]"; |
| 45 | } |
| 46 | shift; |
| 47 | } |
| 48 | |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 49 | chomp(my $srcroot = `llvm-config --src-root`); |
| 50 | my $llvmdo = "$srcroot/utils/llvmdo"; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 51 | my %Stats; |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 52 | my %FileStats; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 53 | |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 54 | my $annotate = "cvs -z6 annotate -lf "; |
| 55 | if (length($tag) > 0) |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 56 | { |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 57 | $annotate = $annotate . " -r" . $tag; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 58 | } |
| 59 | |
| 60 | sub GetCVSFiles |
| 61 | { |
| 62 | my $d = $_[0]; |
| 63 | my $files =""; |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 64 | open FILELIST, |
| 65 | "$llvmdo -dirs \"$d\" echo |" || die "Can't get list of files with llvmdo"; |
| 66 | while ( defined($line = <FILELIST>) ) { |
| 67 | chomp($file = $line); |
| 68 | $files = "$files $file"; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 69 | } |
| 70 | return $files; |
| 71 | } |
| 72 | |
Reid Spencer | 092ae90 | 2006-08-11 18:36:55 +0000 | [diff] [blame] | 73 | sub ScanDir |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 74 | { |
| 75 | my $Dir = $_[0]; |
| 76 | my $files = GetCVSFiles($Dir); |
| 77 | |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 78 | open (DATA,"$annotate $files 2>&1 |") |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 79 | || die "Can't read cvs annotation data"; |
| 80 | |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 81 | my $curfile = ""; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 82 | while ( defined($line = <DATA>) ) |
| 83 | { |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 84 | chomp($line); |
| 85 | if ($line =~ '^Annotations for.*') { |
| 86 | $curfile = $line; |
| 87 | $curfile =~ s#^Annotations for ([[:print:]]*)#$1#; |
| 88 | } elsif ($line =~ /^[0-9.]*[ \t]*\([^)]*\):/) { |
| 89 | $uname = $line; |
| 90 | $uname =~ s#^[0-9.]*[ \t]*\(([a-zA-Z0-9_.-]*) [^)]*\):.*#$1#; |
| 91 | $Stats{$uname}++; |
| 92 | if ($filedetails) { |
| 93 | $FileStats{$uname} = {} unless exists $FileStats{$uname}; |
| 94 | ${$FileStats{$uname}}{$curfile}++; |
| 95 | } |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 96 | } |
| 97 | } |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 98 | close DATA; |
| 99 | } |
| 100 | |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 101 | sub printStats |
| 102 | { |
| 103 | my $dir = $_[0]; |
| 104 | my $hash = $_[1]; |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 105 | my $user; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 106 | my $total = 0; |
| 107 | |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 108 | foreach $user (keys %Stats) { $total += $Stats{$user}; } |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 109 | |
| 110 | if ($html) { |
| 111 | print "<table>"; |
| 112 | print " <tr><th style=\"text-align:right\">LOC</th>\n"; |
| 113 | print " <th style=\"text-align:right\">\%LOC</th>\n"; |
| 114 | print " <th style=\"text-align:left\">User</th>\n"; |
| 115 | print "</tr>\n"; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 116 | } |
| 117 | |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 118 | foreach $user ( sort keys %Stats ) |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 119 | { |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 120 | my $v = $Stats{$user}; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 121 | if (defined($v)) |
| 122 | { |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 123 | if ($html) { |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 124 | printf "<tr><td style=\"text-align:right\">%d</td><td style=\"text-align:right\">(%4.1f%%)</td><td style=\"text-align:left\">", $v, (100.0/$total)*$v; |
| 125 | if ($filedetails) { |
| 126 | print "<a href=\"#$user\">$user</a></td></tr>"; |
| 127 | } else { |
| 128 | print $user,"</td></tr>"; |
| 129 | } |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 130 | } else { |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 131 | printf "%8d (%4.1f%%) %s\n", $v, (100.0/$total)*$v, $user; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 132 | } |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 133 | } |
| 134 | } |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 135 | print "</table>\n" if ($html); |
| 136 | |
| 137 | if ($filedetails) { |
| 138 | foreach $user (sort keys %FileStats) { |
| 139 | my $total = 0; |
| 140 | foreach $file (sort keys %{$FileStats{$user}}) { |
| 141 | $total += ${$FileStats{$user}}{$file} |
| 142 | } |
| 143 | if ($html) { |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 144 | print "<table><tr><th style=\"text-align:left\" colspan=\"3\"><a name=\"$user\">$user</a></th></tr>\n"; |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 145 | } else { |
| 146 | print $user,":\n"; |
| 147 | } |
| 148 | foreach $file (sort keys %{$FileStats{$user}}) { |
| 149 | my $v = ${$FileStats{$user}}{$file}; |
| 150 | if ($html) { |
| 151 | printf "<tr><td style=\"text-align:right\"> %d</td><td |
| 152 | style=\"text-align:right\"> %4.1f%%</td><td |
| 153 | style=\"text-align:left\">%s</td></tr>",$v, (100.0/$total)*$v,$file; |
| 154 | } else { |
| 155 | printf "%8d (%4.1f%%) %s\n", $v, (100.0/$total)*$v, $file; |
| 156 | } |
| 157 | } |
| 158 | if ($html) { print "</table>\n"; } |
| 159 | } |
| 160 | } |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 161 | } |
| 162 | |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 163 | |
| 164 | if ($html) |
| 165 | { |
| 166 | print "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01//EN\" \"http://www.w3.org/TR/html4/strict.dtd\">\n"; |
| 167 | print "<html>\n<head>\n"; |
| 168 | print " <title>LLVM LOC Based On CVS Annotation</title>\n"; |
| 169 | print " <link rel=\"stylesheet\" href=\"llvm.css\" type=\"text/css\"/>\n"; |
| 170 | print "</head>\n"; |
| 171 | print "<body><div class=\"doc_title\">LLVM LOC Based On CVS Annotation</div>\n"; |
| 172 | print "<p>This document shows the total lines of code per user in each\n"; |
| 173 | print "LLVM directory. Lines of code are attributed by the user that last\n"; |
| 174 | print "committed the line. This does not necessarily reflect authorship.</p>\n"; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 175 | } |
| 176 | |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 177 | my @DIRS; |
| 178 | if ($#ARGV > 0) { |
| 179 | @DIRS = @ARGV; |
| 180 | } else { |
| 181 | push @DIRS, 'include'; |
| 182 | push @DIRS, 'lib'; |
| 183 | push @DIRS, 'tools'; |
| 184 | push @DIRS, 'runtime'; |
| 185 | push @DIRS, 'docs'; |
| 186 | push @DIRS, 'test'; |
| 187 | push @DIRS, 'utils'; |
| 188 | push @DIRS, 'examples'; |
| 189 | push @DIRS, 'projects/Stacker'; |
| 190 | push @DIRS, 'projects/sample'; |
| 191 | push @DIRS, 'autoconf'; |
| 192 | } |
| 193 | |
| 194 | for $Index ( 0 .. $#DIRS) { |
| 195 | ScanDir($DIRS[$Index]); |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 196 | } |
| 197 | |
Reid Spencer | 1f0b9a1 | 2006-08-11 20:44:17 +0000 | [diff] [blame] | 198 | printStats; |
Reid Spencer | 8dd9127 | 2005-05-16 18:30:38 +0000 | [diff] [blame] | 199 | |
Reid Spencer | 8ca43d7 | 2006-08-11 23:50:27 +0000 | [diff] [blame^] | 200 | print "</body></html>\n" if ($html) ; |