1 #!/usr/bin/perl -w 2 # 3 # Copyright (c) International Business Machines Corp., 2002 4 # 5 # This program is free software; you can redistribute it and/or modify 6 # it under the terms of the GNU General Public License as published by 7 # the Free Software Foundation; either version 2 of the License, or (at 8 # your option) any later version. 9 # 10 # This program is distributed in the hope that it will be useful, but 11 # WITHOUT ANY WARRANTY; without even the implied warranty of 12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 13 # General Public License for more details. 14 # 15 # You should have received a copy of the GNU General Public License 16 # along with this program; if not, write to the Free Software 17 # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 18 # 19 # 20 # genpng 21 # 22 # This script creates an overview PNG image of a source code file by 23 # representing each source code character by a single pixel. 24 # 25 # Note that the PERL module GD.pm is required for this script to work. 26 # It may be obtained from http://www.cpan.org 27 # 28 # History: 29 # 2002-08-26: created by Peter Oberparleiter <Peter.Oberparleiter (at] de.ibm.com> 30 # 31 32 use strict; 33 use File::Basename; 34 use Getopt::Long; 35 36 37 # Constants 38 our $lcov_version = 'LCOV version 1.9'; 39 our $lcov_url = "http://ltp.sourceforge.net/coverage/lcov.php"; 40 our $tool_name = basename($0); 41 42 43 # Prototypes 44 sub gen_png($$$@); 45 sub check_and_load_module($); 46 sub genpng_print_usage(*); 47 sub genpng_process_file($$$$); 48 sub genpng_warn_handler($); 49 sub genpng_die_handler($); 50 51 52 # 53 # Code entry point 54 # 55 56 # Prettify version string 57 $lcov_version =~ s/\$\s*Revision\s*:?\s*(\S+)\s*\$/$1/; 58 59 # Check whether required module GD.pm is installed 60 if (check_and_load_module("GD")) 61 { 62 # Note: cannot use die() to print this message because inserting this 63 # code into another script via do() would not fail as required! 64 print(STDERR <<END_OF_TEXT) 65 ERROR: required module GD.pm not found on this system (see www.cpan.org). 66 END_OF_TEXT 67 ; 68 exit(2); 69 } 70 71 # Check whether we're called from the command line or from another script 72 if (!caller) 73 { 74 my $filename; 75 my $tab_size = 4; 76 my $width = 80; 77 my $out_filename; 78 my $help; 79 my $version; 80 81 $SIG{__WARN__} = \&genpng_warn_handler; 82 $SIG{__DIE__} = \&genpng_die_handler; 83 84 # Parse command line options 85 if (!GetOptions("tab-size=i" => \$tab_size, 86 "width=i" => \$width, 87 "output-filename=s" => \$out_filename, 88 "help" => \$help, 89 "version" => \$version)) 90 { 91 print(STDERR "Use $tool_name --help to get usage ". 92 "information\n"); 93 exit(1); 94 } 95 96 $filename = $ARGV[0]; 97 98 # Check for help flag 99 if ($help) 100 { 101 genpng_print_usage(*STDOUT); 102 exit(0); 103 } 104 105 # Check for version flag 106 if ($version) 107 { 108 print("$tool_name: $lcov_version\n"); 109 exit(0); 110 } 111 112 # Check options 113 if (!$filename) 114 { 115 die("No filename specified\n"); 116 } 117 118 # Check for output filename 119 if (!$out_filename) 120 { 121 $out_filename = "$filename.png"; 122 } 123 124 genpng_process_file($filename, $out_filename, $width, $tab_size); 125 exit(0); 126 } 127 128 129 # 130 # genpng_print_usage(handle) 131 # 132 # Write out command line usage information to given filehandle. 133 # 134 135 sub genpng_print_usage(*) 136 { 137 local *HANDLE = $_[0]; 138 139 print(HANDLE <<END_OF_USAGE) 140 Usage: $tool_name [OPTIONS] SOURCEFILE 141 142 Create an overview image for a given source code file of either plain text 143 or .gcov file format. 144 145 -h, --help Print this help, then exit 146 -v, --version Print version number, then exit 147 -t, --tab-size TABSIZE Use TABSIZE spaces in place of tab 148 -w, --width WIDTH Set width of output image to WIDTH pixel 149 -o, --output-filename FILENAME Write image to FILENAME 150 151 For more information see: $lcov_url 152 END_OF_USAGE 153 ; 154 } 155 156 157 # 158 # check_and_load_module(module_name) 159 # 160 # Check whether a module by the given name is installed on this system 161 # and make it known to the interpreter if available. Return undefined if it 162 # is installed, an error message otherwise. 163 # 164 165 sub check_and_load_module($) 166 { 167 eval("use $_[0];"); 168 return $@; 169 } 170 171 172 # 173 # genpng_process_file(filename, out_filename, width, tab_size) 174 # 175 176 sub genpng_process_file($$$$) 177 { 178 my $filename = $_[0]; 179 my $out_filename = $_[1]; 180 my $width = $_[2]; 181 my $tab_size = $_[3]; 182 local *HANDLE; 183 my @source; 184 185 open(HANDLE, "<$filename") 186 or die("ERROR: cannot open $filename!\n"); 187 188 # Check for .gcov filename extension 189 if ($filename =~ /^(.*).gcov$/) 190 { 191 # Assume gcov text format 192 while (<HANDLE>) 193 { 194 if (/^\t\t(.*)$/) 195 { 196 # Uninstrumented line 197 push(@source, ":$1"); 198 } 199 elsif (/^ ###### (.*)$/) 200 { 201 # Line with zero execution count 202 push(@source, "0:$1"); 203 } 204 elsif (/^( *)(\d*) (.*)$/) 205 { 206 # Line with positive execution count 207 push(@source, "$2:$3"); 208 } 209 } 210 } 211 else 212 { 213 # Plain text file 214 while (<HANDLE>) { push(@source, ":$_"); } 215 } 216 close(HANDLE); 217 218 gen_png($out_filename, $width, $tab_size, @source); 219 } 220 221 222 # 223 # gen_png(filename, width, tab_size, source) 224 # 225 # Write an overview PNG file to FILENAME. Source code is defined by SOURCE 226 # which is a list of lines <count>:<source code> per source code line. 227 # The output image will be made up of one pixel per character of source, 228 # coloring will be done according to execution counts. WIDTH defines the 229 # image width. TAB_SIZE specifies the number of spaces to use as replacement 230 # string for tabulator signs in source code text. 231 # 232 # Die on error. 233 # 234 235 sub gen_png($$$@) 236 { 237 my $filename = shift(@_); # Filename for PNG file 238 my $overview_width = shift(@_); # Imagewidth for image 239 my $tab_size = shift(@_); # Replacement string for tab signs 240 my @source = @_; # Source code as passed via argument 2 241 my $height = scalar(@source); # Height as define by source size 242 my $overview; # Source code overview image data 243 my $col_plain_back; # Color for overview background 244 my $col_plain_text; # Color for uninstrumented text 245 my $col_cov_back; # Color for background of covered lines 246 my $col_cov_text; # Color for text of covered lines 247 my $col_nocov_back; # Color for background of lines which 248 # were not covered (count == 0) 249 my $col_nocov_text; # Color for test of lines which were not 250 # covered (count == 0) 251 my $col_hi_back; # Color for background of highlighted lines 252 my $col_hi_text; # Color for text of highlighted lines 253 my $line; # Current line during iteration 254 my $row = 0; # Current row number during iteration 255 my $column; # Current column number during iteration 256 my $color_text; # Current text color during iteration 257 my $color_back; # Current background color during iteration 258 my $last_count; # Count of last processed line 259 my $count; # Count of current line 260 my $source; # Source code of current line 261 my $replacement; # Replacement string for tabulator chars 262 local *PNG_HANDLE; # Handle for output PNG file 263 264 # Create image 265 $overview = new GD::Image($overview_width, $height) 266 or die("ERROR: cannot allocate overview image!\n"); 267 268 # Define colors 269 $col_plain_back = $overview->colorAllocate(0xff, 0xff, 0xff); 270 $col_plain_text = $overview->colorAllocate(0xaa, 0xaa, 0xaa); 271 $col_cov_back = $overview->colorAllocate(0xaa, 0xa7, 0xef); 272 $col_cov_text = $overview->colorAllocate(0x5d, 0x5d, 0xea); 273 $col_nocov_back = $overview->colorAllocate(0xff, 0x00, 0x00); 274 $col_nocov_text = $overview->colorAllocate(0xaa, 0x00, 0x00); 275 $col_hi_back = $overview->colorAllocate(0x00, 0xff, 0x00); 276 $col_hi_text = $overview->colorAllocate(0x00, 0xaa, 0x00); 277 278 # Visualize each line 279 foreach $line (@source) 280 { 281 # Replace tabs with spaces to keep consistent with source 282 # code view 283 while ($line =~ /^([^\t]*)(\t)/) 284 { 285 $replacement = " "x($tab_size - ((length($1) - 1) % 286 $tab_size)); 287 $line =~ s/^([^\t]*)(\t)/$1$replacement/; 288 } 289 290 # Skip lines which do not follow the <count>:<line> 291 # specification, otherwise $1 = count, $2 = source code 292 if (!($line =~ /(\*?)(\d*):(.*)$/)) { next; } 293 $count = $2; 294 $source = $3; 295 296 # Decide which color pair to use 297 298 # If this line was not instrumented but the one before was, 299 # take the color of that line to widen color areas in 300 # resulting image 301 if (($count eq "") && defined($last_count) && 302 ($last_count ne "")) 303 { 304 $count = $last_count; 305 } 306 307 if ($count eq "") 308 { 309 # Line was not instrumented 310 $color_text = $col_plain_text; 311 $color_back = $col_plain_back; 312 } 313 elsif ($count == 0) 314 { 315 # Line was instrumented but not executed 316 $color_text = $col_nocov_text; 317 $color_back = $col_nocov_back; 318 } 319 elsif ($1 eq "*") 320 { 321 # Line was highlighted 322 $color_text = $col_hi_text; 323 $color_back = $col_hi_back; 324 } 325 else 326 { 327 # Line was instrumented and executed 328 $color_text = $col_cov_text; 329 $color_back = $col_cov_back; 330 } 331 332 # Write one pixel for each source character 333 $column = 0; 334 foreach (split("", $source)) 335 { 336 # Check for width 337 if ($column >= $overview_width) { last; } 338 339 if ($_ eq " ") 340 { 341 # Space 342 $overview->setPixel($column++, $row, 343 $color_back); 344 } 345 else 346 { 347 # Text 348 $overview->setPixel($column++, $row, 349 $color_text); 350 } 351 } 352 353 # Fill rest of line 354 while ($column < $overview_width) 355 { 356 $overview->setPixel($column++, $row, $color_back); 357 } 358 359 $last_count = $2; 360 361 $row++; 362 } 363 364 # Write PNG file 365 open (PNG_HANDLE, ">$filename") 366 or die("ERROR: cannot write png file $filename!\n"); 367 binmode(*PNG_HANDLE); 368 print(PNG_HANDLE $overview->png()); 369 close(PNG_HANDLE); 370 } 371 372 sub genpng_warn_handler($) 373 { 374 my ($msg) = @_; 375 376 warn("$tool_name: $msg"); 377 } 378 379 sub genpng_die_handler($) 380 { 381 my ($msg) = @_; 382 383 die("$tool_name: $msg"); 384 } 385