| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1 | #!/usr/bin/env perl | 
 | 2 | # | 
 | 3 | #                     The LLVM Compiler Infrastructure | 
 | 4 | # | 
 | 5 | # This file is distributed under the University of Illinois Open Source | 
 | 6 | # License. See LICENSE.TXT for details. | 
 | 7 | # | 
 | 8 | ##===----------------------------------------------------------------------===## | 
 | 9 | # | 
 | 10 | # A script designed to wrap a build so that all calls to gcc are intercepted | 
 | 11 | # and piped to the static analyzer. | 
 | 12 | # | 
 | 13 | ##===----------------------------------------------------------------------===## | 
 | 14 |  | 
 | 15 | use strict; | 
 | 16 | use warnings; | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 17 | use FindBin qw($RealBin); | 
| Ted Kremenek | a6e2481 | 2008-04-19 18:05:48 +0000 | [diff] [blame] | 18 | use Digest::MD5; | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 19 | use File::Basename; | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 20 | use Term::ANSIColor; | 
 | 21 | use Term::ANSIColor qw(:constants); | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 22 | use Cwd; | 
 | 23 | use Sys::Hostname; | 
 | 24 | use File::Basename; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 25 |  | 
 | 26 | my $Verbose = 0;       # Verbose output from this script. | 
 | 27 | my $Prog = "scan-build"; | 
| Ted Kremenek | f4cdf41 | 2008-05-23 18:17:05 +0000 | [diff] [blame] | 28 | my $BuildName; | 
 | 29 | my $BuildDate; | 
| Ted Kremenek | 95aa105 | 2008-09-04 17:52:41 +0000 | [diff] [blame] | 30 | my $CXX;  # Leave undefined initially. | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 31 |  | 
| Ted Kremenek | 0e68938 | 2008-09-11 18:17:51 +0000 | [diff] [blame] | 32 | my $TERM = $ENV{'TERM'}; | 
 | 33 | my $UseColor = (defined $TERM and $TERM eq 'xterm-color' and -t STDOUT | 
 | 34 |                 and defined $ENV{'SCAN_BUILD_COLOR'}); | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 35 |  | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 36 | my $UserName = HtmlEscape(getpwuid($<) || 'unknown'); | 
 | 37 | my $HostName = HtmlEscape(hostname() || 'unknown'); | 
 | 38 | my $CurrentDir = HtmlEscape(getcwd()); | 
 | 39 | my $CurrentDirSuffix = basename($CurrentDir); | 
 | 40 |  | 
 | 41 | my $CmdArgs; | 
 | 42 |  | 
 | 43 | my $HtmlTitle; | 
 | 44 |  | 
 | 45 | my $Date = localtime(); | 
 | 46 |  | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 47 | ##----------------------------------------------------------------------------## | 
 | 48 | # Diagnostics | 
 | 49 | ##----------------------------------------------------------------------------## | 
 | 50 |  | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 51 | sub Diag { | 
 | 52 |   if ($UseColor) { | 
 | 53 |     print BOLD, MAGENTA "$Prog: @_"; | 
 | 54 |     print RESET; | 
 | 55 |   } | 
 | 56 |   else { | 
 | 57 |     print "$Prog: @_"; | 
 | 58 |   }   | 
 | 59 | } | 
 | 60 |  | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 61 | sub DiagCrashes { | 
 | 62 |   my $Dir = shift; | 
 | 63 |   Diag ("The analyzer crashed on some source files.\n"); | 
| Ted Kremenek | 386c693 | 2008-09-03 17:59:35 +0000 | [diff] [blame] | 64 |   Diag ("Preprocessed versions of crashed files were deposited in '$Dir/crashes'.\n"); | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 65 |   Diag ("Please consider submitting a bug report using these files:\n"); | 
 | 66 |   Diag ("  http://clang.llvm.org/StaticAnalysisUsage.html#filingbugs\n") | 
 | 67 | } | 
 | 68 |  | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 69 | sub DieDiag { | 
 | 70 |   if ($UseColor) { | 
 | 71 |     print BOLD, RED "$Prog: "; | 
 | 72 |     print RESET, RED @_; | 
 | 73 |     print RESET; | 
 | 74 |   } | 
 | 75 |   else { | 
 | 76 |     print "$Prog: ", @_; | 
 | 77 |   } | 
 | 78 |   exit(0); | 
 | 79 | } | 
 | 80 |  | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 81 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 82 | # Some initial preprocessing of Clang options. | 
 | 83 | ##----------------------------------------------------------------------------## | 
 | 84 |  | 
 | 85 | my $ClangSB = "$RealBin/clang"; | 
 | 86 | my $Clang = $ClangSB; | 
 | 87 |  | 
 | 88 | if (! -x $ClangSB) { | 
 | 89 |   $Clang = "clang"; | 
 | 90 | } | 
 | 91 |  | 
 | 92 | my %AvailableAnalyses; | 
 | 93 |  | 
 | 94 | # Query clang for analysis options. | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 95 | open(PIPE, "-|", $Clang, "--help") or | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 96 |   DieDiag("Cannot execute '$Clang'"); | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 97 |  | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 98 | my $FoundAnalysis = 0; | 
 | 99 |  | 
 | 100 | while(<PIPE>) { | 
 | 101 |   if ($FoundAnalysis == 0) { | 
 | 102 |     if (/Available Source Code Analyses/) { | 
 | 103 |       $FoundAnalysis = 1; | 
 | 104 |     } | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 105 |  | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 106 |     next; | 
 | 107 |   } | 
 | 108 |      | 
 | 109 |   if (/^\s\s\s\s([^\s]+)\s(.+)$/) { | 
 | 110 |     next if ($1 =~ /-dump/ or $1 =~ /-view/  | 
 | 111 |              or $1 =~ /-checker-simple/ or $1 =~ /-warn-uninit/); | 
 | 112 |               | 
 | 113 |     $AvailableAnalyses{$1} = $2; | 
 | 114 |     next; | 
 | 115 |   } | 
 | 116 |    | 
 | 117 |   last; | 
 | 118 | } | 
 | 119 |  | 
 | 120 | close (PIPE); | 
 | 121 |  | 
 | 122 | my %AnalysesDefaultEnabled = ( | 
 | 123 |   '-warn-dead-stores' => 1, | 
 | 124 |   '-checker-cfref' => 1, | 
| Ted Kremenek | 9012599 | 2008-07-15 23:41:32 +0000 | [diff] [blame] | 125 |   '-warn-objc-methodsigs' => 1, | 
| Ted Kremenek | bde3a05 | 2008-07-25 20:35:01 +0000 | [diff] [blame] | 126 |   '-warn-objc-missing-dealloc' => 1, | 
| Ted Kremenek | 5d44349 | 2008-09-18 06:34:16 +0000 | [diff] [blame] | 127 |   '-warn-objc-unused-ivars' => 1, | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 128 | ); | 
 | 129 |  | 
 | 130 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 131 | # GetHTMLRunDir - Construct an HTML directory name for the current sub-run. | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 132 | ##----------------------------------------------------------------------------## | 
 | 133 |  | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 134 | sub GetHTMLRunDir {   | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 135 |  | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 136 |   die "Not enough arguments." if (@_ == 0);   | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 137 |   my $Dir = shift @_; | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 138 |      | 
 | 139 |   my $TmpMode = 0; | 
 | 140 |   if (!defined $Dir) { | 
 | 141 |     $Dir = "/tmp"; | 
 | 142 |     $TmpMode = 1; | 
 | 143 |   } | 
 | 144 |  | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 145 |   # Get current date and time. | 
 | 146 |    | 
 | 147 |   my @CurrentTime = localtime(); | 
 | 148 |    | 
 | 149 |   my $year  = $CurrentTime[5] + 1900; | 
 | 150 |   my $day   = $CurrentTime[3]; | 
 | 151 |   my $month = $CurrentTime[4] + 1; | 
 | 152 |    | 
| Ted Kremenek | 9d7405f | 2008-05-14 17:23:56 +0000 | [diff] [blame] | 153 |   my $DateString = sprintf("%d-%02d-%02d", $year, $month, $day); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 154 |    | 
 | 155 |   # Determine the run number. | 
 | 156 |    | 
 | 157 |   my $RunNumber; | 
 | 158 |    | 
 | 159 |   if (-d $Dir) { | 
 | 160 |      | 
 | 161 |     if (! -r $Dir) { | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 162 |       DieDiag("directory '$Dir' exists but is not readable.\n"); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 163 |     } | 
 | 164 |      | 
 | 165 |     # Iterate over all files in the specified directory. | 
 | 166 |      | 
 | 167 |     my $max = 0; | 
 | 168 |      | 
 | 169 |     opendir(DIR, $Dir); | 
| Ted Kremenek | 29da6c5 | 2008-08-07 17:57:34 +0000 | [diff] [blame] | 170 |     my @FILES = grep { -d "$Dir/$_" } readdir(DIR); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 171 |     closedir(DIR); | 
 | 172 |      | 
 | 173 |     foreach my $f (@FILES) { | 
 | 174 |  | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 175 |       # Strip the prefix '$Prog-' if we are dumping files to /tmp. | 
 | 176 |       if ($TmpMode) { | 
 | 177 |         next if (!($f =~ /^$Prog-(.+)/)); | 
 | 178 |         $f = $1; | 
 | 179 |       } | 
 | 180 |  | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 181 |        | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 182 |       my @x = split/-/, $f; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 183 |       next if (scalar(@x) != 4); | 
 | 184 |       next if ($x[0] != $year); | 
 | 185 |       next if ($x[1] != $month); | 
 | 186 |       next if ($x[2] != $day); | 
 | 187 |        | 
 | 188 |       if ($x[3] > $max) { | 
 | 189 |         $max = $x[3]; | 
 | 190 |       }       | 
 | 191 |     } | 
 | 192 |      | 
 | 193 |     $RunNumber = $max + 1; | 
 | 194 |   } | 
 | 195 |   else { | 
 | 196 |      | 
 | 197 |     if (-x $Dir) { | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 198 |       DieDiag("'$Dir' exists but is not a directory.\n"); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 199 |     } | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 200 |  | 
 | 201 |     if ($TmpMode) { | 
 | 202 |       DieDiag("The directory '/tmp' does not exist or cannot be accessed."); | 
 | 203 |     } | 
 | 204 |  | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 205 |     # $Dir does not exist.  It will be automatically created by the  | 
 | 206 |     # clang driver.  Set the run number to 1.   | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 207 |  | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 208 |     $RunNumber = 1; | 
 | 209 |   } | 
 | 210 |    | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 211 |   die "RunNumber must be defined!" if (!defined $RunNumber); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 212 |    | 
 | 213 |   # Append the run number. | 
| Ted Kremenek | fc0898a | 2008-09-04 23:56:36 +0000 | [diff] [blame] | 214 |   my $NewDir; | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 215 |   if ($TmpMode) { | 
| Ted Kremenek | fc0898a | 2008-09-04 23:56:36 +0000 | [diff] [blame] | 216 |     $NewDir = "$Dir/$Prog-$DateString-$RunNumber"; | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 217 |   } | 
 | 218 |   else { | 
| Ted Kremenek | fc0898a | 2008-09-04 23:56:36 +0000 | [diff] [blame] | 219 |     $NewDir = "$Dir/$DateString-$RunNumber"; | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 220 |   } | 
| Ted Kremenek | fc0898a | 2008-09-04 23:56:36 +0000 | [diff] [blame] | 221 |   system 'mkdir','-p',$NewDir; | 
 | 222 |   return $NewDir; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 223 | } | 
 | 224 |  | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 225 | sub SetHtmlEnv { | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 226 |    | 
 | 227 |   die "Wrong number of arguments." if (scalar(@_) != 2); | 
 | 228 |    | 
 | 229 |   my $Args = shift; | 
 | 230 |   my $Dir = shift; | 
 | 231 |    | 
 | 232 |   die "No build command." if (scalar(@$Args) == 0); | 
 | 233 |    | 
 | 234 |   my $Cmd = $$Args[0]; | 
 | 235 |    | 
 | 236 |   if ($Cmd =~ /configure/) { | 
 | 237 |     return; | 
 | 238 |   } | 
 | 239 |    | 
 | 240 |   if ($Verbose) { | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 241 |     Diag("Emitting reports for this run to '$Dir'.\n"); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 242 |   } | 
 | 243 |    | 
 | 244 |   $ENV{'CCC_ANALYZER_HTML'} = $Dir; | 
 | 245 | } | 
 | 246 |  | 
 | 247 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | 57cf446 | 2008-04-18 15:09:30 +0000 | [diff] [blame] | 248 | # ComputeDigest - Compute a digest of the specified file. | 
 | 249 | ##----------------------------------------------------------------------------## | 
 | 250 |  | 
 | 251 | sub ComputeDigest { | 
 | 252 |   my $FName = shift; | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 253 |   DieDiag("Cannot read $FName to compute Digest.\n") if (! -r $FName);   | 
| Ted Kremenek | a6e2481 | 2008-04-19 18:05:48 +0000 | [diff] [blame] | 254 |    | 
 | 255 |   # Use Digest::MD5.  We don't have to be cryptographically secure.  We're | 
| Ted Kremenek | 7ea02e6 | 2008-04-19 18:07:44 +0000 | [diff] [blame] | 256 |   # just looking for duplicate files that come from a non-malicious source. | 
 | 257 |   # We use Digest::MD5 because it is a standard Perl module that should | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 258 |   # come bundled on most systems.   | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 259 |   open(FILE, $FName) or DieDiag("Cannot open $FName when computing Digest.\n"); | 
| Ted Kremenek | a6e2481 | 2008-04-19 18:05:48 +0000 | [diff] [blame] | 260 |   binmode FILE; | 
 | 261 |   my $Result = Digest::MD5->new->addfile(*FILE)->hexdigest; | 
 | 262 |   close(FILE); | 
 | 263 |    | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 264 |   # Return the digest.   | 
| Ted Kremenek | a6e2481 | 2008-04-19 18:05:48 +0000 | [diff] [blame] | 265 |   return $Result; | 
| Ted Kremenek | 57cf446 | 2008-04-18 15:09:30 +0000 | [diff] [blame] | 266 | } | 
 | 267 |  | 
 | 268 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 269 | #  UpdatePrefix - Compute the common prefix of files. | 
 | 270 | ##----------------------------------------------------------------------------## | 
 | 271 |  | 
 | 272 | my $Prefix; | 
 | 273 |  | 
 | 274 | sub UpdatePrefix { | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 275 |   my $x = shift; | 
 | 276 |   my $y = basename($x); | 
 | 277 |   $x =~ s/\Q$y\E$//; | 
 | 278 |    | 
 | 279 |   # Ignore /usr, /Library, /System, /Developer | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 280 |   return if ( $x =~ /^\/usr/ or $x =~ /^\/Library/ | 
 | 281 |               or $x =~ /^\/System/ or $x =~ /^\/Developer/); | 
 | 282 |  | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 283 |   if (!defined $Prefix) { | 
 | 284 |     $Prefix = $x; | 
 | 285 |     return; | 
 | 286 |   } | 
 | 287 |    | 
