blob: 4da2f4029250163d1cf06d25e17592e7f6e28823 [file] [log] [blame]
Reid Spencer4dd4e6a2005-05-16 18:30:38 +00001#!/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 Spencerc0d5ed32006-08-11 23:50:27 +000011# Syntax: userloc.pl [-tag=tag|-html... <directory>...
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000012#
13# Options:
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000014# -tag=tag
15# Use "tag" to select the revision (as per cvs -r option)
Reid Spencerae2e6222006-08-11 20:44:17 +000016# -filedetails
17# Report details about lines of code in each file for each user
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000018# -html
19# Generate HTML output instead of text output
Reid Spencerca7592a2006-08-14 18:49:05 +000020# -topdir
21# Specify where the top llvm source directory is. Otherwise the
22# llvm-config tool is used to find it.
Reid Spencerc0d5ed32006-08-11 23:50:27 +000023# Directories:
24# The directories passed after the options should be relative paths to
25# directories of interest from the top of the llvm source tree, e.g. "lib"
26# or "include", etc.
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000027
Reid Spencerc0d5ed32006-08-11 23:50:27 +000028die "Usage userloc.pl [-tag=tag|-html] <directories>..."
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000029 if ($#ARGV < 0);
30
31my $tag = "";
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000032my $html = 0;
Reid Spencerbb6f6eb2006-08-11 18:36:55 +000033my $debug = 0;
Reid Spencerae2e6222006-08-11 20:44:17 +000034my $filedetails = "";
Reid Spencerca7592a2006-08-14 18:49:05 +000035my $srcroot = "";
Reid Spencerc0d5ed32006-08-11 23:50:27 +000036while ( defined($ARGV[0]) && substr($ARGV[0],0,1) eq '-' )
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000037{
Reid Spencerc0d5ed32006-08-11 23:50:27 +000038 if ($ARGV[0] =~ /-tag=.*/) {
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000039 $tag = $ARGV[0];
40 $tag =~ s#-tag=(.*)#$1#;
Reid Spencerae2e6222006-08-11 20:44:17 +000041 } elsif ($ARGV[0] =~ /-filedetails/) {
42 $filedetails = 1;
Reid Spencerbb6f6eb2006-08-11 18:36:55 +000043 } elsif ($ARGV[0] eq "-html") {
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000044 $html = 1;
Reid Spencerbb6f6eb2006-08-11 18:36:55 +000045 } elsif ($ARGV[0] eq "-debug") {
46 $debug = 1;
Reid Spencerca7592a2006-08-14 18:49:05 +000047 } elsif ($ARGV[0] eq "-topdir") {
48 shift; $srcroot = $ARGV[0]; shift;
Reid Spencerbb6f6eb2006-08-11 18:36:55 +000049 } else {
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000050 die "Invalid option: $ARGV[0]";
51 }
52 shift;
53}
54
Reid Spencerca7592a2006-08-14 18:49:05 +000055if (length($srcroot) == 0) {
56 chomp($srcroot = `llvm-config --src-root`);
57}
58if (! -d "$srcroot") {
59 die "Invalid source root: $srcroot\n";
60}
Reid Spencer32195632006-08-13 19:03:06 +000061chdir($srcroot);
Reid Spencerca7592a2006-08-14 18:49:05 +000062my $llvmdo = "$srcroot/utils/llvmdo -topdir '$srcroot'";
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000063my %Stats;
Reid Spencerae2e6222006-08-11 20:44:17 +000064my %FileStats;
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000065
Reid Spencerc0d5ed32006-08-11 23:50:27 +000066my $annotate = "cvs -z6 annotate -lf ";
67if (length($tag) > 0)
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000068{
Reid Spencerc0d5ed32006-08-11 23:50:27 +000069 $annotate = $annotate . " -r" . $tag;
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000070}
71
72sub GetCVSFiles
73{
74 my $d = $_[0];
75 my $files ="";
Reid Spencerc0d5ed32006-08-11 23:50:27 +000076 open FILELIST,
Reid Spencer32195632006-08-13 19:03:06 +000077 "$llvmdo -dirs \"$d\" -code-only echo |" || die "Can't get list of files with llvmdo";
Reid Spencerc0d5ed32006-08-11 23:50:27 +000078 while ( defined($line = <FILELIST>) ) {
79 chomp($file = $line);
Reid Spencer32195632006-08-13 19:03:06 +000080 print "File: $file\n" if ($debug);
Reid Spencerc0d5ed32006-08-11 23:50:27 +000081 $files = "$files $file";
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000082 }
83 return $files;
84}
85
Reid Spencerbb6f6eb2006-08-11 18:36:55 +000086sub ScanDir
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000087{
88 my $Dir = $_[0];
89 my $files = GetCVSFiles($Dir);
90
Reid Spencerae2e6222006-08-11 20:44:17 +000091 open (DATA,"$annotate $files 2>&1 |")
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000092 || die "Can't read cvs annotation data";
93
Reid Spencerae2e6222006-08-11 20:44:17 +000094 my $curfile = "";
Reid Spencer4dd4e6a2005-05-16 18:30:38 +000095 while ( defined($line = <DATA>) )
96 {
Reid Spencerae2e6222006-08-11 20:44:17 +000097 chomp($line);
98 if ($line =~ '^Annotations for.*') {
99 $curfile = $line;
100 $curfile =~ s#^Annotations for ([[:print:]]*)#$1#;
Reid Spencer32195632006-08-13 19:03:06 +0000101 print "Scanning: $curfile\n" if ($debug);
Reid Spencerae2e6222006-08-11 20:44:17 +0000102 } elsif ($line =~ /^[0-9.]*[ \t]*\([^)]*\):/) {
103 $uname = $line;
104 $uname =~ s#^[0-9.]*[ \t]*\(([a-zA-Z0-9_.-]*) [^)]*\):.*#$1#;
105 $Stats{$uname}++;
106 if ($filedetails) {
107 $FileStats{$uname} = {} unless exists $FileStats{$uname};
108 ${$FileStats{$uname}}{$curfile}++;
109 }
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000110 }
111 }
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000112 close DATA;
113}
114
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000115sub printStats
116{
117 my $dir = $_[0];
118 my $hash = $_[1];
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000119 my $user;
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000120 my $total = 0;
121
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000122 foreach $user (keys %Stats) { $total += $Stats{$user}; }
Reid Spencerae2e6222006-08-11 20:44:17 +0000123
124 if ($html) {
Reid Spencer32195632006-08-13 19:03:06 +0000125 print "<p>Total Source Lines: $total<br/></p>\n";
Reid Spencerae2e6222006-08-11 20:44:17 +0000126 print "<table>";
127 print " <tr><th style=\"text-align:right\">LOC</th>\n";
128 print " <th style=\"text-align:right\">\%LOC</th>\n";
129 print " <th style=\"text-align:left\">User</th>\n";
130 print "</tr>\n";
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000131 }
132
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000133 foreach $user ( sort keys %Stats )
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000134 {
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000135 my $v = $Stats{$user};
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000136 if (defined($v))
137 {
Reid Spencerae2e6222006-08-11 20:44:17 +0000138 if ($html) {
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000139 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;
140 if ($filedetails) {
141 print "<a href=\"#$user\">$user</a></td></tr>";
142 } else {
143 print $user,"</td></tr>";
144 }
Reid Spencerae2e6222006-08-11 20:44:17 +0000145 } else {
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000146 printf "%8d (%4.1f%%) %s\n", $v, (100.0/$total)*$v, $user;
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000147 }
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000148 }
149 }
Reid Spencerae2e6222006-08-11 20:44:17 +0000150 print "</table>\n" if ($html);
151
152 if ($filedetails) {
153 foreach $user (sort keys %FileStats) {
154 my $total = 0;
155 foreach $file (sort keys %{$FileStats{$user}}) {
156 $total += ${$FileStats{$user}}{$file}
157 }
158 if ($html) {
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000159 print "<table><tr><th style=\"text-align:left\" colspan=\"3\"><a name=\"$user\">$user</a></th></tr>\n";
Reid Spencerae2e6222006-08-11 20:44:17 +0000160 } else {
161 print $user,":\n";
162 }
163 foreach $file (sort keys %{$FileStats{$user}}) {
164 my $v = ${$FileStats{$user}}{$file};
165 if ($html) {
166 printf "<tr><td style=\"text-align:right\">&nbsp;&nbsp;%d</td><td
167 style=\"text-align:right\">&nbsp;%4.1f%%</td><td
168 style=\"text-align:left\">%s</td></tr>",$v, (100.0/$total)*$v,$file;
169 } else {
170 printf "%8d (%4.1f%%) %s\n", $v, (100.0/$total)*$v, $file;
171 }
172 }
173 if ($html) { print "</table>\n"; }
174 }
175 }
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000176}
177
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000178
179if ($html)
180{
181print "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01//EN\" \"http://www.w3.org/TR/html4/strict.dtd\">\n";
182print "<html>\n<head>\n";
183print " <title>LLVM LOC Based On CVS Annotation</title>\n";
184print " <link rel=\"stylesheet\" href=\"llvm.css\" type=\"text/css\"/>\n";
185print "</head>\n";
186print "<body><div class=\"doc_title\">LLVM LOC Based On CVS Annotation</div>\n";
187print "<p>This document shows the total lines of code per user in each\n";
188print "LLVM directory. Lines of code are attributed by the user that last\n";
189print "committed the line. This does not necessarily reflect authorship.</p>\n";
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000190}
191
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000192my @DIRS;
193if ($#ARGV > 0) {
194 @DIRS = @ARGV;
195} else {
196 push @DIRS, 'include';
197 push @DIRS, 'lib';
198 push @DIRS, 'tools';
199 push @DIRS, 'runtime';
200 push @DIRS, 'docs';
201 push @DIRS, 'test';
202 push @DIRS, 'utils';
203 push @DIRS, 'examples';
204 push @DIRS, 'projects/Stacker';
205 push @DIRS, 'projects/sample';
206 push @DIRS, 'autoconf';
207}
208
209for $Index ( 0 .. $#DIRS) {
Reid Spencer32195632006-08-13 19:03:06 +0000210 print "Scanning Dir: $DIRS[$Index]\n" if ($debug);
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000211 ScanDir($DIRS[$Index]);
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000212}
213
Reid Spencerae2e6222006-08-11 20:44:17 +0000214printStats;
Reid Spencer4dd4e6a2005-05-16 18:30:38 +0000215
Reid Spencerc0d5ed32006-08-11 23:50:27 +0000216print "</body></html>\n" if ($html) ;