| Ted Kremenek | 20b2bae | 2008-09-11 21:15:10 +0000 | [diff] [blame] | 288 |   chop $Prefix while (!($x =~ /^\Q$Prefix/)); | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 289 | } | 
 | 290 |  | 
 | 291 | sub GetPrefix { | 
 | 292 |   return $Prefix; | 
 | 293 | } | 
 | 294 |  | 
 | 295 | ##----------------------------------------------------------------------------## | 
 | 296 | #  UpdateInFilePath - Update the path in the report file. | 
 | 297 | ##----------------------------------------------------------------------------## | 
 | 298 |  | 
 | 299 | sub UpdateInFilePath { | 
 | 300 |   my $fname = shift; | 
 | 301 |   my $regex = shift; | 
 | 302 |   my $newtext = shift; | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 303 |  | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 304 |   open (RIN, $fname) or die "cannot open $fname"; | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 305 |   open (ROUT, ">", "$fname.tmp") or die "cannot open $fname.tmp"; | 
 | 306 |  | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 307 |   while (<RIN>) { | 
 | 308 |     s/$regex/$newtext/; | 
 | 309 |     print ROUT $_; | 
 | 310 |   } | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 311 |  | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 312 |   close (ROUT); | 
 | 313 |   close (RIN); | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 314 |   system("mv", "$fname.tmp", $fname); | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 315 | } | 
 | 316 |  | 
 | 317 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 318 | # ScanFile - Scan a report file for various identifying attributes. | 
 | 319 | ##----------------------------------------------------------------------------## | 
 | 320 |  | 
| Ted Kremenek | 57cf446 | 2008-04-18 15:09:30 +0000 | [diff] [blame] | 321 | # Sometimes a source file is scanned more than once, and thus produces | 
 | 322 | # multiple error reports.  We use a cache to solve this problem. | 
 | 323 |  | 
 | 324 | my %AlreadyScanned; | 
 | 325 |  | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 326 | sub ScanFile { | 
 | 327 |    | 
 | 328 |   my $Index = shift; | 
 | 329 |   my $Dir = shift; | 
 | 330 |   my $FName = shift; | 
 | 331 |    | 
| Ted Kremenek | 57cf446 | 2008-04-18 15:09:30 +0000 | [diff] [blame] | 332 |   # Compute a digest for the report file.  Determine if we have already | 
 | 333 |   # scanned a file that looks just like it. | 
 | 334 |    | 
 | 335 |   my $digest = ComputeDigest("$Dir/$FName"); | 
 | 336 |  | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 337 |   if (defined $AlreadyScanned{$digest}) { | 
| Ted Kremenek | 57cf446 | 2008-04-18 15:09:30 +0000 | [diff] [blame] | 338 |     # Redundant file.  Remove it. | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 339 |     system ("rm", "-f", "$Dir/$FName"); | 
| Ted Kremenek | 57cf446 | 2008-04-18 15:09:30 +0000 | [diff] [blame] | 340 |     return; | 
 | 341 |   } | 
 | 342 |    | 
 | 343 |   $AlreadyScanned{$digest} = 1; | 
 | 344 |    | 
| Ted Kremenek | 809709f | 2008-04-18 16:58:34 +0000 | [diff] [blame] | 345 |   # At this point the report file is not world readable.  Make it happen. | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 346 |   system ("chmod", "644", "$Dir/$FName"); | 
| Ted Kremenek | 684bb09 | 2008-04-18 15:18:20 +0000 | [diff] [blame] | 347 |    | 
 | 348 |   # Scan the report file for tags. | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 349 |   open(IN, "$Dir/$FName") or DieDiag("Cannot open '$Dir/$FName'\n"); | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 350 |  | 
 | 351 |   my $BugDesc = ""; | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 352 |   my $BugFile = ""; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 353 |   my $BugCategory; | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 354 |   my $BugPathLength = 1; | 
 | 355 |   my $BugLine = 0; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 356 |   my $found = 0; | 
 | 357 |  | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 358 |   while (<IN>) { | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 359 |  | 
 | 360 |     last if ($found == 5); | 
 | 361 |  | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 362 |     if (/<!-- BUGDESC (.*) -->$/) { | 
 | 363 |       $BugDesc = $1; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 364 |       ++$found; | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 365 |     } | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 366 |     elsif (/<!-- BUGFILE (.*) -->$/) { | 
 | 367 |       $BugFile = $1; | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 368 |       UpdatePrefix($BugFile); | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 369 |       ++$found; | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 370 |     } | 
 | 371 |     elsif (/<!-- BUGPATHLENGTH (.*) -->$/) { | 
 | 372 |       $BugPathLength = $1; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 373 |       ++$found; | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 374 |     } | 
 | 375 |     elsif (/<!-- BUGLINE (.*) -->$/) { | 
 | 376 |       $BugLine = $1;     | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 377 |       ++$found; | 
 | 378 |     } | 
 | 379 |     elsif (/<!-- BUGCATEGORY (.*) -->$/) { | 
 | 380 |       $BugCategory = $1; | 
 | 381 |       ++$found; | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 382 |     } | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 383 |   } | 
 | 384 |  | 
 | 385 |   close(IN); | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 386 |    | 
 | 387 |   if (!defined $BugCategory) { | 
 | 388 |     $BugCategory = "Other"; | 
 | 389 |   } | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 390 |      | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 391 |   push @$Index,[ $FName, $BugCategory, $BugDesc, $BugFile, $BugLine, | 
 | 392 |                  $BugPathLength ]; | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 393 | } | 
 | 394 |  | 
 | 395 | ##----------------------------------------------------------------------------## | 
 | 396 | # CopyJS - Copy JavaScript code to target directory. | 
 | 397 | ##----------------------------------------------------------------------------## | 
 | 398 |  | 
 | 399 | sub CopyJS { | 
 | 400 |  | 
 | 401 |   my $Dir = shift; | 
 | 402 |    | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 403 |   DieDiag("Cannot find 'sorttable.js'.\n") | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 404 |     if (! -r "$RealBin/sorttable.js");   | 
 | 405 |  | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 406 |   system ("cp", "$RealBin/sorttable.js", "$Dir"); | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 407 |  | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 408 |   DieDiag("Could not copy 'sorttable.js' to '$Dir'.\n") | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 409 |     if (! -r "$Dir/sorttable.js"); | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 410 | } | 
 | 411 |  | 
 | 412 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 413 | # Postprocess - Postprocess the results of an analysis scan. | 
 | 414 | ##----------------------------------------------------------------------------## | 
 | 415 |  | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 416 | sub Postprocess { | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 417 |    | 
 | 418 |   my $Dir = shift; | 
| Ted Kremenek | 684bb09 | 2008-04-18 15:18:20 +0000 | [diff] [blame] | 419 |   my $BaseDir = shift; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 420 |    | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 421 |   die "No directory specified." if (!defined $Dir); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 422 |    | 
 | 423 |   if (! -d $Dir) { | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 424 |     Diag("No bugs found.\n"); | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 425 |     return 0; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 426 |   } | 
 | 427 |    | 
 | 428 |   opendir(DIR, $Dir); | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 429 |   my $Crashes = 0; | 
 | 430 |   my @files = grep { if ($_ eq "crashes") { $Crashes++; } | 
 | 431 |                      /^report-.*\.html$/; } readdir(DIR); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 432 |   closedir(DIR); | 
 | 433 |  | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 434 |   if (scalar(@files) == 0 and $Crashes == 0) { | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 435 |     Diag("Removing directory '$Dir' because it contains no reports.\n"); | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 436 |     system ("rm", "-fR", $Dir); | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 437 |     return 0; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 438 |   } | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 439 |    | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 440 |   # Scan each report file and build an index.   | 
 | 441 |   my @Index;     | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 442 |   foreach my $file (@files) { ScanFile(\@Index, $Dir, $file); } | 
 | 443 |    | 
| Ted Kremenek | d52e425 | 2008-08-25 20:45:07 +0000 | [diff] [blame] | 444 |   # Scan the crashes directory and use the information in the .info files | 
 | 445 |   # to update the common prefix directory. | 
 | 446 |   if (-d "$Dir/crashes") { | 
 | 447 |     opendir(DIR, "$Dir/crashes"); | 
 | 448 |     my @files = grep { /[.]info$/; } readdir(DIR); | 
 | 449 |     closedir(DIR); | 
 | 450 |     foreach my $file (@files) { | 
 | 451 |       open IN, "$Dir/crashes/$file" or DieDiag("cannot open $file\n"); | 
 | 452 |       my $Path = <IN>; | 
 | 453 |       if (defined $Path) { UpdatePrefix($Path); } | 
 | 454 |       close IN; | 
 | 455 |     }     | 
 | 456 |   } | 
 | 457 |    | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 458 |   # Generate an index.html file.   | 
 | 459 |   my $FName = "$Dir/index.html";   | 
 | 460 |   open(OUT, ">", $FName) or DieDiag("Cannot create file '$FName'\n"); | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 461 |    | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 462 |   # Print out the header. | 
 | 463 |    | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 464 | print OUT <<ENDTEXT; | 
 | 465 | <html> | 
 | 466 | <head> | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 467 | <title>${HtmlTitle}</title> | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 468 | <style type="text/css"> | 
 | 469 |  body { color:#000000; background-color:#ffffff } | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 470 |  body { font-family: Helvetica, sans-serif; font-size:9pt } | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 471 |  h1 { font-size: 14pt; } | 
 | 472 |  h2 { font-size: 12pt; } | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 473 |  table { font-size:9pt } | 
 | 474 |  table { border-spacing: 0px; border: 1px solid black } | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 475 |  th, table thead { | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 476 |    background-color:#eee; color:#666666; | 
 | 477 |    font-weight: bold; cursor: default; | 
| Ted Kremenek | bba1cf5 | 2008-04-03 05:50:51 +0000 | [diff] [blame] | 478 |    text-align:center; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 479 |    font-weight: bold; font-family: Verdana; | 
 | 480 |    white-space:nowrap; | 
| Ted Kremenek | bba1cf5 | 2008-04-03 05:50:51 +0000 | [diff] [blame] | 481 |  }  | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 482 |  .W { font-size:0px } | 
 | 483 |  th, td { padding:5px; padding-left:8px; text-align:left } | 
 | 484 |  td.SUMM_DESC { padding-left:12px } | 
 | 485 |  td.DESC { white-space:pre } | 
 | 486 |  td.Q { text-align:right } | 
 | 487 |  td { text-align:left } | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 488 |  tbody.scrollContent { overflow:auto } | 
 | 489 | } | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 490 | </style> | 
| Ted Kremenek | 12a467f | 2008-09-21 20:10:46 +0000 | [diff] [blame] | 491 | <script language='javascript' type="text/javascript"> | 
 | 492 | if (document.styleSheets && RegExp(" AppleWebKit/").test(navigator.userAgent)) | 
 | 493 | { | 
 | 494 |   var sheet = document.styleSheets[0]; | 
 | 495 |   if (sheet) { | 
 | 496 |     var rules = sheet.cssRules; | 
 | 497 |     if (rules) { | 
 | 498 |       sheet.insertRule("td.View a { white-space: nowrap; -webkit-appearance:square-button; padding-left:1em; padding-right:1em; padding-top:0.5ex; padding-bottom:0.5ex; text-decoration:none; color:black }", rules.length); | 
 | 499 |     } | 
 | 500 |   } | 
 | 501 | } | 
 | 502 | </script> | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 503 | <script src="sorttable.js"></script> | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 504 | <script language='javascript' type="text/javascript"> | 
 | 505 | function SetDisplay(RowClass, DisplayVal) | 
 | 506 | { | 
 | 507 |   var Rows = document.getElementsByTagName("tr"); | 
 | 508 |   for ( var i = 0 ; i < Rows.length; ++i ) { | 
 | 509 |     if (Rows[i].className == RowClass) { | 
 | 510 |       Rows[i].style.display = DisplayVal; | 
 | 511 |     } | 
 | 512 |   } | 
 | 513 | } | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 514 |  | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 515 | function ToggleDisplay(CheckButton, ClassName) { | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 516 |   if (CheckButton.checked) { | 
 | 517 |     SetDisplay(ClassName, ""); | 
 | 518 |   } | 
 | 519 |   else { | 
 | 520 |     SetDisplay(ClassName, "none"); | 
 | 521 |   } | 
 | 522 | } | 
 | 523 | </script> | 
 | 524 | </head> | 
 | 525 | <body> | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 526 | <h1>${HtmlTitle}</h1> | 
 | 527 |  | 
 | 528 | <table> | 
 | 529 | <tr><th>User:</th><td>${UserName}\@${HostName}</td></tr> | 
 | 530 | <tr><th>Working Directory:</th><td>${CurrentDir}</td></tr> | 
 | 531 | <tr><th>Command Line:</th><td>${CmdArgs}</td></tr> | 
 | 532 | <tr><th>Date:</th><td>${Date}</td></tr> | 
 | 533 | ENDTEXT | 
 | 534 |  | 
 | 535 | print OUT "<tr><th>Version:</th><td>${BuildName} (${BuildDate})</td></tr>\n" | 
 | 536 |   if (defined($BuildName) && defined($BuildDate)); | 
 | 537 |  | 
 | 538 | print OUT <<ENDTEXT; | 
 | 539 | </table> | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 540 | ENDTEXT | 
 | 541 |  | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 542 |   if (scalar(@files)) { | 
 | 543 |     # Print out the summary table. | 
 | 544 |     my %Totals; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 545 |  | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 546 |     for my $row ( @Index ) { | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 547 |       my $bug_type = ($row->[2]); | 
 | 548 |       my $bug_category = ($row->[1]); | 
 | 549 |       my $key = "$bug_category:$bug_type"; | 
 | 550 |  | 
 | 551 |       if (!defined $Totals{$key}) { $Totals{$key} = [1,$bug_category,$bug_type]; } | 
 | 552 |       else { $Totals{$key}->[0]++; } | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 553 |     } | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 554 |  | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 555 |     print OUT "<h2>Bug Summary</h2>"; | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 556 |  | 
 | 557 |     if (defined $BuildName) { | 
 | 558 |       print OUT "\n<p>Results in this analysis run are based on analyzer build <b>$BuildName</b>.</p>\n" | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 559 |     } | 
| Ted Kremenek | f4cdf41 | 2008-05-23 18:17:05 +0000 | [diff] [blame] | 560 |    | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 561 | print OUT <<ENDTEXT; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 562 | <table> | 
 | 563 | <thead><tr><td>Bug Type</td><td>Quantity</td><td class="sorttable_nosort">Display?</td></tr></thead> | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 564 | ENDTEXT | 
 | 565 |    | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 566 |     my $last_category; | 
 | 567 |  | 
 | 568 |     for my $key ( | 
 | 569 |       sort { | 
 | 570 |         my $x = $Totals{$a}; | 
 | 571 |         my $y = $Totals{$b}; | 
 | 572 |         my $res = $x->[1] cmp $y->[1]; | 
 | 573 |         $res = $x->[2] cmp $y->[2] if ($res == 0); | 
 | 574 |         $res | 
 | 575 |       } keys %Totals )  | 
 | 576 |     { | 
 | 577 |       my $val = $Totals{$key}; | 
 | 578 |       my $category = $val->[1]; | 
 | 579 |       if (!defined $last_category or $last_category ne $category) { | 
 | 580 |         $last_category = $category; | 
 | 581 |         print OUT "<tr><th>$category</th><th colspan=2></th></tr>\n"; | 
 | 582 |       }       | 
 | 583 |       my $x = lc $key; | 
 | 584 |       $x =~ s/[ ,'":\/()]+/_/g; | 
 | 585 |       print OUT "<tr><td class=\"SUMM_DESC\">"; | 
 | 586 |       print OUT $val->[2]; | 
 | 587 |       print OUT "</td><td>"; | 
 | 588 |       print OUT $val->[0]; | 
 | 589 |       print OUT "</td><td><center><input type=\"checkbox\" onClick=\"ToggleDisplay(this,'bt_$x');\" checked/></center></td></tr>\n"; | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 590 |     } | 
| Ted Kremenek | 6e6eff7 | 2008-04-15 20:47:02 +0000 | [diff] [blame] | 591 |  | 
 | 592 |   # Print out the table of errors. | 
 | 593 |  | 
 | 594 | print OUT <<ENDTEXT; | 
 | 595 | </table> | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 596 | <h2>Reports</h2> | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 597 |  | 
 | 598 | <table class="sortable" style="table-layout:automatic"> | 
 | 599 | <thead><tr> | 
 | 600 |   <td>Bug Group</td> | 
 | 601 |   <td class="sorttable_sorted">Bug Type<span id="sorttable_sortfwdind"> ▾</span></td> | 
| Ted Kremenek | bba1cf5 | 2008-04-03 05:50:51 +0000 | [diff] [blame] | 602 |   <td>File</td> | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 603 |   <td class="Q">Line</td> | 
 | 604 |   <td class="Q">Path Length</td> | 
| Ted Kremenek | 2645c77 | 2008-07-07 16:58:44 +0000 | [diff] [blame] | 605 |   <td class="sorttable_nosort"></td> | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 606 |   <!-- REPORTBUGCOL --> | 
 | 607 | </tr></thead> | 
 | 608 | <tbody> | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 609 | ENDTEXT | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 610 |  | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 611 |     my $prefix = GetPrefix(); | 
 | 612 |     my $regex; | 
 | 613 |     my $InFileRegex; | 
 | 614 |     my $InFilePrefix = "File:</td><td>"; | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 615 |    | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 616 |     if (defined $prefix) {  | 
 | 617 |       $regex = qr/^\Q$prefix\E/is;     | 
 | 618 |       $InFileRegex = qr/\Q$InFilePrefix$prefix\E/is; | 
 | 619 |     }     | 
| Ted Kremenek | 7a4648d | 2008-05-02 22:04:53 +0000 | [diff] [blame] | 620 |  | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 621 |     for my $row ( sort { $a->[2] cmp $b->[2] } @Index ) { | 
 | 622 |       my $x = "$row->[1]:$row->[2]"; | 
 | 623 |       $x = lc $x; | 
 | 624 |       $x =~ s/[ ,'":\/()]+/_/g; | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 625 |      | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 626 |       my $ReportFile = $row->[0]; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 627 |            | 
 | 628 |       print OUT "<tr class=\"bt_$x\">"; | 
 | 629 |       print OUT "<td class=\"DESC\">"; | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 630 |       print OUT $row->[1]; | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 631 |       print OUT "</td>"; | 
 | 632 |       print OUT "<td class=\"DESC\">"; | 
 | 633 |       print OUT $row->[2]; | 
 | 634 |       print OUT "</td>"; | 
 | 635 |        | 
 | 636 |       # Update the file prefix.       | 
 | 637 |       my $fname = $row->[3]; | 
 | 638 |       my $full_fname = $fname; | 
 | 639 |  | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 640 |       if (defined $regex) { | 
 | 641 |         $fname =~ s/$regex//; | 
 | 642 |         UpdateInFilePath("$Dir/$ReportFile", $InFileRegex, $InFilePrefix) | 
 | 643 |       } | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 644 |        | 
 | 645 |       print OUT "<td>"; | 
 | 646 |       my $has_fname = 0; | 
 | 647 |       if (-r $full_fname) { | 
 | 648 |         $has_fname = 1; | 
 | 649 |         print OUT "<a href=\"$full_fname\">" | 
 | 650 |       } | 
 | 651 |        | 
 | 652 |       my @fname = split /\//,$fname; | 
 | 653 |       if ($#fname > 0) { | 
 | 654 |         while ($#fname >= 0) { | 
 | 655 |           my $x = shift @fname; | 
 | 656 |           print OUT $x; | 
 | 657 |           if ($#fname >= 0) { | 
 | 658 |             print OUT "<span class=\"W\"> </span>/"; | 
 | 659 |           } | 
 | 660 |         } | 
 | 661 |       } | 
 | 662 |       else { | 
 | 663 |         print OUT $fname; | 
 | 664 |       } | 
 | 665 |        | 
 | 666 |       if ($has_fname) { | 
 | 667 |         print OUT "</a>" | 
 | 668 |       } | 
 | 669 |       print OUT "</td>"; | 
 | 670 |        | 
 | 671 |       # Print out the quantities. | 
 | 672 |       for my $j ( 4 .. 5 ) { | 
 | 673 |         print OUT "<td class=\"Q\">$row->[$j]</td>";         | 
 | 674 |       } | 
 | 675 |        | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 676 |       # Print the rest of the columns. | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 677 |       for (my $j = 6; $j <= $#{$row}; ++$j) { | 
 | 678 |         print OUT "<td>$row->[$j]</td>" | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 679 |       } | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 680 |  | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 681 |       # Emit the "View" link. | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 682 |       print OUT "<td class=\"View\"><a href=\"$ReportFile#EndPath\">View Report</a></td>"; | 
| Ted Kremenek | 3cea9ee | 2008-07-30 17:58:08 +0000 | [diff] [blame] | 683 |          | 
| Daniel Dunbar | e43038e | 2008-09-19 23:18:44 +0000 | [diff] [blame] | 684 |       # Emit REPORTBUG markers. | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 685 |       print OUT "\n<!-- REPORTBUG id=\"$ReportFile\" -->\n"; | 
| Daniel Dunbar | e43038e | 2008-09-19 23:18:44 +0000 | [diff] [blame] | 686 |          | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 687 |       # End the row. | 
 | 688 |       print OUT "</tr>\n"; | 
 | 689 |     } | 
 | 690 |    | 
| Ted Kremenek | ebb7413 | 2008-09-21 06:58:09 +0000 | [diff] [blame] | 691 |     print OUT "</tbody>\n</table>\n\n"; | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 692 |   } | 
 | 693 |  | 
 | 694 |   if ($Crashes) { | 
 | 695 |     # Read the crash directory for files. | 
 | 696 |     opendir(DIR, "$Dir/crashes"); | 
 | 697 |     my @files = grep { /[.]info$/ } readdir(DIR); | 
 | 698 |     closedir(DIR); | 
 | 699 |  | 
 | 700 |     if (scalar(@files)) { | 
 | 701 |       print OUT <<ENDTEXT; | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 702 | <h2>Analyzer Failures</h2> | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 703 |  | 
| Ted Kremenek | 5d31f83 | 2008-08-18 18:38:29 +0000 | [diff] [blame] | 704 | <p>The analyzer had problems processing the following files:</p> | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 705 |  | 
 | 706 | <table> | 
| Ted Kremenek | 9f9b1fd | 2008-09-12 22:49:36 +0000 | [diff] [blame] | 707 | <thead><tr><td>Problem</td><td>Source File</td><td>Preprocessed File</td><td>STDERR Output</td></tr></thead> | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 708 | ENDTEXT | 
 | 709 |    | 
 | 710 |       foreach my $file (sort @files) { | 
 | 711 |         $file =~ /(.+).info$/; | 
 | 712 |         # Get the preprocessed file. | 
 | 713 |         my $ppfile = $1; | 
 | 714 |         # Open the info file and get the name of the source file. | 
 | 715 |         open (INFO, "$Dir/crashes/$file") or | 
 | 716 |           die "Cannot open $Dir/crashes/$file\n"; | 
 | 717 |         my $srcfile = <INFO>; | 
| Ted Kremenek | 5d31f83 | 2008-08-18 18:38:29 +0000 | [diff] [blame] | 718 |         chomp $srcfile; | 
 | 719 |         my $problem = <INFO>; | 
 | 720 |         chomp $problem; | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 721 |         close (INFO); | 
 | 722 |         # Print the information in the table. | 
| Ted Kremenek | d52e425 | 2008-08-25 20:45:07 +0000 | [diff] [blame] | 723 |         my $prefix = GetPrefix(); | 
| Ted Kremenek | 9f9b1fd | 2008-09-12 22:49:36 +0000 | [diff] [blame] | 724 |         if (defined $prefix) { $srcfile =~ s/^\Q$prefix//; } | 
 | 725 |         print OUT "<tr><td>$problem</td><td>$srcfile</td><td><a href=\"crashes/$ppfile\">$ppfile</a></td><td><a href=\"crashes/$ppfile.stderr.txt\">$ppfile.stderr.txt</a></td></tr>\n"; | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 726 |       } | 
 | 727 |  | 
 | 728 |       print OUT <<ENDTEXT; | 
 | 729 | </table> | 
 | 730 | <p>Please consider submitting preprocessed files as <a href="http://clang.llvm.org/StaticAnalysisUsage.html#filingbugs">bug reports</a>.</p> | 
 | 731 | ENDTEXT | 
 | 732 |     } | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 733 |   } | 
 | 734 |    | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 735 |   print OUT "</body></html>\n";   | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 736 |   close(OUT); | 
| Ted Kremenek | 22d6a63 | 2008-04-02 20:43:36 +0000 | [diff] [blame] | 737 |   CopyJS($Dir); | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 738 |  | 
 | 739 |   # Make sure $Dir and $BaseDir are world readable/executable. | 
 | 740 |   system("chmod", "755", $Dir); | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 741 |   if (defined $BaseDir) { system("chmod", "755", $BaseDir); } | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 742 |  | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 743 |   my $Num = scalar(@Index); | 
| Ted Kremenek | 150c212 | 2008-07-11 19:15:05 +0000 | [diff] [blame] | 744 |   Diag("$Num bugs found.\n"); | 
 | 745 |   if ($Num > 0 && -r "$Dir/index.html") { | 
 | 746 |     Diag("Open '$Dir/index.html' to examine bug reports.\n"); | 
 | 747 |   } | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 748 |    | 
| Ted Kremenek | 991c54b | 2008-08-08 20:46:42 +0000 | [diff] [blame] | 749 |   DiagCrashes($Dir) if ($Crashes); | 
 | 750 |    | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 751 |   return $Num; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 752 | } | 
 | 753 |  | 
 | 754 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 755 | # RunBuildCommand - Run the build command. | 
 | 756 | ##----------------------------------------------------------------------------## | 
 | 757 |  | 
| Ted Kremenek | 6b62898 | 2008-04-30 23:47:12 +0000 | [diff] [blame] | 758 | sub AddIfNotPresent { | 
 | 759 |   my $Args = shift; | 
 | 760 |   my $Arg = shift;   | 
 | 761 |   my $found = 0; | 
 | 762 |    | 
 | 763 |   foreach my $k (@$Args) { | 
 | 764 |     if ($k eq $Arg) { | 
 | 765 |       $found = 1; | 
 | 766 |       last; | 
 | 767 |     } | 
 | 768 |   } | 
 | 769 |    | 
 | 770 |   if ($found == 0) { | 
 | 771 |     push @$Args, $Arg; | 
 | 772 |   } | 
 | 773 | } | 
 | 774 |  | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 775 | sub RunBuildCommand { | 
 | 776 |    | 
 | 777 |   my $Args = shift; | 
| Ted Kremenek | 7442ca6 | 2008-04-02 16:04:51 +0000 | [diff] [blame] | 778 |   my $IgnoreErrors = shift; | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 779 |   my $Cmd = $Args->[0]; | 
| Ted Kremenek | 6195c37 | 2008-06-02 21:52:47 +0000 | [diff] [blame] | 780 |   my $CCAnalyzer = shift; | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 781 |    | 
| Ted Kremenek | 3301cb1 | 2008-06-30 18:18:16 +0000 | [diff] [blame] | 782 |   # Get only the part of the command after the last '/'. | 
 | 783 |   if ($Cmd =~ /\/([^\/]+)$/) { | 
 | 784 |     $Cmd = $1; | 
 | 785 |   } | 
 | 786 |    | 
| Ted Kremenek | 63c2017 | 2008-08-04 17:34:06 +0000 | [diff] [blame] | 787 |   if ($Cmd eq "gcc" or $Cmd eq "cc" or $Cmd eq "llvm-gcc"  | 
 | 788 |    or $Cmd eq "ccc-analyzer") { | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 789 |     shift @$Args; | 
| Ted Kremenek | 6195c37 | 2008-06-02 21:52:47 +0000 | [diff] [blame] | 790 |     unshift @$Args, $CCAnalyzer; | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 791 |   } | 
| Ted Kremenek | 7442ca6 | 2008-04-02 16:04:51 +0000 | [diff] [blame] | 792 |   elsif ($IgnoreErrors) { | 
 | 793 |     if ($Cmd eq "make" or $Cmd eq "gmake") { | 
| Ted Kremenek | 6b62898 | 2008-04-30 23:47:12 +0000 | [diff] [blame] | 794 |       AddIfNotPresent($Args,"-k"); | 
| Ted Kremenek | 8912b54 | 2008-05-13 21:28:02 +0000 | [diff] [blame] | 795 |       AddIfNotPresent($Args,"-i"); | 
| Ted Kremenek | 7442ca6 | 2008-04-02 16:04:51 +0000 | [diff] [blame] | 796 |     } | 
 | 797 |     elsif ($Cmd eq "xcodebuild") { | 
| Ted Kremenek | 6b62898 | 2008-04-30 23:47:12 +0000 | [diff] [blame] | 798 |       AddIfNotPresent($Args,"-PBXBuildsContinueAfterErrors=YES"); | 
| Ted Kremenek | 7442ca6 | 2008-04-02 16:04:51 +0000 | [diff] [blame] | 799 |     } | 
| Ted Kremenek | 6b62898 | 2008-04-30 23:47:12 +0000 | [diff] [blame] | 800 |   }  | 
 | 801 |    | 
| Ted Kremenek | 6b62898 | 2008-04-30 23:47:12 +0000 | [diff] [blame] | 802 |   if ($Cmd eq "xcodebuild") { | 
| Ted Kremenek | cfd4c7b | 2008-05-23 22:18:16 +0000 | [diff] [blame] | 803 |     # Disable distributed builds for xcodebuild. | 
| Ted Kremenek | 6b62898 | 2008-04-30 23:47:12 +0000 | [diff] [blame] | 804 |     AddIfNotPresent($Args,"-nodistribute"); | 
| Ted Kremenek | cfd4c7b | 2008-05-23 22:18:16 +0000 | [diff] [blame] | 805 |  | 
 | 806 |     # Disable PCH files until clang supports them. | 
 | 807 |     AddIfNotPresent($Args,"GCC_PRECOMPILE_PREFIX_HEADER=NO"); | 
| Ted Kremenek | 915e972 | 2008-05-27 23:18:07 +0000 | [diff] [blame] | 808 |      | 
 | 809 |     # When 'CC' is set, xcodebuild uses it to do all linking, even if we are | 
 | 810 |     # linking C++ object files.  Set 'LDPLUSPLUS' so that xcodebuild uses 'g++' | 
 | 811 |     # when linking such files. | 
| Ted Kremenek | 95aa105 | 2008-09-04 17:52:41 +0000 | [diff] [blame] | 812 |     die if (!defined $CXX); | 
 | 813 |     my $LDPLUSPLUS = `which $CXX`; | 
| Ted Kremenek | 915e972 | 2008-05-27 23:18:07 +0000 | [diff] [blame] | 814 |     $LDPLUSPLUS =~ s/\015?\012//;  # strip newlines | 
 | 815 |     $ENV{'LDPLUSPLUS'} = $LDPLUSPLUS;     | 
| Ted Kremenek | 6b62898 | 2008-04-30 23:47:12 +0000 | [diff] [blame] | 816 |   } | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 817 |    | 
| Ted Kremenek | 5a4ddaf | 2008-08-25 20:10:45 +0000 | [diff] [blame] | 818 |   return (system(@$Args) >> 8); | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 819 | } | 
 | 820 |  | 
| Ted Kremenek | dab1110 | 2008-04-02 04:43:42 +0000 | [diff] [blame] | 821 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 822 | # DisplayHelp - Utility function to display all help options. | 
 | 823 | ##----------------------------------------------------------------------------## | 
 | 824 |  | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 825 | sub DisplayHelp { | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 826 |    | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 827 | print <<ENDTEXT; | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 828 | USAGE: $Prog [options] <build command> [build options] | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 829 |  | 
| Ted Kremenek | f4cdf41 | 2008-05-23 18:17:05 +0000 | [diff] [blame] | 830 | ENDTEXT | 
 | 831 |  | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 832 |   if (defined $BuildName) { | 
| Ted Kremenek | f4cdf41 | 2008-05-23 18:17:05 +0000 | [diff] [blame] | 833 |     print "ANALYZER BUILD: $BuildName ($BuildDate)\n\n"; | 
 | 834 |   } | 
 | 835 |  | 
 | 836 | print <<ENDTEXT; | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 837 | OPTIONS: | 
 | 838 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 839 |  -o             - Target directory for HTML report files.  Subdirectories | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 840 |                   will be created as needed to represent separate "runs" of | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 841 |                   the analyzer.  If this option is not specified, a directory | 
 | 842 |                   is created in /tmp to store the reports. | 
| Ted Kremenek | 1262fc4 | 2008-05-14 20:10:33 +0000 | [diff] [blame] | 843 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 844 |  -h             - Display this message. | 
 | 845 |  --help | 
| Ted Kremenek | 1262fc4 | 2008-05-14 20:10:33 +0000 | [diff] [blame] | 846 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 847 |  -k             - Add a "keep on going" option to the specified build command. | 
 | 848 |  --keep-going     This option currently supports make and xcodebuild. | 
| Ted Kremenek | f02e8db | 2008-04-02 16:41:25 +0000 | [diff] [blame] | 849 |                   This is a convenience option; one can specify this | 
 | 850 |                   behavior directly using build options. | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 851 |  | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 852 |  --html-title [title]       - Specify the title used on generated HTML pages. | 
 | 853 |  --html-title=[title]         If not specified, a default title will be used. | 
 | 854 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 855 |  --status-bugs  - By default, the exit status of $Prog is the same as the | 
 | 856 |                   executed build command.  Specifying this option causes the | 
 | 857 |                   exit status of $Prog to be 1 if it found potential bugs | 
 | 858 |                   and 0 otherwise. | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 859 |  | 
| Ted Kremenek | 386c693 | 2008-09-03 17:59:35 +0000 | [diff] [blame] | 860 |  --use-cc [compiler path]   - By default, $Prog uses 'gcc' to compile and link | 
 | 861 |  --use-cc=[compiler path]     your C and Objective-C code. Use this option | 
 | 862 |                               to specify an alternate compiler. | 
 | 863 |  | 
 | 864 |  --use-c++ [compiler path]  - By default, $Prog uses 'g++' to compile and link | 
 | 865 |  --use-c++=[compiler path]    your C++ and Objective-C++ code. Use this option | 
 | 866 |                               to specify an alternate compiler. | 
| Ted Kremenek | f17ef3c | 2008-08-21 21:47:09 +0000 | [diff] [blame] | 867 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 868 |  -v             - Verbose output from $Prog and the analyzer. | 
| Ted Kremenek | 386c693 | 2008-09-03 17:59:35 +0000 | [diff] [blame] | 869 |                   A second and third '-v' increases verbosity. | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 870 |  | 
 | 871 |  -V             - View analysis results in a web browser when the build | 
 | 872 |  --view           completes. | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 873 |  | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 874 |  | 
| Ted Kremenek | 386c693 | 2008-09-03 17:59:35 +0000 | [diff] [blame] | 875 | AVAILABLE ANALYSES (multiple analyses may be specified): | 
| Ted Kremenek | d52e425 | 2008-08-25 20:45:07 +0000 | [diff] [blame] | 876 |  | 
 | 877 | ENDTEXT | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 878 |  | 
 | 879 |   foreach my $Analysis (sort keys %AvailableAnalyses) { | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 880 |     if (defined $AnalysesDefaultEnabled{$Analysis}) { | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 881 |       print " (+)"; | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 882 |     } | 
 | 883 |     else { | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 884 |       print "    "; | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 885 |     } | 
 | 886 |      | 
 | 887 |     print " $Analysis  $AvailableAnalyses{$Analysis}\n"; | 
 | 888 |   } | 
 | 889 |    | 
 | 890 | print <<ENDTEXT | 
 | 891 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 892 |  NOTE: "(+)" indicates that an analysis is enabled by default unless one | 
 | 893 |        or more analysis options are specified | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 894 |  | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 895 | BUILD OPTIONS | 
 | 896 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 897 |  You can specify any build option acceptable to the build command. | 
| Ted Kremenek | 39eefde | 2008-04-02 16:47:27 +0000 | [diff] [blame] | 898 |  | 
| Ted Kremenek | 5744dc2 | 2008-04-02 18:03:36 +0000 | [diff] [blame] | 899 | EXAMPLE | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 900 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 901 |  $Prog -o /tmp/myhtmldir make -j4 | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 902 |       | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 903 |  The above example causes analysis reports to be deposited into | 
 | 904 |  a subdirectory of "/tmp/myhtmldir" and to run "make" with the "-j4" option. | 
 | 905 |  A different subdirectory is created each time $Prog analyzes a project. | 
 | 906 |  The analyzer should support most parallel builds, but not distributed builds. | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 907 |  | 
 | 908 | ENDTEXT | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 909 | } | 
 | 910 |  | 
 | 911 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 912 | # HtmlEscape - HTML entity encode characters that are special in HTML | 
 | 913 | ##----------------------------------------------------------------------------## | 
 | 914 |  | 
 | 915 | sub HtmlEscape { | 
 | 916 |   # copy argument to new variable so we don't clobber the original | 
 | 917 |   my $arg = shift || ''; | 
 | 918 |   my $tmp = $arg; | 
 | 919 |  | 
 | 920 |   $tmp =~ s/([\<\>\'\"])/sprintf("&#%02x;", chr($1))/ge; | 
 | 921 |  | 
 | 922 |   return $tmp; | 
 | 923 | } | 
 | 924 |  | 
 | 925 | ##----------------------------------------------------------------------------## | 
 | 926 | # ShellEscape - backslash escape characters that are special to the shell | 
 | 927 | ##----------------------------------------------------------------------------## | 
 | 928 |  | 
 | 929 | sub ShellEscape { | 
 | 930 |   # copy argument to new variable so we don't clobber the original | 
 | 931 |   my $arg = shift || ''; | 
 | 932 |   my $tmp = $arg; | 
 | 933 |  | 
 | 934 |   $tmp =~ s/([\!\;\\\'\"\`\<\>\|\s\(\)\[\]\?\#\$\^\&\*\=])/\\$1/g; | 
 | 935 |  | 
 | 936 |   return $tmp; | 
 | 937 | } | 
 | 938 |  | 
 | 939 | ##----------------------------------------------------------------------------## | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 940 | # Process command-line arguments. | 
 | 941 | ##----------------------------------------------------------------------------## | 
 | 942 |  | 
 | 943 | my $HtmlDir;           # Parent directory to store HTML files. | 
 | 944 | my $IgnoreErrors = 0;  # Ignore build errors. | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 945 | my $ViewResults  = 0;  # View results when the build terminates. | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 946 | my $ExitStatusFoundBugs = 0; # Exit status reflects whether bugs were found | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 947 | my @AnalysesToRun; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 948 |  | 
 | 949 | if (!@ARGV) { | 
 | 950 |   DisplayHelp(); | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 951 |   exit 1; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 952 | } | 
 | 953 |  | 
 | 954 | while (@ARGV) { | 
 | 955 |    | 
 | 956 |   # Scan for options we recognize. | 
 | 957 |    | 
 | 958 |   my $arg = $ARGV[0]; | 
 | 959 |  | 
| Sam Bishop | 2f2418e | 2008-04-03 14:29:47 +0000 | [diff] [blame] | 960 |   if ($arg eq "-h" or $arg eq "--help") { | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 961 |     DisplayHelp(); | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 962 |     exit 0; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 963 |   } | 
 | 964 |    | 
| Ted Kremenek | fc1d340 | 2008-08-04 18:15:26 +0000 | [diff] [blame] | 965 |   if (defined $AvailableAnalyses{$arg}) { | 
| Ted Kremenek | 1262fc4 | 2008-05-14 20:10:33 +0000 | [diff] [blame] | 966 |     shift @ARGV; | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 967 |     push @AnalysesToRun, $arg; | 
| Ted Kremenek | 1262fc4 | 2008-05-14 20:10:33 +0000 | [diff] [blame] | 968 |     next; | 
 | 969 |   } | 
 | 970 |    | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 971 |   if ($arg eq "-o") { | 
 | 972 |     shift @ARGV; | 
 | 973 |          | 
 | 974 |     if (!@ARGV) { | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 975 |       DieDiag("'-o' option requires a target directory name.\n"); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 976 |     } | 
 | 977 |      | 
 | 978 |     $HtmlDir = shift @ARGV; | 
 | 979 |     next; | 
 | 980 |   } | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 981 |  | 
 | 982 |   if ($arg =~ /^--html-title(=(.+))?$/) { | 
 | 983 |     shift @ARGV; | 
 | 984 |  | 
 | 985 |     if ($2 eq '') { | 
 | 986 |       if (!@ARGV) { | 
 | 987 |         DieDiag("'--html-title' option requires a string.\n"); | 
 | 988 |       } | 
 | 989 |  | 
 | 990 |       $HtmlTitle = shift @ARGV; | 
 | 991 |     } else { | 
 | 992 |       $HtmlTitle = $2; | 
 | 993 |     } | 
 | 994 |  | 
 | 995 |     next; | 
 | 996 |   } | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 997 |    | 
| Ted Kremenek | 2b74ab6 | 2008-04-01 21:22:03 +0000 | [diff] [blame] | 998 |   if ($arg eq "-k" or $arg eq "--keep-going") { | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 999 |     shift @ARGV; | 
 | 1000 |     $IgnoreErrors = 1; | 
 | 1001 |     next; | 
 | 1002 |   } | 
 | 1003 |    | 
| Ted Kremenek | f17ef3c | 2008-08-21 21:47:09 +0000 | [diff] [blame] | 1004 |   if ($arg =~ /^--use-cc(=(.+))?$/) { | 
 | 1005 |     shift @ARGV; | 
 | 1006 |     my $cc; | 
 | 1007 |      | 
 | 1008 |     if ($2 eq "") { | 
 | 1009 |       if (!@ARGV) { | 
 | 1010 |         DieDiag("'--use-cc' option requires a compiler executable name.\n"); | 
 | 1011 |       } | 
 | 1012 |       $cc = shift @ARGV; | 
 | 1013 |     } | 
 | 1014 |     else { | 
 | 1015 |       $cc = $2; | 
 | 1016 |     } | 
 | 1017 |      | 
 | 1018 |     $ENV{"CCC_CC"} = $cc; | 
 | 1019 |     next; | 
 | 1020 |   } | 
 | 1021 |    | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 1022 |   if ($arg =~ /^--use-c\+\+(=(.+))?$/) { | 
| Ted Kremenek | 386c693 | 2008-09-03 17:59:35 +0000 | [diff] [blame] | 1023 |     shift @ARGV; | 
 | 1024 |      | 
 | 1025 |     if ($2 eq "") { | 
 | 1026 |       if (!@ARGV) { | 
 | 1027 |         DieDiag("'--use-c++' option requires a compiler executable name.\n"); | 
 | 1028 |       } | 
 | 1029 |       $CXX = shift @ARGV; | 
 | 1030 |     } | 
 | 1031 |     else { | 
 | 1032 |       $CXX = $2; | 
 | 1033 |     } | 
 | 1034 |     next; | 
 | 1035 |   } | 
 | 1036 |    | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1037 |   if ($arg eq "-v") { | 
 | 1038 |     shift @ARGV; | 
 | 1039 |     $Verbose++; | 
 | 1040 |     next; | 
 | 1041 |   } | 
 | 1042 |    | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 1043 |   if ($arg eq "-V" or $arg eq "--view") { | 
 | 1044 |     shift @ARGV; | 
 | 1045 |     $ViewResults = 1;     | 
 | 1046 |     next; | 
 | 1047 |   } | 
 | 1048 |    | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 1049 |   if ($arg eq "--status-bugs") { | 
 | 1050 |     shift @ARGV; | 
 | 1051 |     $ExitStatusFoundBugs = 1; | 
 | 1052 |     next; | 
 | 1053 |   } | 
 | 1054 |    | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 1055 |   DieDiag("unrecognized option '$arg'\n") if ($arg =~ /^-/); | 
| Ted Kremenek | 0062ad4 | 2008-04-02 16:35:01 +0000 | [diff] [blame] | 1056 |    | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1057 |   last; | 
 | 1058 | } | 
 | 1059 |  | 
 | 1060 | if (!@ARGV) { | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 1061 |   Diag("No build command specified.\n\n"); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1062 |   DisplayHelp(); | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 1063 |   exit 1; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1064 | } | 
 | 1065 |  | 
| Ted Kremenek | 7cba112 | 2008-09-22 01:35:58 +0000 | [diff] [blame] | 1066 | $CmdArgs = HtmlEscape(join(' ', map(ShellEscape($_), @ARGV))); | 
 | 1067 | $HtmlTitle = "${CurrentDirSuffix} - scan-build results" | 
 | 1068 |   unless (defined($HtmlTitle)); | 
| Ted Kremenek | 386c693 | 2008-09-03 17:59:35 +0000 | [diff] [blame] | 1069 |  | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1070 | # Determine the output directory for the HTML reports. | 
| Ted Kremenek | 684bb09 | 2008-04-18 15:18:20 +0000 | [diff] [blame] | 1071 | my $BaseDir = $HtmlDir; | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 1072 | $HtmlDir = GetHTMLRunDir($HtmlDir); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1073 |  | 
 | 1074 | # Set the appropriate environment variables. | 
| Sam Bishop | a0e2266 | 2008-04-02 03:35:43 +0000 | [diff] [blame] | 1075 | SetHtmlEnv(\@ARGV, $HtmlDir); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1076 |  | 
| Ted Kremenek | 0b6c153 | 2008-04-08 20:22:12 +0000 | [diff] [blame] | 1077 | my $Cmd = "$RealBin/ccc-analyzer"; | 
 | 1078 |  | 
| Ted Kremenek | 23cfca3 | 2008-06-16 22:40:14 +0000 | [diff] [blame] | 1079 | DieDiag("Executable 'ccc-analyzer' does not exist at '$Cmd'\n") | 
| Ted Kremenek | 0b6c153 | 2008-04-08 20:22:12 +0000 | [diff] [blame] | 1080 |   if (! -x $Cmd); | 
| Ted Kremenek | f22eacb | 2008-04-18 22:00:56 +0000 | [diff] [blame] | 1081 |  | 
| Ted Kremenek | b7770c0 | 2008-07-15 17:06:13 +0000 | [diff] [blame] | 1082 | if (! -x $ClangSB) { | 
 | 1083 |   Diag("'clang' executable not found in '$RealBin'.\n"); | 
 | 1084 |   Diag("Using 'clang' from path.\n"); | 
| Ted Kremenek | f22eacb | 2008-04-18 22:00:56 +0000 | [diff] [blame] | 1085 | } | 
| Ted Kremenek | 0b6c153 | 2008-04-08 20:22:12 +0000 | [diff] [blame] | 1086 |  | 
| Ted Kremenek | 95aa105 | 2008-09-04 17:52:41 +0000 | [diff] [blame] | 1087 | if (defined $CXX) { | 
 | 1088 |   $ENV{'CXX'} = $CXX; | 
 | 1089 | } | 
 | 1090 | else { | 
 | 1091 |   $CXX = 'g++';  # This variable is used by other parts of scan-build | 
 | 1092 |                  # that need to know a default C++ compiler to fall back to. | 
 | 1093 | } | 
 | 1094 |    | 
| Ted Kremenek | 4f4b17d | 2008-04-03 20:08:18 +0000 | [diff] [blame] | 1095 | $ENV{'CC'} = $Cmd; | 
| Ted Kremenek | f22eacb | 2008-04-18 22:00:56 +0000 | [diff] [blame] | 1096 | $ENV{'CLANG'} = $Clang; | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1097 |  | 
 | 1098 | if ($Verbose >= 2) { | 
 | 1099 |   $ENV{'CCC_ANALYZER_VERBOSE'} = 1; | 
 | 1100 | } | 
 | 1101 |  | 
| Ted Kremenek | a9525c9 | 2008-05-12 22:07:14 +0000 | [diff] [blame] | 1102 | if ($Verbose >= 3) { | 
 | 1103 |   $ENV{'CCC_ANALYZER_LOG'} = 1; | 
 | 1104 | } | 
 | 1105 |  | 
| Ted Kremenek | 9012599 | 2008-07-15 23:41:32 +0000 | [diff] [blame] | 1106 | if (scalar(@AnalysesToRun) == 0) { | 
 | 1107 |   foreach my $key (keys %AnalysesDefaultEnabled) { | 
 | 1108 |     push @AnalysesToRun,$key; | 
 | 1109 |   } | 
| Ted Kremenek | 0100678 | 2008-07-02 23:16:10 +0000 | [diff] [blame] | 1110 | } | 
| Ted Kremenek | 1262fc4 | 2008-05-14 20:10:33 +0000 | [diff] [blame] | 1111 |  | 
| Ted Kremenek | 9012599 | 2008-07-15 23:41:32 +0000 | [diff] [blame] | 1112 | $ENV{'CCC_ANALYZER_ANALYSIS'} = join ' ',@AnalysesToRun; | 
 | 1113 |  | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1114 | # Run the build. | 
| Ted Kremenek | 5656a98 | 2008-07-15 17:09:28 +0000 | [diff] [blame] | 1115 | my $ExitStatus = RunBuildCommand(\@ARGV, $IgnoreErrors, $Cmd); | 
| Ted Kremenek | 9cc8c2c | 2008-04-01 20:47:38 +0000 | [diff] [blame] | 1116 |  | 
 | 1117 | # Postprocess the HTML directory. | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 1118 | my $NumBugs = Postprocess($HtmlDir, $BaseDir); | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 1119 |  | 
 | 1120 | if ($ViewResults and -r "$HtmlDir/index.html") { | 
 | 1121 |   # Only works on Mac OS X (for now). | 
 | 1122 |   print "Viewing analysis results: '$HtmlDir/index.html'\n"; | 
| Ted Kremenek | 20161e9 | 2008-07-15 20:18:21 +0000 | [diff] [blame] | 1123 |   system("open", "$HtmlDir/index.html"); | 
| Ted Kremenek | 7f8a325 | 2008-04-02 18:42:49 +0000 | [diff] [blame] | 1124 | } | 
| Ted Kremenek | 5656a98 | 2008-07-15 17:09:28 +0000 | [diff] [blame] | 1125 |  | 
| Ted Kremenek | 363dc3f | 2008-07-15 22:03:09 +0000 | [diff] [blame] | 1126 | if ($ExitStatusFoundBugs) { | 
 | 1127 |   exit 1 if ($NumBugs > 0); | 
 | 1128 |   exit 0; | 
 | 1129 | } | 
 | 1130 |  | 
| Ted Kremenek | 5656a98 | 2008-07-15 17:09:28 +0000 | [diff] [blame] | 1131 | exit $ExitStatus; | 
 | 1132 |  |