| OLD | NEW |
| 1 #!/usr/bin/perl -w | 1 #!/usr/bin/perl -w |
| 2 # | 2 # |
| 3 # Copyright (c) International Business Machines Corp., 2002 | 3 # Copyright (c) International Business Machines Corp., 2002,2012 |
| 4 # | 4 # |
| 5 # This program is free software; you can redistribute it and/or modify | 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 | 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 | 7 # the Free Software Foundation; either version 2 of the License, or (at |
| 8 # your option) any later version. | 8 # your option) any later version. |
| 9 # | 9 # |
| 10 # This program is distributed in the hope that it will be useful, but | 10 # This program is distributed in the hope that it will be useful, but |
| 11 # WITHOUT ANY WARRANTY; without even the implied warranty of | 11 # WITHOUT ANY WARRANTY; without even the implied warranty of |
| 12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | 12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 13 # General Public License for more details. | 13 # General Public License for more details. |
| (...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 65 # | 65 # |
| 66 | 66 |
| 67 use strict; | 67 use strict; |
| 68 use File::Basename; | 68 use File::Basename; |
| 69 use Getopt::Long; | 69 use Getopt::Long; |
| 70 use Digest::MD5 qw(md5_base64); | 70 use Digest::MD5 qw(md5_base64); |
| 71 | 71 |
| 72 | 72 |
| 73 # Global constants | 73 # Global constants |
| 74 our $title = "LCOV - code coverage report"; | 74 our $title = "LCOV - code coverage report"; |
| 75 our $lcov_version» = "LCOV version 1.7"; | 75 our $lcov_version» = 'LCOV version 1.10'; |
| 76 our $lcov_url = "http://ltp.sourceforge.net/coverage/lcov.php"; | 76 our $lcov_url = "http://ltp.sourceforge.net/coverage/lcov.php"; |
| 77 our $tool_name = basename($0); | 77 our $tool_name = basename($0); |
| 78 | 78 |
| 79 # Specify coverage rate limits (in %) for classifying file entries | 79 # Specify coverage rate limits (in %) for classifying file entries |
| 80 # HI: $hi_limit <= rate <= 100 graph color: green | 80 # HI: $hi_limit <= rate <= 100 graph color: green |
| 81 # MED: $med_limit <= rate < $hi_limit graph color: orange | 81 # MED: $med_limit <= rate < $hi_limit graph color: orange |
| 82 # LO: 0 <= rate < $med_limit graph color: red | 82 # LO: 0 <= rate < $med_limit graph color: red |
| 83 | 83 |
| 84 # For line coverage | 84 # For line coverage/all coverage types if not specified |
| 85 our $hi_limit» = 50; | 85 our $hi_limit = 90; |
| 86 our $med_limit» = 15; | 86 our $med_limit = 75; |
| 87 | 87 |
| 88 # For function coverage | 88 # For function coverage |
| 89 our $fn_hi_limit» = 90; | 89 our $fn_hi_limit; |
| 90 our $fn_med_limit» = 75; | 90 our $fn_med_limit; |
| 91 |
| 92 # For branch coverage |
| 93 our $br_hi_limit; |
| 94 our $br_med_limit; |
| 91 | 95 |
| 92 # Width of overview image | 96 # Width of overview image |
| 93 our $overview_width = 80; | 97 our $overview_width = 80; |
| 94 | 98 |
| 95 # Resolution of overview navigation: this number specifies the maximum | 99 # Resolution of overview navigation: this number specifies the maximum |
| 96 # difference in lines between the position a user selected from the overview | 100 # difference in lines between the position a user selected from the overview |
| 97 # and the position the source code window is scrolled to. | 101 # and the position the source code window is scrolled to. |
| 98 our $nav_resolution = 4; | 102 our $nav_resolution = 4; |
| 99 | 103 |
| 100 # Clicking a line in the overview image should show the source code view at | 104 # Clicking a line in the overview image should show the source code view at |
| 101 # a position a bit further up so that the requested line is not the first | 105 # a position a bit further up so that the requested line is not the first |
| 102 # line in the window. This number specifies that offset in lines. | 106 # line in the window. This number specifies that offset in lines. |
| 103 our $nav_offset = 10; | 107 our $nav_offset = 10; |
| 104 | 108 |
| 105 # Clicking on a function name should show the source code at a position a | 109 # Clicking on a function name should show the source code at a position a |
| 106 # few lines before the first line of code of that function. This number | 110 # few lines before the first line of code of that function. This number |
| 107 # specifies that offset in lines. | 111 # specifies that offset in lines. |
| 108 our $func_offset = 2; | 112 our $func_offset = 2; |
| 109 | 113 |
| 110 our $overview_title = "directory"; | 114 our $overview_title = "top level"; |
| 115 |
| 116 # Width for line coverage information in the source code view |
| 117 our $line_field_width = 12; |
| 118 |
| 119 # Width for branch coverage information in the source code view |
| 120 our $br_field_width = 16; |
| 121 |
| 122 # Internal Constants |
| 123 |
| 124 # Header types |
| 125 our $HDR_DIR» » = 0; |
| 126 our $HDR_FILE» » = 1; |
| 127 our $HDR_SOURCE»» = 2; |
| 128 our $HDR_TESTDESC» = 3; |
| 129 our $HDR_FUNC» » = 4; |
| 130 |
| 131 # Sort types |
| 132 our $SORT_FILE» » = 0; |
| 133 our $SORT_LINE» » = 1; |
| 134 our $SORT_FUNC» » = 2; |
| 135 our $SORT_BRANCH» = 3; |
| 136 |
| 137 # Fileview heading types |
| 138 our $HEAD_NO_DETAIL» = 1; |
| 139 our $HEAD_DETAIL_HIDDEN»= 2; |
| 140 our $HEAD_DETAIL_SHOWN» = 3; |
| 141 |
| 142 # Offsets for storing branch coverage data in vectors |
| 143 our $BR_BLOCK» » = 0; |
| 144 our $BR_BRANCH» » = 1; |
| 145 our $BR_TAKEN» » = 2; |
| 146 our $BR_VEC_ENTRIES» = 3; |
| 147 our $BR_VEC_WIDTH» = 32; |
| 148 |
| 149 # Additional offsets used when converting branch coverage data to HTML |
| 150 our $BR_LEN» = 3; |
| 151 our $BR_OPEN» = 4; |
| 152 our $BR_CLOSE» = 5; |
| 153 |
| 154 # Branch data combination types |
| 155 our $BR_SUB = 0; |
| 156 our $BR_ADD = 1; |
| 157 |
| 158 # Error classes which users may specify to ignore during processing |
| 159 our $ERROR_SOURCE» = 0; |
| 160 our %ERROR_ID = ( |
| 161 » "source" => $ERROR_SOURCE, |
| 162 ); |
| 111 | 163 |
| 112 # Data related prototypes | 164 # Data related prototypes |
| 113 sub print_usage(*); | 165 sub print_usage(*); |
| 114 sub gen_html(); | 166 sub gen_html(); |
| 115 sub html_create($$); | 167 sub html_create($$); |
| 116 sub process_dir($); | 168 sub process_dir($); |
| 117 sub process_file($$$); | 169 sub process_file($$$); |
| 118 sub info(@); | 170 sub info(@); |
| 119 sub read_info_file($); | 171 sub read_info_file($); |
| 120 sub get_info_entry($); | 172 sub get_info_entry($); |
| 121 sub set_info_entry($$$$$$$;$$$$); | 173 sub set_info_entry($$$$$$$$$;$$$$$$); |
| 122 sub get_prefix(@); | 174 sub get_prefix($@); |
| 123 sub shorten_prefix($); | 175 sub shorten_prefix($); |
| 124 sub get_dir_list(@); | 176 sub get_dir_list(@); |
| 125 sub get_relative_base_path($); | 177 sub get_relative_base_path($); |
| 126 sub read_testfile($); | 178 sub read_testfile($); |
| 127 sub get_date_string(); | 179 sub get_date_string(); |
| 128 sub split_filename($); | |
| 129 sub create_sub_dir($); | 180 sub create_sub_dir($); |
| 130 sub subtract_counts($$); | 181 sub subtract_counts($$); |
| 131 sub add_counts($$); | 182 sub add_counts($$); |
| 132 sub apply_baseline($$); | 183 sub apply_baseline($$); |
| 133 sub remove_unused_descriptions(); | 184 sub remove_unused_descriptions(); |
| 134 sub get_found_and_hit($); | 185 sub get_found_and_hit($); |
| 135 sub get_affecting_tests($$); | 186 sub get_affecting_tests($$$); |
| 136 sub combine_info_files($$); | 187 sub combine_info_files($$); |
| 137 sub merge_checksums($$$); | 188 sub merge_checksums($$$); |
| 138 sub combine_info_entries($$$); | 189 sub combine_info_entries($$$); |
| 139 sub apply_prefix($$); | 190 sub apply_prefix($$); |
| 140 sub system_no_output($@); | 191 sub system_no_output($@); |
| 141 sub read_config($); | 192 sub read_config($); |
| 142 sub apply_config($); | 193 sub apply_config($); |
| 143 sub get_html_prolog($); | 194 sub get_html_prolog($); |
| 144 sub get_html_epilog($); | 195 sub get_html_epilog($); |
| 196 sub write_dir_page($$$$$$$$$$$$$$$$$); |
| 197 sub classify_rate($$$$); |
| 198 sub br_taken_add($$); |
| 199 sub br_taken_sub($$); |
| 200 sub br_ivec_len($); |
| 201 sub br_ivec_get($$); |
| 202 sub br_ivec_push($$$$); |
| 203 sub combine_brcount($$$); |
| 204 sub get_br_found_and_hit($); |
| 205 sub warn_handler($); |
| 206 sub die_handler($); |
| 207 sub parse_ignore_errors(@); |
| 208 sub rate($$;$$$); |
| 145 | 209 |
| 146 | 210 |
| 147 # HTML related prototypes | 211 # HTML related prototypes |
| 148 sub escape_html($); | 212 sub escape_html($); |
| 149 sub get_bar_graph_code($$$); | 213 sub get_bar_graph_code($$$); |
| 150 | 214 |
| 151 sub write_png_files(); | 215 sub write_png_files(); |
| 152 sub write_htaccess_file(); | 216 sub write_htaccess_file(); |
| 153 sub write_css_file(); | 217 sub write_css_file(); |
| 154 sub write_description_file($$$$$); | 218 sub write_description_file($$$$$$$); |
| 155 sub write_function_rable(*$$$); | 219 sub write_function_table(*$$$$$$$$$$); |
| 156 | 220 |
| 157 sub write_html(*$); | 221 sub write_html(*$); |
| 158 sub write_html_prolog(*$$); | 222 sub write_html_prolog(*$$); |
| 159 sub write_html_epilog(*$;$); | 223 sub write_html_epilog(*$;$); |
| 160 | 224 |
| 161 sub write_header(*$$$$$$$$); | 225 sub write_header(*$$$$$$$$$$); |
| 162 sub write_header_prolog(*$); | 226 sub write_header_prolog(*$); |
| 163 sub write_header_line(*$@); | 227 sub write_header_line(*@); |
| 164 sub write_header_epilog(*$); | 228 sub write_header_epilog(*$); |
| 165 | 229 |
| 166 sub write_file_table(*$$$$$$); | 230 sub write_file_table(*$$$$$$$); |
| 167 sub write_file_table_prolog(*$$$); | 231 sub write_file_table_prolog(*$@); |
| 168 sub write_file_table_entry(*$$$$$$$); | 232 sub write_file_table_entry(*$$$@); |
| 169 sub write_file_table_detail_heading(*$$$); | 233 sub write_file_table_detail_entry(*$@); |
| 170 sub write_file_table_detail_entry(*$$$$$); | |
| 171 sub write_file_table_epilog(*); | 234 sub write_file_table_epilog(*); |
| 172 | 235 |
| 173 sub write_test_table_prolog(*$); | 236 sub write_test_table_prolog(*$); |
| 174 sub write_test_table_entry(*$$); | 237 sub write_test_table_entry(*$$); |
| 175 sub write_test_table_epilog(*); | 238 sub write_test_table_epilog(*); |
| 176 | 239 |
| 177 sub write_source($$$$$$); | 240 sub write_source($$$$$$$); |
| 178 sub write_source_prolog(*); | 241 sub write_source_prolog(*); |
| 179 sub write_source_line(*$$$$$); | 242 sub write_source_line(*$$$$$$); |
| 180 sub write_source_epilog(*); | 243 sub write_source_epilog(*); |
| 181 | 244 |
| 182 sub write_frameset(*$$$); | 245 sub write_frameset(*$$$); |
| 183 sub write_overview_line(*$$$); | 246 sub write_overview_line(*$$$); |
| 184 sub write_overview(*$$$$); | 247 sub write_overview(*$$$$); |
| 185 | 248 |
| 186 # External prototype (defined in genpng) | 249 # External prototype (defined in genpng) |
| 187 sub gen_png($$$@); | 250 sub gen_png($$$@); |
| 188 | 251 |
| 189 | 252 |
| 190 # Global variables & initialization | 253 # Global variables & initialization |
| 191 our %info_data; # Hash containing all data from .info file | 254 our %info_data; # Hash containing all data from .info file |
| 192 our $dir_prefix; # Prefix to remove from all sub directories | 255 our $dir_prefix; # Prefix to remove from all sub directories |
| 193 our %test_description; # Hash containing test descriptions if available | 256 our %test_description; # Hash containing test descriptions if available |
| 194 our $date = get_date_string(); | 257 our $date = get_date_string(); |
| 195 | 258 |
| 196 our @info_filenames; # List of .info files to use as data source | 259 our @info_filenames; # List of .info files to use as data source |
| 197 our $test_title; # Title for output as written to each page header | 260 our $test_title; # Title for output as written to each page header |
| 198 our $output_directory; # Name of directory in which to store output | 261 our $output_directory; # Name of directory in which to store output |
| 199 our $base_filename; # Optional name of file containing baseline data | 262 our $base_filename; # Optional name of file containing baseline data |
| 200 our $desc_filename; # Name of file containing test descriptions | 263 our $desc_filename; # Name of file containing test descriptions |
| 201 our $css_filename; # Optional name of external stylesheet file to use | 264 our $css_filename; # Optional name of external stylesheet file to use |
| 202 our $quiet; # If set, suppress information messages | 265 our $quiet; # If set, suppress information messages |
| 203 our $help; # Help option flag | 266 our $help; # Help option flag |
| 204 our $version; # Version option flag | 267 our $version; # Version option flag |
| 205 our $show_details; # If set, generate detailed directory view | 268 our $show_details; # If set, generate detailed directory view |
| 206 our $no_prefix; # If set, do not remove filename prefix | 269 our $no_prefix; # If set, do not remove filename prefix |
| 207 our $func_coverage = 1;»# If set, generate function coverage statistics | 270 our $func_coverage;» # If set, generate function coverage statistics |
| 208 our $no_func_coverage; # Disable func_coverage | 271 our $no_func_coverage; # Disable func_coverage |
| 272 our $br_coverage; # If set, generate branch coverage statistics |
| 273 our $no_br_coverage; # Disable br_coverage |
| 209 our $sort = 1; # If set, provide directory listings with sorted entries | 274 our $sort = 1; # If set, provide directory listings with sorted entries |
| 210 our $no_sort; # Disable sort | 275 our $no_sort; # Disable sort |
| 211 our $frames; # If set, use frames for source code view | 276 our $frames; # If set, use frames for source code view |
| 212 our $keep_descriptions; # If set, do not remove unused test case descriptions | 277 our $keep_descriptions; # If set, do not remove unused test case descriptions |
| 213 our $no_sourceview; # If set, do not create a source code view for each file | 278 our $no_sourceview; # If set, do not create a source code view for each file |
| 214 our $highlight; # If set, highlight lines covered by converted data only | 279 our $highlight; # If set, highlight lines covered by converted data only |
| 215 our $legend; # If set, include legend in output | 280 our $legend; # If set, include legend in output |
| 216 our $tab_size = 8; # Number of spaces to use in place of tab | 281 our $tab_size = 8; # Number of spaces to use in place of tab |
| 217 our $config; # Configuration file contents | 282 our $config; # Configuration file contents |
| 218 our $html_prolog_file; # Custom HTML prolog file (up to and including <body>) | 283 our $html_prolog_file; # Custom HTML prolog file (up to and including <body>) |
| 219 our $html_epilog_file; # Custom HTML epilog file (from </body> onwards) | 284 our $html_epilog_file; # Custom HTML epilog file (from </body> onwards) |
| 220 our $html_prolog; # Actual HTML prolog | 285 our $html_prolog; # Actual HTML prolog |
| 221 our $html_epilog; # Actual HTML epilog | 286 our $html_epilog; # Actual HTML epilog |
| 222 our $html_ext = "html"; # Extension for generated HTML files | 287 our $html_ext = "html"; # Extension for generated HTML files |
| 223 our $html_gzip = 0; # Compress with gzip | 288 our $html_gzip = 0; # Compress with gzip |
| 289 our $demangle_cpp = 0; # Demangle C++ function names |
| 290 our @opt_ignore_errors; # Ignore certain error classes during processing |
| 291 our @ignore; |
| 292 our $opt_config_file; # User-specified configuration file location |
| 293 our %opt_rc; |
| 294 our $charset = "UTF-8"; # Default charset for HTML pages |
| 224 our @fileview_sortlist; | 295 our @fileview_sortlist; |
| 225 our @fileview_sortname = ("", "-sort-l", "-sort-f"); | 296 our @fileview_sortname = ("", "-sort-l", "-sort-f", "-sort-b"); |
| 226 our @funcview_sortlist; | 297 our @funcview_sortlist; |
| 227 our @rate_name = ("Lo", "Med", "Hi"); | 298 our @rate_name = ("Lo", "Med", "Hi"); |
| 228 our @rate_png = ("ruby.png", "amber.png", "emerald.png"); | 299 our @rate_png = ("ruby.png", "amber.png", "emerald.png"); |
| 300 our $lcov_func_coverage = 1; |
| 301 our $lcov_branch_coverage = 0; |
| 229 | 302 |
| 230 our $cwd = `pwd`; # Current working directory | 303 our $cwd = `pwd`; # Current working directory |
| 231 chomp($cwd); | 304 chomp($cwd); |
| 232 our $tool_dir = dirname($0); # Directory where genhtml tool is installed | 305 our $tool_dir = dirname($0); # Directory where genhtml tool is installed |
| 233 | 306 |
| 234 | 307 |
| 235 # | 308 # |
| 236 # Code entry point | 309 # Code entry point |
| 237 # | 310 # |
| 238 | 311 |
| 239 $SIG{__WARN__} = \&warn_handler; | 312 $SIG{__WARN__} = \&warn_handler; |
| 240 $SIG{__DIE__} = \&die_handler; | 313 $SIG{__DIE__} = \&die_handler; |
| 241 | 314 |
| 315 # Prettify version string |
| 316 $lcov_version =~ s/\$\s*Revision\s*:?\s*(\S+)\s*\$/$1/; |
| 317 |
| 242 # Add current working directory if $tool_dir is not already an absolute path | 318 # Add current working directory if $tool_dir is not already an absolute path |
| 243 if (! ($tool_dir =~ /^\/(.*)$/)) | 319 if (! ($tool_dir =~ /^\/(.*)$/)) |
| 244 { | 320 { |
| 245 $tool_dir = "$cwd/$tool_dir"; | 321 $tool_dir = "$cwd/$tool_dir"; |
| 246 } | 322 } |
| 247 | 323 |
| 324 # Check command line for a configuration file name |
| 325 Getopt::Long::Configure("pass_through", "no_auto_abbrev"); |
| 326 GetOptions("config-file=s" => \$opt_config_file, |
| 327 "rc=s%" => \%opt_rc); |
| 328 Getopt::Long::Configure("default"); |
| 329 |
| 248 # Read configuration file if available | 330 # Read configuration file if available |
| 249 if (-r $ENV{"HOME"}."/.lcovrc") | 331 if (defined($opt_config_file)) { |
| 332 » $config = read_config($opt_config_file); |
| 333 } elsif (defined($ENV{"HOME"}) && (-r $ENV{"HOME"}."/.lcovrc")) |
| 250 { | 334 { |
| 251 $config = read_config($ENV{"HOME"}."/.lcovrc"); | 335 $config = read_config($ENV{"HOME"}."/.lcovrc"); |
| 252 } | 336 } |
| 253 elsif (-r "/etc/lcovrc") | 337 elsif (-r "/etc/lcovrc") |
| 254 { | 338 { |
| 255 $config = read_config("/etc/lcovrc"); | 339 $config = read_config("/etc/lcovrc"); |
| 256 } | 340 } |
| 257 | 341 |
| 258 if ($config) | 342 if ($config || %opt_rc) |
| 259 { | 343 { |
| 260 » # Copy configuration file values to variables | 344 » # Copy configuration file and --rc values to variables |
| 261 apply_config({ | 345 apply_config({ |
| 262 "genhtml_css_file" => \$css_filename, | 346 "genhtml_css_file" => \$css_filename, |
| 263 "genhtml_hi_limit" => \$hi_limit, | 347 "genhtml_hi_limit" => \$hi_limit, |
| 264 "genhtml_med_limit" => \$med_limit, | 348 "genhtml_med_limit" => \$med_limit, |
| 349 "genhtml_line_field_width" => \$line_field_width, |
| 265 "genhtml_overview_width" => \$overview_width, | 350 "genhtml_overview_width" => \$overview_width, |
| 266 "genhtml_nav_resolution" => \$nav_resolution, | 351 "genhtml_nav_resolution" => \$nav_resolution, |
| 267 "genhtml_nav_offset" => \$nav_offset, | 352 "genhtml_nav_offset" => \$nav_offset, |
| 268 "genhtml_keep_descriptions" => \$keep_descriptions, | 353 "genhtml_keep_descriptions" => \$keep_descriptions, |
| 269 "genhtml_no_prefix" => \$no_prefix, | 354 "genhtml_no_prefix" => \$no_prefix, |
| 270 "genhtml_no_source" => \$no_sourceview, | 355 "genhtml_no_source" => \$no_sourceview, |
| 271 "genhtml_num_spaces" => \$tab_size, | 356 "genhtml_num_spaces" => \$tab_size, |
| 272 "genhtml_highlight" => \$highlight, | 357 "genhtml_highlight" => \$highlight, |
| 273 "genhtml_legend" => \$legend, | 358 "genhtml_legend" => \$legend, |
| 274 "genhtml_html_prolog" => \$html_prolog_file, | 359 "genhtml_html_prolog" => \$html_prolog_file, |
| 275 "genhtml_html_epilog" => \$html_epilog_file, | 360 "genhtml_html_epilog" => \$html_epilog_file, |
| 276 "genhtml_html_extension" => \$html_ext, | 361 "genhtml_html_extension" => \$html_ext, |
| 277 "genhtml_html_gzip" => \$html_gzip, | 362 "genhtml_html_gzip" => \$html_gzip, |
| 278 "genhtml_function_hi_limit" => \$fn_hi_limit, | 363 "genhtml_function_hi_limit" => \$fn_hi_limit, |
| 279 "genhtml_function_med_limit" => \$fn_med_limit, | 364 "genhtml_function_med_limit" => \$fn_med_limit, |
| 280 "genhtml_function_coverage" => \$func_coverage, | 365 "genhtml_function_coverage" => \$func_coverage, |
| 366 "genhtml_branch_hi_limit" => \$br_hi_limit, |
| 367 "genhtml_branch_med_limit" => \$br_med_limit, |
| 368 "genhtml_branch_coverage" => \$br_coverage, |
| 369 "genhtml_branch_field_width" => \$br_field_width, |
| 281 "genhtml_sort" => \$sort, | 370 "genhtml_sort" => \$sort, |
| 371 "genhtml_charset" => \$charset, |
| 372 "lcov_function_coverage" => \$lcov_func_coverage, |
| 373 "lcov_branch_coverage" => \$lcov_branch_coverage, |
| 282 }); | 374 }); |
| 283 } | 375 } |
| 284 | 376 |
| 377 # Copy related values if not specified |
| 378 $fn_hi_limit = $hi_limit if (!defined($fn_hi_limit)); |
| 379 $fn_med_limit = $med_limit if (!defined($fn_med_limit)); |
| 380 $br_hi_limit = $hi_limit if (!defined($br_hi_limit)); |
| 381 $br_med_limit = $med_limit if (!defined($br_med_limit)); |
| 382 $func_coverage = $lcov_func_coverage if (!defined($func_coverage)); |
| 383 $br_coverage = $lcov_branch_coverage if (!defined($br_coverage)); |
| 384 |
| 285 # Parse command line options | 385 # Parse command line options |
| 286 if (!GetOptions("output-directory=s"» => \$output_directory, | 386 if (!GetOptions("output-directory|o=s"» => \$output_directory, |
| 287 » » "title=s"» » => \$test_title, | 387 » » "title|t=s"» » => \$test_title, |
| 288 » » "description-file=s"» => \$desc_filename, | 388 » » "description-file|d=s"» => \$desc_filename, |
| 289 » » "keep-descriptions"» => \$keep_descriptions, | 389 » » "keep-descriptions|k"» => \$keep_descriptions, |
| 290 » » "css-file=s"» » => \$css_filename, | 390 » » "css-file|c=s"» » => \$css_filename, |
| 291 » » "baseline-file=s"» => \$base_filename, | 391 » » "baseline-file|b=s"» => \$base_filename, |
| 292 » » "prefix=s"» » => \$dir_prefix, | 392 » » "prefix|p=s"» » => \$dir_prefix, |
| 293 "num-spaces=i" => \$tab_size, | 393 "num-spaces=i" => \$tab_size, |
| 294 "no-prefix" => \$no_prefix, | 394 "no-prefix" => \$no_prefix, |
| 295 "no-sourceview" => \$no_sourceview, | 395 "no-sourceview" => \$no_sourceview, |
| 296 » » "show-details"» » => \$show_details, | 396 » » "show-details|s"» => \$show_details, |
| 297 » » "frames"» » => \$frames, | 397 » » "frames|f"» » => \$frames, |
| 298 "highlight" => \$highlight, | 398 "highlight" => \$highlight, |
| 299 "legend" => \$legend, | 399 "legend" => \$legend, |
| 300 » » "quiet"»» » => \$quiet, | 400 » » "quiet|q"» » => \$quiet, |
| 301 "help|h|?" => \$help, | 401 "help|h|?" => \$help, |
| 302 » » "version"» » => \$version, | 402 » » "version|v"» » => \$version, |
| 303 "html-prolog=s" => \$html_prolog_file, | 403 "html-prolog=s" => \$html_prolog_file, |
| 304 "html-epilog=s" => \$html_epilog_file, | 404 "html-epilog=s" => \$html_epilog_file, |
| 305 "html-extension=s" => \$html_ext, | 405 "html-extension=s" => \$html_ext, |
| 306 "html-gzip" => \$html_gzip, | 406 "html-gzip" => \$html_gzip, |
| 307 "function-coverage" => \$func_coverage, | 407 "function-coverage" => \$func_coverage, |
| 308 "no-function-coverage" => \$no_func_coverage, | 408 "no-function-coverage" => \$no_func_coverage, |
| 409 "branch-coverage" => \$br_coverage, |
| 410 "no-branch-coverage" => \$no_br_coverage, |
| 309 "sort" => \$sort, | 411 "sort" => \$sort, |
| 310 "no-sort" => \$no_sort, | 412 "no-sort" => \$no_sort, |
| 413 "demangle-cpp" => \$demangle_cpp, |
| 414 "ignore-errors=s" => \@opt_ignore_errors, |
| 415 "config-file=s" => \$opt_config_file, |
| 416 "rc=s%" => \%opt_rc, |
| 311 )) | 417 )) |
| 312 { | 418 { |
| 313 print(STDERR "Use $tool_name --help to get usage information\n"); | 419 print(STDERR "Use $tool_name --help to get usage information\n"); |
| 314 exit(1); | 420 exit(1); |
| 315 } else { | 421 } else { |
| 316 # Merge options | 422 # Merge options |
| 317 if ($no_func_coverage) { | 423 if ($no_func_coverage) { |
| 318 $func_coverage = 0; | 424 $func_coverage = 0; |
| 319 } | 425 } |
| 426 if ($no_br_coverage) { |
| 427 $br_coverage = 0; |
| 428 } |
| 320 | 429 |
| 321 # Merge sort options | 430 # Merge sort options |
| 322 if ($no_sort) { | 431 if ($no_sort) { |
| 323 $sort = 0; | 432 $sort = 0; |
| 324 } | 433 } |
| 325 } | 434 } |
| 326 | 435 |
| 327 @info_filenames = @ARGV; | 436 @info_filenames = @ARGV; |
| 328 | 437 |
| 329 # Check for help option | 438 # Check for help option |
| 330 if ($help) | 439 if ($help) |
| 331 { | 440 { |
| 332 print_usage(*STDOUT); | 441 print_usage(*STDOUT); |
| 333 exit(0); | 442 exit(0); |
| 334 } | 443 } |
| 335 | 444 |
| 336 # Check for version option | 445 # Check for version option |
| 337 if ($version) | 446 if ($version) |
| 338 { | 447 { |
| 339 print("$tool_name: $lcov_version\n"); | 448 print("$tool_name: $lcov_version\n"); |
| 340 exit(0); | 449 exit(0); |
| 341 } | 450 } |
| 342 | 451 |
| 452 # Determine which errors the user wants us to ignore |
| 453 parse_ignore_errors(@opt_ignore_errors); |
| 454 |
| 343 # Check for info filename | 455 # Check for info filename |
| 344 if (!@info_filenames) | 456 if (!@info_filenames) |
| 345 { | 457 { |
| 346 die("No filename specified\n". | 458 die("No filename specified\n". |
| 347 "Use $tool_name --help to get usage information\n"); | 459 "Use $tool_name --help to get usage information\n"); |
| 348 } | 460 } |
| 349 | 461 |
| 350 # Generate a title if none is specified | 462 # Generate a title if none is specified |
| 351 if (!$test_title) | 463 if (!$test_title) |
| 352 { | 464 { |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 393 } | 505 } |
| 394 | 506 |
| 395 # Issue a warning if --no-prefix is enabled together with --prefix | 507 # Issue a warning if --no-prefix is enabled together with --prefix |
| 396 if ($no_prefix && defined($dir_prefix)) | 508 if ($no_prefix && defined($dir_prefix)) |
| 397 { | 509 { |
| 398 warn("WARNING: option --prefix disabled because --no-prefix was ". | 510 warn("WARNING: option --prefix disabled because --no-prefix was ". |
| 399 "specified!\n"); | 511 "specified!\n"); |
| 400 $dir_prefix = undef; | 512 $dir_prefix = undef; |
| 401 } | 513 } |
| 402 | 514 |
| 515 @fileview_sortlist = ($SORT_FILE); |
| 516 @funcview_sortlist = ($SORT_FILE); |
| 517 |
| 403 if ($sort) { | 518 if ($sort) { |
| 404 » @funcview_sortlist = (0, 1); | 519 » push(@fileview_sortlist, $SORT_LINE); |
| 405 » if ($func_coverage) { | 520 » push(@fileview_sortlist, $SORT_FUNC) if ($func_coverage); |
| 406 » » @fileview_sortlist = (0, 1, 2); | 521 » push(@fileview_sortlist, $SORT_BRANCH) if ($br_coverage); |
| 407 » } else { | 522 » push(@funcview_sortlist, $SORT_LINE); |
| 408 » » @fileview_sortlist = (0, 1); | |
| 409 » } | |
| 410 } else { | |
| 411 » @fileview_sortlist = (0); | |
| 412 » @funcview_sortlist = (0); | |
| 413 } | 523 } |
| 414 | 524 |
| 415 if ($frames) | 525 if ($frames) |
| 416 { | 526 { |
| 417 # Include genpng code needed for overview image generation | 527 # Include genpng code needed for overview image generation |
| 418 do("$tool_dir/genpng"); | 528 do("$tool_dir/genpng"); |
| 419 } | 529 } |
| 420 | 530 |
| 531 # Ensure that the c++filt tool is available when using --demangle-cpp |
| 532 if ($demangle_cpp) |
| 533 { |
| 534 if (system_no_output(3, "c++filt", "--version")) { |
| 535 die("ERROR: could not find c++filt tool needed for ". |
| 536 "--demangle-cpp\n"); |
| 537 } |
| 538 } |
| 539 |
| 421 # Make sure output_directory exists, create it if necessary | 540 # Make sure output_directory exists, create it if necessary |
| 422 if ($output_directory) | 541 if ($output_directory) |
| 423 { | 542 { |
| 424 stat($output_directory); | 543 stat($output_directory); |
| 425 | 544 |
| 426 if (! -e _) | 545 if (! -e _) |
| 427 { | 546 { |
| 428 » » system("mkdir", "-p", $output_directory) | 547 » » create_sub_dir($output_directory); |
| 429 » » » and die("ERROR: cannot create directory $_!\n"); | |
| 430 } | 548 } |
| 431 } | 549 } |
| 432 | 550 |
| 433 # Do something | 551 # Do something |
| 434 gen_html(); | 552 gen_html(); |
| 435 | 553 |
| 436 exit(0); | 554 exit(0); |
| 437 | 555 |
| 438 | 556 |
| 439 | 557 |
| (...skipping 10 matching lines...) Expand all Loading... |
| 450 print(HANDLE <<END_OF_USAGE); | 568 print(HANDLE <<END_OF_USAGE); |
| 451 Usage: $tool_name [OPTIONS] INFOFILE(S) | 569 Usage: $tool_name [OPTIONS] INFOFILE(S) |
| 452 | 570 |
| 453 Create HTML output for coverage data found in INFOFILE. Note that INFOFILE | 571 Create HTML output for coverage data found in INFOFILE. Note that INFOFILE |
| 454 may also be a list of filenames. | 572 may also be a list of filenames. |
| 455 | 573 |
| 456 Misc: | 574 Misc: |
| 457 -h, --help Print this help, then exit | 575 -h, --help Print this help, then exit |
| 458 -v, --version Print version number, then exit | 576 -v, --version Print version number, then exit |
| 459 -q, --quiet Do not print progress messages | 577 -q, --quiet Do not print progress messages |
| 578 --config-file FILENAME Specify configuration file location |
| 579 --rc SETTING=VALUE Override configuration file setting |
| 580 --ignore-errors ERRORS Continue after ERRORS (source) |
| 460 | 581 |
| 461 Operation: | 582 Operation: |
| 462 -o, --output-directory OUTDIR Write HTML output to OUTDIR | 583 -o, --output-directory OUTDIR Write HTML output to OUTDIR |
| 463 -s, --show-details Generate detailed directory view | 584 -s, --show-details Generate detailed directory view |
| 464 -d, --description-file DESCFILE Read test case descriptions from DESCFILE | 585 -d, --description-file DESCFILE Read test case descriptions from DESCFILE |
| 465 -k, --keep-descriptions Do not remove unused test descriptions | 586 -k, --keep-descriptions Do not remove unused test descriptions |
| 466 -b, --baseline-file BASEFILE Use BASEFILE as baseline file | 587 -b, --baseline-file BASEFILE Use BASEFILE as baseline file |
| 467 -p, --prefix PREFIX Remove PREFIX from all directory names | 588 -p, --prefix PREFIX Remove PREFIX from all directory names |
| 468 --no-prefix Do not remove prefix from directory names | 589 --no-prefix Do not remove prefix from directory names |
| 469 --(no-)function-coverage Enable (disable) function coverage display | 590 --(no-)function-coverage Enable (disable) function coverage display |
| 591 --(no-)branch-coverage Enable (disable) branch coverage display |
| 470 | 592 |
| 471 HTML output: | 593 HTML output: |
| 472 -f, --frames Use HTML frames for source code view | 594 -f, --frames Use HTML frames for source code view |
| 473 -t, --title TITLE Display TITLE in header of all pages | 595 -t, --title TITLE Display TITLE in header of all pages |
| 474 -c, --css-file CSSFILE Use external style sheet file CSSFILE | 596 -c, --css-file CSSFILE Use external style sheet file CSSFILE |
| 475 --no-source Do not create source code view | 597 --no-source Do not create source code view |
| 476 --num-spaces NUM Replace tabs with NUM spaces in source view | 598 --num-spaces NUM Replace tabs with NUM spaces in source view |
| 477 --highlight Highlight lines with converted-only data | 599 --highlight Highlight lines with converted-only data |
| 478 --legend Include color legend in HTML output | 600 --legend Include color legend in HTML output |
| 479 --html-prolog FILE Use FILE as HTML prolog for generated pages | 601 --html-prolog FILE Use FILE as HTML prolog for generated pages |
| 480 --html-epilog FILE Use FILE as HTML epilog for generated pages | 602 --html-epilog FILE Use FILE as HTML epilog for generated pages |
| 481 --html-extension EXT Use EXT as filename extension for pages | 603 --html-extension EXT Use EXT as filename extension for pages |
| 482 --html-gzip Use gzip to compress HTML | 604 --html-gzip Use gzip to compress HTML |
| 483 --(no-)sort Enable (disable) sorted coverage views | 605 --(no-)sort Enable (disable) sorted coverage views |
| 606 --demangle-cpp Demangle C++ function names |
| 484 | 607 |
| 485 For more information see: $lcov_url | 608 For more information see: $lcov_url |
| 486 END_OF_USAGE | 609 END_OF_USAGE |
| 487 ; | 610 ; |
| 488 } | 611 } |
| 489 | 612 |
| 490 | 613 |
| 491 # | 614 # |
| 492 # get_rate(found, hit) | 615 # get_rate(found, hit) |
| 493 # | 616 # |
| 494 # Return a relative value for the specified found&hit values | 617 # Return a relative value for the specified found&hit values |
| 495 # which is used for sorting the corresponding entries in a | 618 # which is used for sorting the corresponding entries in a |
| 496 # file list. | 619 # file list. |
| 497 # | 620 # |
| 498 | 621 |
| 499 sub get_rate($$) | 622 sub get_rate($$) |
| 500 { | 623 { |
| 501 my ($found, $hit) = @_; | 624 my ($found, $hit) = @_; |
| 502 | 625 |
| 503 if ($found == 0) { | 626 if ($found == 0) { |
| 504 return 10000; | 627 return 10000; |
| 505 } | 628 } |
| 506 return int($hit * 1000 / $found) * 10 + 2 - (1 / $found); | 629 return int($hit * 1000 / $found) * 10 + 2 - (1 / $found); |
| 507 } | 630 } |
| 508 | 631 |
| 509 | 632 |
| 510 # | 633 # |
| 634 # get_overall_line(found, hit, name_singular, name_plural) |
| 635 # |
| 636 # Return a string containing overall information for the specified |
| 637 # found/hit data. |
| 638 # |
| 639 |
| 640 sub get_overall_line($$$$) |
| 641 { |
| 642 my ($found, $hit, $name_sn, $name_pl) = @_; |
| 643 my $name; |
| 644 |
| 645 return "no data found" if (!defined($found) || $found == 0); |
| 646 $name = ($found == 1) ? $name_sn : $name_pl; |
| 647 return rate($hit, $found, "% ($hit of $found $name)"); |
| 648 } |
| 649 |
| 650 |
| 651 # |
| 652 # print_overall_rate(ln_do, ln_found, ln_hit, fn_do, fn_found, fn_hit, br_do |
| 653 # br_found, br_hit) |
| 654 # |
| 655 # Print overall coverage rates for the specified coverage types. |
| 656 # |
| 657 |
| 658 sub print_overall_rate($$$$$$$$$) |
| 659 { |
| 660 my ($ln_do, $ln_found, $ln_hit, $fn_do, $fn_found, $fn_hit, |
| 661 $br_do, $br_found, $br_hit) = @_; |
| 662 |
| 663 info("Overall coverage rate:\n"); |
| 664 info(" lines......: %s\n", |
| 665 get_overall_line($ln_found, $ln_hit, "line", "lines")) |
| 666 if ($ln_do); |
| 667 info(" functions..: %s\n", |
| 668 get_overall_line($fn_found, $fn_hit, "function", "functions")) |
| 669 if ($fn_do); |
| 670 info(" branches...: %s\n", |
| 671 get_overall_line($br_found, $br_hit, "branch", "branches")) |
| 672 if ($br_do); |
| 673 } |
| 674 |
| 675 |
| 676 # |
| 511 # gen_html() | 677 # gen_html() |
| 512 # | 678 # |
| 513 # Generate a set of HTML pages from contents of .info file INFO_FILENAME. | 679 # Generate a set of HTML pages from contents of .info file INFO_FILENAME. |
| 514 # Files will be written to the current directory. If provided, test case | 680 # Files will be written to the current directory. If provided, test case |
| 515 # descriptions will be read from .tests file TEST_FILENAME and included | 681 # descriptions will be read from .tests file TEST_FILENAME and included |
| 516 # in ouput. | 682 # in ouput. |
| 517 # | 683 # |
| 518 # Die on error. | 684 # Die on error. |
| 519 # | 685 # |
| 520 | 686 |
| 521 sub gen_html() | 687 sub gen_html() |
| 522 { | 688 { |
| 523 local *HTML_HANDLE; | 689 local *HTML_HANDLE; |
| 524 my %overview; | 690 my %overview; |
| 525 my %base_data; | 691 my %base_data; |
| 526 my $lines_found; | 692 my $lines_found; |
| 527 my $lines_hit; | 693 my $lines_hit; |
| 528 my $fn_found; | 694 my $fn_found; |
| 529 my $fn_hit; | 695 my $fn_hit; |
| 696 my $br_found; |
| 697 my $br_hit; |
| 530 my $overall_found = 0; | 698 my $overall_found = 0; |
| 531 my $overall_hit = 0; | 699 my $overall_hit = 0; |
| 532 my $total_fn_found = 0; | 700 my $total_fn_found = 0; |
| 533 my $total_fn_hit = 0; | 701 my $total_fn_hit = 0; |
| 702 my $total_br_found = 0; |
| 703 my $total_br_hit = 0; |
| 534 my $dir_name; | 704 my $dir_name; |
| 535 my $link_name; | 705 my $link_name; |
| 536 my @dir_list; | 706 my @dir_list; |
| 537 my %new_info; | 707 my %new_info; |
| 538 | 708 |
| 539 # Read in all specified .info files | 709 # Read in all specified .info files |
| 540 foreach (@info_filenames) | 710 foreach (@info_filenames) |
| 541 { | 711 { |
| 542 %new_info = %{read_info_file($_)}; | 712 %new_info = %{read_info_file($_)}; |
| 543 | 713 |
| (...skipping 19 matching lines...) Expand all Loading... |
| 563 @dir_list = get_dir_list(keys(%info_data)); | 733 @dir_list = get_dir_list(keys(%info_data)); |
| 564 | 734 |
| 565 if ($no_prefix) | 735 if ($no_prefix) |
| 566 { | 736 { |
| 567 # User requested that we leave filenames alone | 737 # User requested that we leave filenames alone |
| 568 info("User asked not to remove filename prefix\n"); | 738 info("User asked not to remove filename prefix\n"); |
| 569 } | 739 } |
| 570 elsif (!defined($dir_prefix)) | 740 elsif (!defined($dir_prefix)) |
| 571 { | 741 { |
| 572 # Get prefix common to most directories in list | 742 # Get prefix common to most directories in list |
| 573 » » $dir_prefix = get_prefix(@dir_list); | 743 » » $dir_prefix = get_prefix(1, keys(%info_data)); |
| 574 | 744 |
| 575 if ($dir_prefix) | 745 if ($dir_prefix) |
| 576 { | 746 { |
| 577 info("Found common filename prefix \"$dir_prefix\"\n"); | 747 info("Found common filename prefix \"$dir_prefix\"\n"); |
| 578 } | 748 } |
| 579 else | 749 else |
| 580 { | 750 { |
| 581 info("No common filename prefix found!\n"); | 751 info("No common filename prefix found!\n"); |
| 582 $no_prefix=1; | 752 $no_prefix=1; |
| 583 } | 753 } |
| (...skipping 34 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 618 { | 788 { |
| 619 info("Writing .htaccess file.\n"); | 789 info("Writing .htaccess file.\n"); |
| 620 write_htaccess_file(); | 790 write_htaccess_file(); |
| 621 } | 791 } |
| 622 | 792 |
| 623 info("Generating output.\n"); | 793 info("Generating output.\n"); |
| 624 | 794 |
| 625 # Process each subdirectory and collect overview information | 795 # Process each subdirectory and collect overview information |
| 626 foreach $dir_name (@dir_list) | 796 foreach $dir_name (@dir_list) |
| 627 { | 797 { |
| 628 » » ($lines_found, $lines_hit, $fn_found, $fn_hit) | 798 » » ($lines_found, $lines_hit, $fn_found, $fn_hit, |
| 799 » » $br_found, $br_hit) |
| 629 = process_dir($dir_name); | 800 = process_dir($dir_name); |
| 630 | 801 |
| 802 # Handle files in root directory gracefully |
| 803 $dir_name = "root" if ($dir_name eq ""); |
| 804 |
| 631 # Remove prefix if applicable | 805 # Remove prefix if applicable |
| 632 if (!$no_prefix && $dir_prefix) | 806 if (!$no_prefix && $dir_prefix) |
| 633 { | 807 { |
| 634 # Match directory names beginning with $dir_prefix | 808 # Match directory names beginning with $dir_prefix |
| 635 $dir_name = apply_prefix($dir_name, $dir_prefix); | 809 $dir_name = apply_prefix($dir_name, $dir_prefix); |
| 636 } | 810 } |
| 637 | 811 |
| 638 # Generate name for directory overview HTML page | 812 # Generate name for directory overview HTML page |
| 639 if ($dir_name =~ /^\/(.*)$/) | 813 if ($dir_name =~ /^\/(.*)$/) |
| 640 { | 814 { |
| 641 $link_name = substr($dir_name, 1)."/index.$html_ext"; | 815 $link_name = substr($dir_name, 1)."/index.$html_ext"; |
| 642 } | 816 } |
| 643 else | 817 else |
| 644 { | 818 { |
| 645 $link_name = $dir_name."/index.$html_ext"; | 819 $link_name = $dir_name."/index.$html_ext"; |
| 646 } | 820 } |
| 647 | 821 |
| 648 $overview{$dir_name} = [$lines_found, $lines_hit, $fn_found, | 822 $overview{$dir_name} = [$lines_found, $lines_hit, $fn_found, |
| 649 » » » » » $fn_hit, $link_name, | 823 » » » » » $fn_hit, $br_found, $br_hit, $link_name, |
| 650 get_rate($lines_found, $lines_hit), | 824 get_rate($lines_found, $lines_hit), |
| 651 » » » » » get_rate($fn_found, $fn_hit)]; | 825 » » » » » get_rate($fn_found, $fn_hit), |
| 826 » » » » » get_rate($br_found, $br_hit)]; |
| 652 $overall_found += $lines_found; | 827 $overall_found += $lines_found; |
| 653 $overall_hit += $lines_hit; | 828 $overall_hit += $lines_hit; |
| 654 $total_fn_found += $fn_found; | 829 $total_fn_found += $fn_found; |
| 655 $total_fn_hit += $fn_hit; | 830 $total_fn_hit += $fn_hit; |
| 831 $total_br_found += $br_found; |
| 832 $total_br_hit += $br_hit; |
| 656 } | 833 } |
| 657 | 834 |
| 658 # Generate overview page | 835 # Generate overview page |
| 659 info("Writing directory view page.\n"); | 836 info("Writing directory view page.\n"); |
| 660 | 837 |
| 661 # Create sorted pages | 838 # Create sorted pages |
| 662 foreach (@fileview_sortlist) { | 839 foreach (@fileview_sortlist) { |
| 663 write_dir_page($fileview_sortname[$_], ".", "", $test_title, | 840 write_dir_page($fileview_sortname[$_], ".", "", $test_title, |
| 664 undef, $overall_found, $overall_hit, | 841 undef, $overall_found, $overall_hit, |
| 665 » » » $total_fn_found, $total_fn_hit, \%overview, | 842 » » » $total_fn_found, $total_fn_hit, $total_br_found, |
| 666 » » » {}, {}, 0, $_); | 843 » » » $total_br_hit, \%overview, {}, {}, {}, 0, $_); |
| 667 } | 844 } |
| 668 | 845 |
| 669 # Check if there are any test case descriptions to write out | 846 # Check if there are any test case descriptions to write out |
| 670 if (%test_description) | 847 if (%test_description) |
| 671 { | 848 { |
| 672 info("Writing test case description file.\n"); | 849 info("Writing test case description file.\n"); |
| 673 write_description_file( \%test_description, | 850 write_description_file( \%test_description, |
| 674 $overall_found, $overall_hit, | 851 $overall_found, $overall_hit, |
| 675 » » » » » $total_fn_found, $total_fn_hit); | 852 » » » » » $total_fn_found, $total_fn_hit, |
| 853 » » » » » $total_br_found, $total_br_hit); |
| 676 } | 854 } |
| 677 | 855 |
| 856 print_overall_rate(1, $overall_found, $overall_hit, |
| 857 $func_coverage, $total_fn_found, $total_fn_hit, |
| 858 $br_coverage, $total_br_found, $total_br_hit); |
| 859 |
| 678 chdir($cwd); | 860 chdir($cwd); |
| 679 | |
| 680 info("Overall coverage rate:\n"); | |
| 681 | |
| 682 if ($overall_found == 0) | |
| 683 { | |
| 684 info(" lines......: no data found\n"); | |
| 685 return; | |
| 686 } | |
| 687 info(" lines......: %.1f%% (%d of %d lines)\n", | |
| 688 $overall_hit * 100 / $overall_found, $overall_hit, | |
| 689 $overall_found,); | |
| 690 | |
| 691 if ($func_coverage) | |
| 692 { | |
| 693 if ($total_fn_found == 0) | |
| 694 { | |
| 695 info(" functions..: no data found\n"); | |
| 696 } | |
| 697 else | |
| 698 { | |
| 699 info(" functions..: %.1f%% (%d of %d functions)\n", | |
| 700 $total_fn_hit * 100 / $total_fn_found, | |
| 701 $total_fn_hit, $total_fn_found); | |
| 702 | |
| 703 } | |
| 704 } | |
| 705 | |
| 706 } | 861 } |
| 707 | 862 |
| 708 # | 863 # |
| 709 # html_create(handle, filename) | 864 # html_create(handle, filename) |
| 710 # | 865 # |
| 711 | 866 |
| 712 sub html_create($$) | 867 sub html_create($$) |
| 713 { | 868 { |
| 714 my $handle = $_[0]; | 869 my $handle = $_[0]; |
| 715 my $filename = $_[1]; | 870 my $filename = $_[1]; |
| 716 | 871 |
| 717 if ($html_gzip) | 872 if ($html_gzip) |
| 718 { | 873 { |
| 719 » » open($handle, "|gzip -c >$filename") | 874 » » open($handle, "|-", "gzip -c >'$filename'") |
| 720 or die("ERROR: cannot open $filename for writing ". | 875 or die("ERROR: cannot open $filename for writing ". |
| 721 "(gzip)!\n"); | 876 "(gzip)!\n"); |
| 722 } | 877 } |
| 723 else | 878 else |
| 724 { | 879 { |
| 725 » » open($handle, ">$filename") | 880 » » open($handle, ">", $filename) |
| 726 or die("ERROR: cannot open $filename for writing!\n"); | 881 or die("ERROR: cannot open $filename for writing!\n"); |
| 727 } | 882 } |
| 728 } | 883 } |
| 729 | 884 |
| 730 sub write_dir_page($$$$$$$$$$$$$$) | 885 sub write_dir_page($$$$$$$$$$$$$$$$$) |
| 731 { | 886 { |
| 732 my ($name, $rel_dir, $base_dir, $title, $trunc_dir, $overall_found, | 887 my ($name, $rel_dir, $base_dir, $title, $trunc_dir, $overall_found, |
| 733 » $overall_hit, $total_fn_found, $total_fn_hit, $overview, | 888 » $overall_hit, $total_fn_found, $total_fn_hit, $total_br_found, |
| 734 » $testhash, $testfnchash, $view_type, $sort_type) = @_; | 889 » $total_br_hit, $overview, $testhash, $testfnchash, $testbrhash, |
| 890 » $view_type, $sort_type) = @_; |
| 735 | 891 |
| 736 # Generate directory overview page including details | 892 # Generate directory overview page including details |
| 737 html_create(*HTML_HANDLE, "$rel_dir/index$name.$html_ext"); | 893 html_create(*HTML_HANDLE, "$rel_dir/index$name.$html_ext"); |
| 738 if (!defined($trunc_dir)) { | 894 if (!defined($trunc_dir)) { |
| 739 $trunc_dir = ""; | 895 $trunc_dir = ""; |
| 740 } | 896 } |
| 897 $title .= " - " if ($trunc_dir ne ""); |
| 741 write_html_prolog(*HTML_HANDLE, $base_dir, "LCOV - $title$trunc_dir"); | 898 write_html_prolog(*HTML_HANDLE, $base_dir, "LCOV - $title$trunc_dir"); |
| 742 write_header(*HTML_HANDLE, $view_type, $trunc_dir, $rel_dir, | 899 write_header(*HTML_HANDLE, $view_type, $trunc_dir, $rel_dir, |
| 743 $overall_found, $overall_hit, $total_fn_found, | 900 $overall_found, $overall_hit, $total_fn_found, |
| 744 » » $total_fn_hit, $sort_type); | 901 » » $total_fn_hit, $total_br_found, $total_br_hit, $sort_type); |
| 745 write_file_table(*HTML_HANDLE, $base_dir, $overview, $testhash, | 902 write_file_table(*HTML_HANDLE, $base_dir, $overview, $testhash, |
| 746 » » » $testfnchash, $view_type, $sort_type); | 903 » » » $testfnchash, $testbrhash, $view_type, $sort_type); |
| 747 write_html_epilog(*HTML_HANDLE, $base_dir); | 904 write_html_epilog(*HTML_HANDLE, $base_dir); |
| 748 close(*HTML_HANDLE); | 905 close(*HTML_HANDLE); |
| 749 } | 906 } |
| 750 | 907 |
| 751 | 908 |
| 752 # | 909 # |
| 753 # process_dir(dir_name) | 910 # process_dir(dir_name) |
| 754 # | 911 # |
| 755 | 912 |
| 756 sub process_dir($) | 913 sub process_dir($) |
| 757 { | 914 { |
| 758 my $abs_dir = $_[0]; | 915 my $abs_dir = $_[0]; |
| 759 my $trunc_dir; | 916 my $trunc_dir; |
| 760 my $rel_dir = $abs_dir; | 917 my $rel_dir = $abs_dir; |
| 761 my $base_dir; | 918 my $base_dir; |
| 762 my $filename; | 919 my $filename; |
| 763 my %overview; | 920 my %overview; |
| 764 my $lines_found; | 921 my $lines_found; |
| 765 my $lines_hit; | 922 my $lines_hit; |
| 766 my $fn_found; | 923 my $fn_found; |
| 767 my $fn_hit; | 924 my $fn_hit; |
| 925 my $br_found; |
| 926 my $br_hit; |
| 768 my $overall_found=0; | 927 my $overall_found=0; |
| 769 my $overall_hit=0; | 928 my $overall_hit=0; |
| 770 my $total_fn_found=0; | 929 my $total_fn_found=0; |
| 771 my $total_fn_hit=0; | 930 my $total_fn_hit=0; |
| 931 my $total_br_found = 0; |
| 932 my $total_br_hit = 0; |
| 772 my $base_name; | 933 my $base_name; |
| 773 my $extension; | 934 my $extension; |
| 774 my $testdata; | 935 my $testdata; |
| 775 my %testhash; | 936 my %testhash; |
| 776 my $testfncdata; | 937 my $testfncdata; |
| 777 my %testfnchash; | 938 my %testfnchash; |
| 939 my $testbrdata; |
| 940 my %testbrhash; |
| 778 my @sort_list; | 941 my @sort_list; |
| 779 local *HTML_HANDLE; | 942 local *HTML_HANDLE; |
| 780 | 943 |
| 781 # Remove prefix if applicable | 944 # Remove prefix if applicable |
| 782 if (!$no_prefix) | 945 if (!$no_prefix) |
| 783 { | 946 { |
| 784 # Match directory name beginning with $dir_prefix | 947 # Match directory name beginning with $dir_prefix |
| 785 $rel_dir = apply_prefix($rel_dir, $dir_prefix); | 948 $rel_dir = apply_prefix($rel_dir, $dir_prefix); |
| 786 } | 949 } |
| 787 | 950 |
| 788 $trunc_dir = $rel_dir; | 951 $trunc_dir = $rel_dir; |
| 789 | 952 |
| 790 # Remove leading / | 953 # Remove leading / |
| 791 if ($rel_dir =~ /^\/(.*)$/) | 954 if ($rel_dir =~ /^\/(.*)$/) |
| 792 { | 955 { |
| 793 $rel_dir = substr($rel_dir, 1); | 956 $rel_dir = substr($rel_dir, 1); |
| 794 } | 957 } |
| 795 | 958 |
| 959 # Handle files in root directory gracefully |
| 960 $rel_dir = "root" if ($rel_dir eq ""); |
| 961 $trunc_dir = "root" if ($trunc_dir eq ""); |
| 962 |
| 796 $base_dir = get_relative_base_path($rel_dir); | 963 $base_dir = get_relative_base_path($rel_dir); |
| 797 | 964 |
| 798 create_sub_dir($rel_dir); | 965 create_sub_dir($rel_dir); |
| 799 | 966 |
| 800 # Match filenames which specify files in this directory, not including | 967 # Match filenames which specify files in this directory, not including |
| 801 # sub-directories | 968 # sub-directories |
| 802 foreach $filename (grep(/^\Q$abs_dir\E\/[^\/]*$/,keys(%info_data))) | 969 foreach $filename (grep(/^\Q$abs_dir\E\/[^\/]*$/,keys(%info_data))) |
| 803 { | 970 { |
| 804 my $page_link; | 971 my $page_link; |
| 805 my $func_link; | 972 my $func_link; |
| 806 | 973 |
| 807 » » ($lines_found, $lines_hit, $fn_found, $fn_hit, $testdata, | 974 » » ($lines_found, $lines_hit, $fn_found, $fn_hit, $br_found, |
| 808 » » $testfncdata) = process_file($trunc_dir, $rel_dir, $filename); | 975 » » $br_hit, $testdata, $testfncdata, $testbrdata) = |
| 976 » » » process_file($trunc_dir, $rel_dir, $filename); |
| 809 | 977 |
| 810 $base_name = basename($filename); | 978 $base_name = basename($filename); |
| 811 | 979 |
| 812 if ($no_sourceview) { | 980 if ($no_sourceview) { |
| 813 $page_link = ""; | 981 $page_link = ""; |
| 814 } elsif ($frames) { | 982 } elsif ($frames) { |
| 815 # Link to frameset page | 983 # Link to frameset page |
| 816 $page_link = "$base_name.gcov.frameset.$html_ext"; | 984 $page_link = "$base_name.gcov.frameset.$html_ext"; |
| 817 } else { | 985 } else { |
| 818 # Link directory to source code view page | 986 # Link directory to source code view page |
| 819 $page_link = "$base_name.gcov.$html_ext"; | 987 $page_link = "$base_name.gcov.$html_ext"; |
| 820 } | 988 } |
| 821 $overview{$base_name} = [$lines_found, $lines_hit, $fn_found, | 989 $overview{$base_name} = [$lines_found, $lines_hit, $fn_found, |
| 822 » » » » » $fn_hit, $page_link, | 990 » » » » » $fn_hit, $br_found, $br_hit, |
| 991 » » » » » $page_link, |
| 823 get_rate($lines_found, $lines_hit), | 992 get_rate($lines_found, $lines_hit), |
| 824 » » » » » get_rate($fn_found, $fn_hit)]; | 993 » » » » » get_rate($fn_found, $fn_hit), |
| 994 » » » » » get_rate($br_found, $br_hit)]; |
| 825 | 995 |
| 826 $testhash{$base_name} = $testdata; | 996 $testhash{$base_name} = $testdata; |
| 827 $testfnchash{$base_name} = $testfncdata; | 997 $testfnchash{$base_name} = $testfncdata; |
| 998 $testbrhash{$base_name} = $testbrdata; |
| 828 | 999 |
| 829 $overall_found += $lines_found; | 1000 $overall_found += $lines_found; |
| 830 $overall_hit += $lines_hit; | 1001 $overall_hit += $lines_hit; |
| 831 | 1002 |
| 832 $total_fn_found += $fn_found; | 1003 $total_fn_found += $fn_found; |
| 833 $total_fn_hit += $fn_hit; | 1004 $total_fn_hit += $fn_hit; |
| 1005 |
| 1006 $total_br_found += $br_found; |
| 1007 $total_br_hit += $br_hit; |
| 834 } | 1008 } |
| 835 | 1009 |
| 836 # Create sorted pages | 1010 # Create sorted pages |
| 837 foreach (@fileview_sortlist) { | 1011 foreach (@fileview_sortlist) { |
| 838 # Generate directory overview page (without details) | 1012 # Generate directory overview page (without details) |
| 839 write_dir_page($fileview_sortname[$_], $rel_dir, $base_dir, | 1013 write_dir_page($fileview_sortname[$_], $rel_dir, $base_dir, |
| 840 $test_title, $trunc_dir, $overall_found, | 1014 $test_title, $trunc_dir, $overall_found, |
| 841 $overall_hit, $total_fn_found, $total_fn_hit, | 1015 $overall_hit, $total_fn_found, $total_fn_hit, |
| 842 » » » \%overview, {}, {}, 1, $_); | 1016 » » » $total_br_found, $total_br_hit, \%overview, {}, |
| 1017 » » » {}, {}, 1, $_); |
| 843 if (!$show_details) { | 1018 if (!$show_details) { |
| 844 next; | 1019 next; |
| 845 } | 1020 } |
| 846 # Generate directory overview page including details | 1021 # Generate directory overview page including details |
| 847 write_dir_page("-detail".$fileview_sortname[$_], $rel_dir, | 1022 write_dir_page("-detail".$fileview_sortname[$_], $rel_dir, |
| 848 $base_dir, $test_title, $trunc_dir, | 1023 $base_dir, $test_title, $trunc_dir, |
| 849 $overall_found, $overall_hit, $total_fn_found, | 1024 $overall_found, $overall_hit, $total_fn_found, |
| 850 » » » $total_fn_hit, \%overview, \%testhash, | 1025 » » » $total_fn_hit, $total_br_found, $total_br_hit, |
| 851 » » » \%testfnchash, 1, $_); | 1026 » » » \%overview, \%testhash, \%testfnchash, |
| 1027 » » » \%testbrhash, 1, $_); |
| 852 } | 1028 } |
| 853 | 1029 |
| 854 # Calculate resulting line counts | 1030 # Calculate resulting line counts |
| 855 » return ($overall_found, $overall_hit, $total_fn_found, $total_fn_hit); | 1031 » return ($overall_found, $overall_hit, $total_fn_found, $total_fn_hit, |
| 1032 » » $total_br_found, $total_br_hit); |
| 856 } | 1033 } |
| 857 | 1034 |
| 858 | 1035 |
| 859 # | 1036 # |
| 860 # get_converted_lines(testdata) | 1037 # get_converted_lines(testdata) |
| 861 # | 1038 # |
| 862 # Return hash of line numbers of those lines which were only covered in | 1039 # Return hash of line numbers of those lines which were only covered in |
| 863 # converted data sets. | 1040 # converted data sets. |
| 864 # | 1041 # |
| 865 | 1042 |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 906 if (!defined($nonconverted{$line})) | 1083 if (!defined($nonconverted{$line})) |
| 907 { | 1084 { |
| 908 $result{$line} = 1; | 1085 $result{$line} = 1; |
| 909 } | 1086 } |
| 910 } | 1087 } |
| 911 | 1088 |
| 912 return \%result; | 1089 return \%result; |
| 913 } | 1090 } |
| 914 | 1091 |
| 915 | 1092 |
| 916 sub write_function_page($$$$$$$$$$$$$$) | 1093 sub write_function_page($$$$$$$$$$$$$$$$$$) |
| 917 { | 1094 { |
| 918 my ($base_dir, $rel_dir, $trunc_dir, $base_name, $title, | 1095 my ($base_dir, $rel_dir, $trunc_dir, $base_name, $title, |
| 919 » $lines_found, $lines_hit, $fn_found, $fn_hit, | 1096 » $lines_found, $lines_hit, $fn_found, $fn_hit, $br_found, $br_hit, |
| 920 » $sumcount, $funcdata, $sumfnccount, $testfncdata, $sort_type) = @_; | 1097 » $sumcount, $funcdata, $sumfnccount, $testfncdata, $sumbrcount, |
| 1098 » $testbrdata, $sort_type) = @_; |
| 921 my $pagetitle; | 1099 my $pagetitle; |
| 922 my $filename; | 1100 my $filename; |
| 923 | 1101 |
| 924 # Generate function table for this file | 1102 # Generate function table for this file |
| 925 if ($sort_type == 0) { | 1103 if ($sort_type == 0) { |
| 926 $filename = "$rel_dir/$base_name.func.$html_ext"; | 1104 $filename = "$rel_dir/$base_name.func.$html_ext"; |
| 927 } else { | 1105 } else { |
| 928 $filename = "$rel_dir/$base_name.func-sort-c.$html_ext"; | 1106 $filename = "$rel_dir/$base_name.func-sort-c.$html_ext"; |
| 929 } | 1107 } |
| 930 html_create(*HTML_HANDLE, $filename); | 1108 html_create(*HTML_HANDLE, $filename); |
| 931 $pagetitle = "LCOV - $title - $trunc_dir/$base_name - functions"; | 1109 $pagetitle = "LCOV - $title - $trunc_dir/$base_name - functions"; |
| 932 write_html_prolog(*HTML_HANDLE, $base_dir, $pagetitle); | 1110 write_html_prolog(*HTML_HANDLE, $base_dir, $pagetitle); |
| 933 write_header(*HTML_HANDLE, 4, "$trunc_dir/$base_name", | 1111 write_header(*HTML_HANDLE, 4, "$trunc_dir/$base_name", |
| 934 "$rel_dir/$base_name", $lines_found, $lines_hit, | 1112 "$rel_dir/$base_name", $lines_found, $lines_hit, |
| 935 » » $fn_found, $fn_hit, $sort_type); | 1113 » » $fn_found, $fn_hit, $br_found, $br_hit, $sort_type); |
| 936 write_function_table(*HTML_HANDLE, "$base_name.gcov.$html_ext", | 1114 write_function_table(*HTML_HANDLE, "$base_name.gcov.$html_ext", |
| 937 $sumcount, $funcdata, | 1115 $sumcount, $funcdata, |
| 938 » » » $sumfnccount, $testfncdata, $base_name, | 1116 » » » $sumfnccount, $testfncdata, $sumbrcount, |
| 1117 » » » $testbrdata, $base_name, |
| 939 $base_dir, $sort_type); | 1118 $base_dir, $sort_type); |
| 940 write_html_epilog(*HTML_HANDLE, $base_dir, 1); | 1119 write_html_epilog(*HTML_HANDLE, $base_dir, 1); |
| 941 close(*HTML_HANDLE); | 1120 close(*HTML_HANDLE); |
| 942 } | 1121 } |
| 943 | 1122 |
| 944 | 1123 |
| 945 # | 1124 # |
| 946 # process_file(trunc_dir, rel_dir, filename) | 1125 # process_file(trunc_dir, rel_dir, filename) |
| 947 # | 1126 # |
| 948 | 1127 |
| 949 sub process_file($$$) | 1128 sub process_file($$$) |
| 950 { | 1129 { |
| 951 info("Processing file ".apply_prefix($_[2], $dir_prefix)."\n"); | 1130 info("Processing file ".apply_prefix($_[2], $dir_prefix)."\n"); |
| 952 | 1131 |
| 953 my $trunc_dir = $_[0]; | 1132 my $trunc_dir = $_[0]; |
| 954 my $rel_dir = $_[1]; | 1133 my $rel_dir = $_[1]; |
| 955 my $filename = $_[2]; | 1134 my $filename = $_[2]; |
| 956 my $base_name = basename($filename); | 1135 my $base_name = basename($filename); |
| 957 my $base_dir = get_relative_base_path($rel_dir); | 1136 my $base_dir = get_relative_base_path($rel_dir); |
| 958 my $testdata; | 1137 my $testdata; |
| 959 my $testcount; | 1138 my $testcount; |
| 960 my $sumcount; | 1139 my $sumcount; |
| 961 my $funcdata; | 1140 my $funcdata; |
| 962 my $checkdata; | 1141 my $checkdata; |
| 963 my $testfncdata; | 1142 my $testfncdata; |
| 964 my $sumfnccount; | 1143 my $sumfnccount; |
| 1144 my $testbrdata; |
| 1145 my $sumbrcount; |
| 965 my $lines_found; | 1146 my $lines_found; |
| 966 my $lines_hit; | 1147 my $lines_hit; |
| 967 my $fn_found; | 1148 my $fn_found; |
| 968 my $fn_hit; | 1149 my $fn_hit; |
| 1150 my $br_found; |
| 1151 my $br_hit; |
| 969 my $converted; | 1152 my $converted; |
| 970 my @source; | 1153 my @source; |
| 971 my $pagetitle; | 1154 my $pagetitle; |
| 972 local *HTML_HANDLE; | 1155 local *HTML_HANDLE; |
| 973 | 1156 |
| 974 ($testdata, $sumcount, $funcdata, $checkdata, $testfncdata, | 1157 ($testdata, $sumcount, $funcdata, $checkdata, $testfncdata, |
| 975 » $sumfnccount, $lines_found, $lines_hit, $fn_found, $fn_hit) | 1158 » $sumfnccount, $testbrdata, $sumbrcount, $lines_found, $lines_hit, |
| 1159 » $fn_found, $fn_hit, $br_found, $br_hit) |
| 976 = get_info_entry($info_data{$filename}); | 1160 = get_info_entry($info_data{$filename}); |
| 977 | 1161 |
| 978 # Return after this point in case user asked us not to generate | 1162 # Return after this point in case user asked us not to generate |
| 979 # source code view | 1163 # source code view |
| 980 if ($no_sourceview) | 1164 if ($no_sourceview) |
| 981 { | 1165 { |
| 982 » » return ($lines_found, $lines_hit, | 1166 » » return ($lines_found, $lines_hit, $fn_found, $fn_hit, |
| 983 » » » $fn_found, $fn_hit, $testdata); | 1167 » » » $br_found, $br_hit, $testdata, $testfncdata, |
| 1168 » » » $testbrdata); |
| 984 } | 1169 } |
| 985 | 1170 |
| 986 $converted = get_converted_lines($testdata); | 1171 $converted = get_converted_lines($testdata); |
| 987 # Generate source code view for this file | 1172 # Generate source code view for this file |
| 988 html_create(*HTML_HANDLE, "$rel_dir/$base_name.gcov.$html_ext"); | 1173 html_create(*HTML_HANDLE, "$rel_dir/$base_name.gcov.$html_ext"); |
| 989 $pagetitle = "LCOV - $test_title - $trunc_dir/$base_name"; | 1174 $pagetitle = "LCOV - $test_title - $trunc_dir/$base_name"; |
| 990 write_html_prolog(*HTML_HANDLE, $base_dir, $pagetitle); | 1175 write_html_prolog(*HTML_HANDLE, $base_dir, $pagetitle); |
| 991 write_header(*HTML_HANDLE, 2, "$trunc_dir/$base_name", | 1176 write_header(*HTML_HANDLE, 2, "$trunc_dir/$base_name", |
| 992 "$rel_dir/$base_name", $lines_found, $lines_hit, | 1177 "$rel_dir/$base_name", $lines_found, $lines_hit, |
| 993 » » $fn_found, $fn_hit, 0); | 1178 » » $fn_found, $fn_hit, $br_found, $br_hit, 0); |
| 994 @source = write_source(*HTML_HANDLE, $filename, $sumcount, $checkdata, | 1179 @source = write_source(*HTML_HANDLE, $filename, $sumcount, $checkdata, |
| 995 » » » $converted, $funcdata); | 1180 » » » $converted, $funcdata, $sumbrcount); |
| 996 | 1181 |
| 997 write_html_epilog(*HTML_HANDLE, $base_dir, 1); | 1182 write_html_epilog(*HTML_HANDLE, $base_dir, 1); |
| 998 close(*HTML_HANDLE); | 1183 close(*HTML_HANDLE); |
| 999 | 1184 |
| 1000 if ($func_coverage) { | 1185 if ($func_coverage) { |
| 1001 # Create function tables | 1186 # Create function tables |
| 1002 foreach (@funcview_sortlist) { | 1187 foreach (@funcview_sortlist) { |
| 1003 write_function_page($base_dir, $rel_dir, $trunc_dir, | 1188 write_function_page($base_dir, $rel_dir, $trunc_dir, |
| 1004 $base_name, $test_title, | 1189 $base_name, $test_title, |
| 1005 $lines_found, $lines_hit, | 1190 $lines_found, $lines_hit, |
| 1006 » » » » » $fn_found, $fn_hit, $sumcount, | 1191 » » » » » $fn_found, $fn_hit, $br_found, |
| 1192 » » » » » $br_hit, $sumcount, |
| 1007 $funcdata, $sumfnccount, | 1193 $funcdata, $sumfnccount, |
| 1008 » » » » » $testfncdata, $_); | 1194 » » » » » $testfncdata, $sumbrcount, |
| 1195 » » » » » $testbrdata, $_); |
| 1009 } | 1196 } |
| 1010 } | 1197 } |
| 1011 | 1198 |
| 1012 # Additional files are needed in case of frame output | 1199 # Additional files are needed in case of frame output |
| 1013 if (!$frames) | 1200 if (!$frames) |
| 1014 { | 1201 { |
| 1015 » » return ($lines_found, $lines_hit, | 1202 » » return ($lines_found, $lines_hit, $fn_found, $fn_hit, |
| 1016 » » » $fn_found, $fn_hit, $testdata); | 1203 » » » $br_found, $br_hit, $testdata, $testfncdata, |
| 1204 » » » $testbrdata); |
| 1017 } | 1205 } |
| 1018 | 1206 |
| 1019 # Create overview png file | 1207 # Create overview png file |
| 1020 gen_png("$rel_dir/$base_name.gcov.png", $overview_width, $tab_size, | 1208 gen_png("$rel_dir/$base_name.gcov.png", $overview_width, $tab_size, |
| 1021 @source); | 1209 @source); |
| 1022 | 1210 |
| 1023 # Create frameset page | 1211 # Create frameset page |
| 1024 html_create(*HTML_HANDLE, | 1212 html_create(*HTML_HANDLE, |
| 1025 "$rel_dir/$base_name.gcov.frameset.$html_ext"); | 1213 "$rel_dir/$base_name.gcov.frameset.$html_ext"); |
| 1026 write_frameset(*HTML_HANDLE, $base_dir, $base_name, $pagetitle); | 1214 write_frameset(*HTML_HANDLE, $base_dir, $base_name, $pagetitle); |
| 1027 close(*HTML_HANDLE); | 1215 close(*HTML_HANDLE); |
| 1028 | 1216 |
| 1029 # Write overview frame | 1217 # Write overview frame |
| 1030 html_create(*HTML_HANDLE, | 1218 html_create(*HTML_HANDLE, |
| 1031 "$rel_dir/$base_name.gcov.overview.$html_ext"); | 1219 "$rel_dir/$base_name.gcov.overview.$html_ext"); |
| 1032 write_overview(*HTML_HANDLE, $base_dir, $base_name, $pagetitle, | 1220 write_overview(*HTML_HANDLE, $base_dir, $base_name, $pagetitle, |
| 1033 scalar(@source)); | 1221 scalar(@source)); |
| 1034 close(*HTML_HANDLE); | 1222 close(*HTML_HANDLE); |
| 1035 | 1223 |
| 1036 » return ($lines_found, $lines_hit, $fn_found, $fn_hit, $testdata, | 1224 » return ($lines_found, $lines_hit, $fn_found, $fn_hit, $br_found, |
| 1037 » » $testfncdata); | 1225 » » $br_hit, $testdata, $testfncdata, $testbrdata); |
| 1038 } | 1226 } |
| 1039 | 1227 |
| 1040 | 1228 |
| 1041 # | 1229 # |
| 1042 # read_info_file(info_filename) | 1230 # read_info_file(info_filename) |
| 1043 # | 1231 # |
| 1044 # Read in the contents of the .info file specified by INFO_FILENAME. Data will | 1232 # Read in the contents of the .info file specified by INFO_FILENAME. Data will |
| 1045 # be returned as a reference to a hash containing the following mappings: | 1233 # be returned as a reference to a hash containing the following mappings: |
| 1046 # | 1234 # |
| 1047 # %result: for each filename found in file -> \%data | 1235 # %result: for each filename found in file -> \%data |
| 1048 # | 1236 # |
| 1049 # %data: "test" -> \%testdata | 1237 # %data: "test" -> \%testdata |
| 1050 # "sum" -> \%sumcount | 1238 # "sum" -> \%sumcount |
| 1051 # "func" -> \%funcdata | 1239 # "func" -> \%funcdata |
| 1052 # "found" -> $lines_found (number of instrumented lines found in file) | 1240 # "found" -> $lines_found (number of instrumented lines found in file) |
| 1053 # "hit" -> $lines_hit (number of executed lines in file) | 1241 # "hit" -> $lines_hit (number of executed lines in file) |
| 1054 # "check" -> \%checkdata | 1242 # "check" -> \%checkdata |
| 1055 # "testfnc" -> \%testfncdata | 1243 # "testfnc" -> \%testfncdata |
| 1056 # "sumfnc" -> \%sumfnccount | 1244 # "sumfnc" -> \%sumfnccount |
| 1245 # "testbr" -> \%testbrdata |
| 1246 # "sumbr" -> \%sumbrcount |
| 1057 # | 1247 # |
| 1058 # %testdata : name of test affecting this file -> \%testcount | 1248 # %testdata : name of test affecting this file -> \%testcount |
| 1059 # %testfncdata: name of test affecting this file -> \%testfnccount | 1249 # %testfncdata: name of test affecting this file -> \%testfnccount |
| 1250 # %testbrdata: name of test affecting this file -> \%testbrcount |
| 1060 # | 1251 # |
| 1061 # %testcount : line number -> execution count for a single test | 1252 # %testcount : line number -> execution count for a single test |
| 1062 # %testfnccount: function name -> execution count for a single test | 1253 # %testfnccount: function name -> execution count for a single test |
| 1254 # %testbrcount : line number -> branch coverage data for a single test |
| 1063 # %sumcount : line number -> execution count for all tests | 1255 # %sumcount : line number -> execution count for all tests |
| 1064 # %sumfnccount : function name -> execution count for all tests | 1256 # %sumfnccount : function name -> execution count for all tests |
| 1257 # %sumbrcount : line number -> branch coverage data for all tests |
| 1065 # %funcdata : function name -> line number | 1258 # %funcdata : function name -> line number |
| 1066 # %checkdata : line number -> checksum of source code line | 1259 # %checkdata : line number -> checksum of source code line |
| 1260 # $brdata : vector of items: block, branch, taken |
| 1067 # | 1261 # |
| 1068 # Note that .info file sections referring to the same file and test name | 1262 # Note that .info file sections referring to the same file and test name |
| 1069 # will automatically be combined by adding all execution counts. | 1263 # will automatically be combined by adding all execution counts. |
| 1070 # | 1264 # |
| 1071 # Note that if INFO_FILENAME ends with ".gz", it is assumed that the file | 1265 # Note that if INFO_FILENAME ends with ".gz", it is assumed that the file |
| 1072 # is compressed using GZIP. If available, GUNZIP will be used to decompress | 1266 # is compressed using GZIP. If available, GUNZIP will be used to decompress |
| 1073 # this file. | 1267 # this file. |
| 1074 # | 1268 # |
| 1075 # Die on error. | 1269 # Die on error. |
| 1076 # | 1270 # |
| 1077 | 1271 |
| 1078 sub read_info_file($) | 1272 sub read_info_file($) |
| 1079 { | 1273 { |
| 1080 my $tracefile = $_[0]; # Name of tracefile | 1274 my $tracefile = $_[0]; # Name of tracefile |
| 1081 my %result; # Resulting hash: file -> data | 1275 my %result; # Resulting hash: file -> data |
| 1082 my $data; # Data handle for current entry | 1276 my $data; # Data handle for current entry |
| 1083 my $testdata; # " " | 1277 my $testdata; # " " |
| 1084 my $testcount; # " " | 1278 my $testcount; # " " |
| 1085 my $sumcount; # " " | 1279 my $sumcount; # " " |
| 1086 my $funcdata; # " " | 1280 my $funcdata; # " " |
| 1087 my $checkdata; # " " | 1281 my $checkdata; # " " |
| 1088 my $testfncdata; | 1282 my $testfncdata; |
| 1089 my $testfnccount; | 1283 my $testfnccount; |
| 1090 my $sumfnccount; | 1284 my $sumfnccount; |
| 1285 my $testbrdata; |
| 1286 my $testbrcount; |
| 1287 my $sumbrcount; |
| 1091 my $line; # Current line read from .info file | 1288 my $line; # Current line read from .info file |
| 1092 my $testname; # Current test name | 1289 my $testname; # Current test name |
| 1093 my $filename; # Current filename | 1290 my $filename; # Current filename |
| 1094 my $hitcount; # Count for lines hit | 1291 my $hitcount; # Count for lines hit |
| 1095 my $count; # Execution count of current line | 1292 my $count; # Execution count of current line |
| 1096 my $negative; # If set, warn about negative counts | 1293 my $negative; # If set, warn about negative counts |
| 1097 my $changed_testname; # If set, warn about changed testname | 1294 my $changed_testname; # If set, warn about changed testname |
| 1098 my $line_checksum; # Checksum of current line | 1295 my $line_checksum; # Checksum of current line |
| 1296 my $br_found; |
| 1297 my $br_hit; |
| 1099 local *INFO_HANDLE; # Filehandle for .info file | 1298 local *INFO_HANDLE; # Filehandle for .info file |
| 1100 | 1299 |
| 1101 info("Reading data file $tracefile\n"); | 1300 info("Reading data file $tracefile\n"); |
| 1102 | 1301 |
| 1103 # Check if file exists and is readable | 1302 # Check if file exists and is readable |
| 1104 stat($_[0]); | 1303 stat($_[0]); |
| 1105 if (!(-r _)) | 1304 if (!(-r _)) |
| 1106 { | 1305 { |
| 1107 die("ERROR: cannot read file $_[0]!\n"); | 1306 die("ERROR: cannot read file $_[0]!\n"); |
| 1108 } | 1307 } |
| (...skipping 10 matching lines...) Expand all Loading... |
| 1119 # Check for availability of GZIP tool | 1318 # Check for availability of GZIP tool |
| 1120 system_no_output(1, "gunzip" ,"-h") | 1319 system_no_output(1, "gunzip" ,"-h") |
| 1121 and die("ERROR: gunzip command not available!\n"); | 1320 and die("ERROR: gunzip command not available!\n"); |
| 1122 | 1321 |
| 1123 # Check integrity of compressed file | 1322 # Check integrity of compressed file |
| 1124 system_no_output(1, "gunzip", "-t", $_[0]) | 1323 system_no_output(1, "gunzip", "-t", $_[0]) |
| 1125 and die("ERROR: integrity check failed for ". | 1324 and die("ERROR: integrity check failed for ". |
| 1126 "compressed file $_[0]!\n"); | 1325 "compressed file $_[0]!\n"); |
| 1127 | 1326 |
| 1128 # Open compressed file | 1327 # Open compressed file |
| 1129 » » open(INFO_HANDLE, "gunzip -c $_[0]|") | 1328 » » open(INFO_HANDLE, "-|", "gunzip -c '$_[0]'") |
| 1130 or die("ERROR: cannot start gunzip to decompress ". | 1329 or die("ERROR: cannot start gunzip to decompress ". |
| 1131 "file $_[0]!\n"); | 1330 "file $_[0]!\n"); |
| 1132 } | 1331 } |
| 1133 else | 1332 else |
| 1134 { | 1333 { |
| 1135 # Open decompressed file | 1334 # Open decompressed file |
| 1136 » » open(INFO_HANDLE, $_[0]) | 1335 » » open(INFO_HANDLE, "<", $_[0]) |
| 1137 or die("ERROR: cannot read file $_[0]!\n"); | 1336 or die("ERROR: cannot read file $_[0]!\n"); |
| 1138 } | 1337 } |
| 1139 | 1338 |
| 1140 $testname = ""; | 1339 $testname = ""; |
| 1141 while (<INFO_HANDLE>) | 1340 while (<INFO_HANDLE>) |
| 1142 { | 1341 { |
| 1143 chomp($_); | 1342 chomp($_); |
| 1144 $line = $_; | 1343 $line = $_; |
| 1145 | 1344 |
| 1146 # Switch statement | 1345 # Switch statement |
| 1147 foreach ($line) | 1346 foreach ($line) |
| 1148 { | 1347 { |
| 1149 » » » /^TN:([^,]*)/ && do | 1348 » » » /^TN:([^,]*)(,diff)?/ && do |
| 1150 { | 1349 { |
| 1151 # Test name information found | 1350 # Test name information found |
| 1152 $testname = defined($1) ? $1 : ""; | 1351 $testname = defined($1) ? $1 : ""; |
| 1153 if ($testname =~ s/\W/_/g) | 1352 if ($testname =~ s/\W/_/g) |
| 1154 { | 1353 { |
| 1155 $changed_testname = 1; | 1354 $changed_testname = 1; |
| 1156 } | 1355 } |
| 1356 $testname .= $2 if (defined($2)); |
| 1157 last; | 1357 last; |
| 1158 }; | 1358 }; |
| 1159 | 1359 |
| 1160 /^[SK]F:(.*)/ && do | 1360 /^[SK]F:(.*)/ && do |
| 1161 { | 1361 { |
| 1162 # Filename information found | 1362 # Filename information found |
| 1163 # Retrieve data for new entry | 1363 # Retrieve data for new entry |
| 1164 $filename = $1; | 1364 $filename = $1; |
| 1165 | 1365 |
| 1166 $data = $result{$filename}; | 1366 $data = $result{$filename}; |
| 1167 ($testdata, $sumcount, $funcdata, $checkdata, | 1367 ($testdata, $sumcount, $funcdata, $checkdata, |
| 1168 » » » » $testfncdata, $sumfnccount) = | 1368 » » » » $testfncdata, $sumfnccount, $testbrdata, |
| 1369 » » » » $sumbrcount) = |
| 1169 get_info_entry($data); | 1370 get_info_entry($data); |
| 1170 | 1371 |
| 1171 if (defined($testname)) | 1372 if (defined($testname)) |
| 1172 { | 1373 { |
| 1173 $testcount = $testdata->{$testname}; | 1374 $testcount = $testdata->{$testname}; |
| 1174 $testfnccount = $testfncdata->{$testname
}; | 1375 $testfnccount = $testfncdata->{$testname
}; |
| 1376 $testbrcount = $testbrdata->{$testname}; |
| 1175 } | 1377 } |
| 1176 else | 1378 else |
| 1177 { | 1379 { |
| 1178 $testcount = {}; | 1380 $testcount = {}; |
| 1179 $testfnccount = {}; | 1381 $testfnccount = {}; |
| 1382 $testbrcount = {}; |
| 1180 } | 1383 } |
| 1181 last; | 1384 last; |
| 1182 }; | 1385 }; |
| 1183 | 1386 |
| 1184 /^DA:(\d+),(-?\d+)(,[^,\s]+)?/ && do | 1387 /^DA:(\d+),(-?\d+)(,[^,\s]+)?/ && do |
| 1185 { | 1388 { |
| 1186 # Fix negative counts | 1389 # Fix negative counts |
| 1187 $count = $2 < 0 ? 0 : $2; | 1390 $count = $2 < 0 ? 0 : $2; |
| 1188 if ($2 < 0) | 1391 if ($2 < 0) |
| 1189 { | 1392 { |
| (...skipping 23 matching lines...) Expand all Loading... |
| 1213 "at $filename:$1\n"); | 1416 "at $filename:$1\n"); |
| 1214 } | 1417 } |
| 1215 | 1418 |
| 1216 $checkdata->{$1} = $line_checksum; | 1419 $checkdata->{$1} = $line_checksum; |
| 1217 } | 1420 } |
| 1218 last; | 1421 last; |
| 1219 }; | 1422 }; |
| 1220 | 1423 |
| 1221 /^FN:(\d+),([^,]+)/ && do | 1424 /^FN:(\d+),([^,]+)/ && do |
| 1222 { | 1425 { |
| 1426 last if (!$func_coverage); |
| 1427 |
| 1223 # Function data found, add to structure | 1428 # Function data found, add to structure |
| 1224 $funcdata->{$2} = $1; | 1429 $funcdata->{$2} = $1; |
| 1225 | 1430 |
| 1226 # Also initialize function call data | 1431 # Also initialize function call data |
| 1227 if (!defined($sumfnccount->{$2})) { | 1432 if (!defined($sumfnccount->{$2})) { |
| 1228 $sumfnccount->{$2} = 0; | 1433 $sumfnccount->{$2} = 0; |
| 1229 } | 1434 } |
| 1230 if (defined($testname)) | 1435 if (defined($testname)) |
| 1231 { | 1436 { |
| 1232 if (!defined($testfnccount->{$2})) { | 1437 if (!defined($testfnccount->{$2})) { |
| 1233 $testfnccount->{$2} = 0; | 1438 $testfnccount->{$2} = 0; |
| 1234 } | 1439 } |
| 1235 } | 1440 } |
| 1236 last; | 1441 last; |
| 1237 }; | 1442 }; |
| 1238 | 1443 |
| 1239 /^FNDA:(\d+),([^,]+)/ && do | 1444 /^FNDA:(\d+),([^,]+)/ && do |
| 1240 { | 1445 { |
| 1446 last if (!$func_coverage); |
| 1241 # Function call count found, add to structure | 1447 # Function call count found, add to structure |
| 1242 # Add summary counts | 1448 # Add summary counts |
| 1243 $sumfnccount->{$2} += $1; | 1449 $sumfnccount->{$2} += $1; |
| 1244 | 1450 |
| 1245 # Add test-specific counts | 1451 # Add test-specific counts |
| 1246 if (defined($testname)) | 1452 if (defined($testname)) |
| 1247 { | 1453 { |
| 1248 $testfnccount->{$2} += $1; | 1454 $testfnccount->{$2} += $1; |
| 1249 } | 1455 } |
| 1250 last; | 1456 last; |
| 1251 }; | 1457 }; |
| 1458 |
| 1459 /^BRDA:(\d+),(\d+),(\d+),(\d+|-)/ && do { |
| 1460 # Branch coverage data found |
| 1461 my ($line, $block, $branch, $taken) = |
| 1462 ($1, $2, $3, $4); |
| 1463 |
| 1464 last if (!$br_coverage); |
| 1465 $sumbrcount->{$line} = |
| 1466 br_ivec_push($sumbrcount->{$line}, |
| 1467 $block, $branch, $taken); |
| 1468 |
| 1469 # Add test-specific counts |
| 1470 if (defined($testname)) { |
| 1471 $testbrcount->{$line} = |
| 1472 br_ivec_push( |
| 1473 $testbrcount->{$line}, |
| 1474 $block, $branch, |
| 1475 $taken); |
| 1476 } |
| 1477 last; |
| 1478 }; |
| 1479 |
| 1252 /^end_of_record/ && do | 1480 /^end_of_record/ && do |
| 1253 { | 1481 { |
| 1254 # Found end of section marker | 1482 # Found end of section marker |
| 1255 if ($filename) | 1483 if ($filename) |
| 1256 { | 1484 { |
| 1257 # Store current section data | 1485 # Store current section data |
| 1258 if (defined($testname)) | 1486 if (defined($testname)) |
| 1259 { | 1487 { |
| 1260 $testdata->{$testname} = | 1488 $testdata->{$testname} = |
| 1261 $testcount; | 1489 $testcount; |
| 1262 $testfncdata->{$testname} = | 1490 $testfncdata->{$testname} = |
| 1263 $testfnccount; | 1491 $testfnccount; |
| 1492 $testbrdata->{$testname} = |
| 1493 $testbrcount; |
| 1264 } | 1494 } |
| 1265 | 1495 |
| 1266 set_info_entry($data, $testdata, | 1496 set_info_entry($data, $testdata, |
| 1267 $sumcount, $funcdata, | 1497 $sumcount, $funcdata, |
| 1268 $checkdata, $testfncdata, | 1498 $checkdata, $testfncdata, |
| 1269 » » » » » » $sumfnccount); | 1499 » » » » » » $sumfnccount, |
| 1500 » » » » » » $testbrdata, |
| 1501 » » » » » » $sumbrcount); |
| 1270 $result{$filename} = $data; | 1502 $result{$filename} = $data; |
| 1271 last; | 1503 last; |
| 1272 } | 1504 } |
| 1273 }; | 1505 }; |
| 1274 | 1506 |
| 1275 # default | 1507 # default |
| 1276 last; | 1508 last; |
| 1277 } | 1509 } |
| 1278 } | 1510 } |
| 1279 close(INFO_HANDLE); | 1511 close(INFO_HANDLE); |
| 1280 | 1512 |
| 1281 # Calculate lines_found and lines_hit for each file | 1513 # Calculate lines_found and lines_hit for each file |
| 1282 foreach $filename (keys(%result)) | 1514 foreach $filename (keys(%result)) |
| 1283 { | 1515 { |
| 1284 $data = $result{$filename}; | 1516 $data = $result{$filename}; |
| 1285 | 1517 |
| 1286 ($testdata, $sumcount, undef, undef, $testfncdata, | 1518 ($testdata, $sumcount, undef, undef, $testfncdata, |
| 1287 » » $sumfnccount) = get_info_entry($data); | 1519 » » $sumfnccount, $testbrdata, $sumbrcount) = |
| 1520 » » » get_info_entry($data); |
| 1288 | 1521 |
| 1289 # Filter out empty files | 1522 # Filter out empty files |
| 1290 if (scalar(keys(%{$sumcount})) == 0) | 1523 if (scalar(keys(%{$sumcount})) == 0) |
| 1291 { | 1524 { |
| 1292 delete($result{$filename}); | 1525 delete($result{$filename}); |
| 1293 next; | 1526 next; |
| 1294 } | 1527 } |
| 1295 # Filter out empty test cases | 1528 # Filter out empty test cases |
| 1296 foreach $testname (keys(%{$testdata})) | 1529 foreach $testname (keys(%{$testdata})) |
| 1297 { | 1530 { |
| (...skipping 18 matching lines...) Expand all Loading... |
| 1316 # Get found/hit values for function call data | 1549 # Get found/hit values for function call data |
| 1317 $data->{"f_found"} = scalar(keys(%{$sumfnccount})); | 1550 $data->{"f_found"} = scalar(keys(%{$sumfnccount})); |
| 1318 $hitcount = 0; | 1551 $hitcount = 0; |
| 1319 | 1552 |
| 1320 foreach (keys(%{$sumfnccount})) { | 1553 foreach (keys(%{$sumfnccount})) { |
| 1321 if ($sumfnccount->{$_} > 0) { | 1554 if ($sumfnccount->{$_} > 0) { |
| 1322 $hitcount++; | 1555 $hitcount++; |
| 1323 } | 1556 } |
| 1324 } | 1557 } |
| 1325 $data->{"f_hit"} = $hitcount; | 1558 $data->{"f_hit"} = $hitcount; |
| 1559 |
| 1560 # Get found/hit values for branch data |
| 1561 ($br_found, $br_hit) = get_br_found_and_hit($sumbrcount); |
| 1562 |
| 1563 $data->{"b_found"} = $br_found; |
| 1564 $data->{"b_hit"} = $br_hit; |
| 1326 } | 1565 } |
| 1327 | 1566 |
| 1328 if (scalar(keys(%result)) == 0) | 1567 if (scalar(keys(%result)) == 0) |
| 1329 { | 1568 { |
| 1330 die("ERROR: no valid records found in tracefile $tracefile\n"); | 1569 die("ERROR: no valid records found in tracefile $tracefile\n"); |
| 1331 } | 1570 } |
| 1332 if ($negative) | 1571 if ($negative) |
| 1333 { | 1572 { |
| 1334 warn("WARNING: negative counts found in tracefile ". | 1573 warn("WARNING: negative counts found in tracefile ". |
| 1335 "$tracefile\n"); | 1574 "$tracefile\n"); |
| (...skipping 19 matching lines...) Expand all Loading... |
| 1355 # | 1594 # |
| 1356 | 1595 |
| 1357 sub get_info_entry($) | 1596 sub get_info_entry($) |
| 1358 { | 1597 { |
| 1359 my $testdata_ref = $_[0]->{"test"}; | 1598 my $testdata_ref = $_[0]->{"test"}; |
| 1360 my $sumcount_ref = $_[0]->{"sum"}; | 1599 my $sumcount_ref = $_[0]->{"sum"}; |
| 1361 my $funcdata_ref = $_[0]->{"func"}; | 1600 my $funcdata_ref = $_[0]->{"func"}; |
| 1362 my $checkdata_ref = $_[0]->{"check"}; | 1601 my $checkdata_ref = $_[0]->{"check"}; |
| 1363 my $testfncdata = $_[0]->{"testfnc"}; | 1602 my $testfncdata = $_[0]->{"testfnc"}; |
| 1364 my $sumfnccount = $_[0]->{"sumfnc"}; | 1603 my $sumfnccount = $_[0]->{"sumfnc"}; |
| 1604 my $testbrdata = $_[0]->{"testbr"}; |
| 1605 my $sumbrcount = $_[0]->{"sumbr"}; |
| 1365 my $lines_found = $_[0]->{"found"}; | 1606 my $lines_found = $_[0]->{"found"}; |
| 1366 my $lines_hit = $_[0]->{"hit"}; | 1607 my $lines_hit = $_[0]->{"hit"}; |
| 1367 my $fn_found = $_[0]->{"f_found"}; | 1608 my $fn_found = $_[0]->{"f_found"}; |
| 1368 my $fn_hit = $_[0]->{"f_hit"}; | 1609 my $fn_hit = $_[0]->{"f_hit"}; |
| 1610 my $br_found = $_[0]->{"b_found"}; |
| 1611 my $br_hit = $_[0]->{"b_hit"}; |
| 1369 | 1612 |
| 1370 return ($testdata_ref, $sumcount_ref, $funcdata_ref, $checkdata_ref, | 1613 return ($testdata_ref, $sumcount_ref, $funcdata_ref, $checkdata_ref, |
| 1371 » » $testfncdata, $sumfnccount, $lines_found, $lines_hit, | 1614 » » $testfncdata, $sumfnccount, $testbrdata, $sumbrcount, |
| 1372 » » $fn_found, $fn_hit); | 1615 » » $lines_found, $lines_hit, $fn_found, $fn_hit, |
| 1616 » » $br_found, $br_hit); |
| 1373 } | 1617 } |
| 1374 | 1618 |
| 1375 | 1619 |
| 1376 # | 1620 # |
| 1377 # set_info_entry(hash_ref, testdata_ref, sumcount_ref, funcdata_ref, | 1621 # set_info_entry(hash_ref, testdata_ref, sumcount_ref, funcdata_ref, |
| 1378 # checkdata_ref, testfncdata_ref, sumfcncount_ref[,lines_found, | 1622 # checkdata_ref, testfncdata_ref, sumfcncount_ref, |
| 1379 # lines_hit, f_found, f_hit]) | 1623 # testbrdata_ref, sumbrcount_ref[,lines_found, |
| 1624 # lines_hit, f_found, f_hit, $b_found, $b_hit]) |
| 1380 # | 1625 # |
| 1381 # Update the hash referenced by HASH_REF with the provided data references. | 1626 # Update the hash referenced by HASH_REF with the provided data references. |
| 1382 # | 1627 # |
| 1383 | 1628 |
| 1384 sub set_info_entry($$$$$$$;$$$$) | 1629 sub set_info_entry($$$$$$$$$;$$$$$$) |
| 1385 { | 1630 { |
| 1386 my $data_ref = $_[0]; | 1631 my $data_ref = $_[0]; |
| 1387 | 1632 |
| 1388 $data_ref->{"test"} = $_[1]; | 1633 $data_ref->{"test"} = $_[1]; |
| 1389 $data_ref->{"sum"} = $_[2]; | 1634 $data_ref->{"sum"} = $_[2]; |
| 1390 $data_ref->{"func"} = $_[3]; | 1635 $data_ref->{"func"} = $_[3]; |
| 1391 $data_ref->{"check"} = $_[4]; | 1636 $data_ref->{"check"} = $_[4]; |
| 1392 $data_ref->{"testfnc"} = $_[5]; | 1637 $data_ref->{"testfnc"} = $_[5]; |
| 1393 $data_ref->{"sumfnc"} = $_[6]; | 1638 $data_ref->{"sumfnc"} = $_[6]; |
| 1639 $data_ref->{"testbr"} = $_[7]; |
| 1640 $data_ref->{"sumbr"} = $_[8]; |
| 1394 | 1641 |
| 1395 » if (defined($_[7])) { $data_ref->{"found"} = $_[7]; } | 1642 » if (defined($_[9])) { $data_ref->{"found"} = $_[9]; } |
| 1396 » if (defined($_[8])) { $data_ref->{"hit"} = $_[8]; } | 1643 » if (defined($_[10])) { $data_ref->{"hit"} = $_[10]; } |
| 1397 » if (defined($_[9])) { $data_ref->{"f_found"} = $_[9]; } | 1644 » if (defined($_[11])) { $data_ref->{"f_found"} = $_[11]; } |
| 1398 » if (defined($_[10])) { $data_ref->{"f_hit"} = $_[10]; } | 1645 » if (defined($_[12])) { $data_ref->{"f_hit"} = $_[12]; } |
| 1646 » if (defined($_[13])) { $data_ref->{"b_found"} = $_[13]; } |
| 1647 » if (defined($_[14])) { $data_ref->{"b_hit"} = $_[14]; } |
| 1399 } | 1648 } |
| 1400 | 1649 |
| 1401 | 1650 |
| 1402 # | 1651 # |
| 1403 # add_counts(data1_ref, data2_ref) | 1652 # add_counts(data1_ref, data2_ref) |
| 1404 # | 1653 # |
| 1405 # DATA1_REF and DATA2_REF are references to hashes containing a mapping | 1654 # DATA1_REF and DATA2_REF are references to hashes containing a mapping |
| 1406 # | 1655 # |
| 1407 # line number -> execution count | 1656 # line number -> execution count |
| 1408 # | 1657 # |
| (...skipping 87 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1496 # | 1745 # |
| 1497 # merge_func_data(funcdata1, funcdata2, filename) | 1746 # merge_func_data(funcdata1, funcdata2, filename) |
| 1498 # | 1747 # |
| 1499 | 1748 |
| 1500 sub merge_func_data($$$) | 1749 sub merge_func_data($$$) |
| 1501 { | 1750 { |
| 1502 my ($funcdata1, $funcdata2, $filename) = @_; | 1751 my ($funcdata1, $funcdata2, $filename) = @_; |
| 1503 my %result; | 1752 my %result; |
| 1504 my $func; | 1753 my $func; |
| 1505 | 1754 |
| 1506 » %result = %{$funcdata1}; | 1755 » if (defined($funcdata1)) { |
| 1756 » » %result = %{$funcdata1}; |
| 1757 » } |
| 1507 | 1758 |
| 1508 foreach $func (keys(%{$funcdata2})) { | 1759 foreach $func (keys(%{$funcdata2})) { |
| 1509 my $line1 = $result{$func}; | 1760 my $line1 = $result{$func}; |
| 1510 my $line2 = $funcdata2->{$func}; | 1761 my $line2 = $funcdata2->{$func}; |
| 1511 | 1762 |
| 1512 if (defined($line1) && ($line1 != $line2)) { | 1763 if (defined($line1) && ($line1 != $line2)) { |
| 1513 warn("WARNING: function data mismatch at ". | 1764 warn("WARNING: function data mismatch at ". |
| 1514 "$filename:$line2\n"); | 1765 "$filename:$line2\n"); |
| 1515 next; | 1766 next; |
| 1516 } | 1767 } |
| (...skipping 11 matching lines...) Expand all Loading... |
| 1528 # | 1779 # |
| 1529 | 1780 |
| 1530 sub add_fnccount($$) | 1781 sub add_fnccount($$) |
| 1531 { | 1782 { |
| 1532 my ($fnccount1, $fnccount2) = @_; | 1783 my ($fnccount1, $fnccount2) = @_; |
| 1533 my %result; | 1784 my %result; |
| 1534 my $fn_found; | 1785 my $fn_found; |
| 1535 my $fn_hit; | 1786 my $fn_hit; |
| 1536 my $function; | 1787 my $function; |
| 1537 | 1788 |
| 1538 » %result = %{$fnccount1}; | 1789 » if (defined($fnccount1)) { |
| 1790 » » %result = %{$fnccount1}; |
| 1791 » } |
| 1539 foreach $function (keys(%{$fnccount2})) { | 1792 foreach $function (keys(%{$fnccount2})) { |
| 1540 $result{$function} += $fnccount2->{$function}; | 1793 $result{$function} += $fnccount2->{$function}; |
| 1541 } | 1794 } |
| 1542 $fn_found = scalar(keys(%result)); | 1795 $fn_found = scalar(keys(%result)); |
| 1543 $fn_hit = 0; | 1796 $fn_hit = 0; |
| 1544 foreach $function (keys(%result)) { | 1797 foreach $function (keys(%result)) { |
| 1545 if ($result{$function} > 0) { | 1798 if ($result{$function} > 0) { |
| 1546 $fn_hit++; | 1799 $fn_hit++; |
| 1547 } | 1800 } |
| 1548 } | 1801 } |
| (...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1582 | 1835 |
| 1583 # Add count data for testnames unique to data set 2 | 1836 # Add count data for testnames unique to data set 2 |
| 1584 foreach $testname (keys(%{$testfncdata2})) { | 1837 foreach $testname (keys(%{$testfncdata2})) { |
| 1585 if (!defined($result{$testname})) { | 1838 if (!defined($result{$testname})) { |
| 1586 $result{$testname} = $testfncdata2->{$testname}; | 1839 $result{$testname} = $testfncdata2->{$testname}; |
| 1587 } | 1840 } |
| 1588 } | 1841 } |
| 1589 return \%result; | 1842 return \%result; |
| 1590 } | 1843 } |
| 1591 | 1844 |
| 1845 |
| 1846 # |
| 1847 # brcount_to_db(brcount) |
| 1848 # |
| 1849 # Convert brcount data to the following format: |
| 1850 # |
| 1851 # db: line number -> block hash |
| 1852 # block hash: block number -> branch hash |
| 1853 # branch hash: branch number -> taken value |
| 1854 # |
| 1855 |
| 1856 sub brcount_to_db($) |
| 1857 { |
| 1858 my ($brcount) = @_; |
| 1859 my $line; |
| 1860 my $db; |
| 1861 |
| 1862 # Add branches from first count to database |
| 1863 foreach $line (keys(%{$brcount})) { |
| 1864 my $brdata = $brcount->{$line}; |
| 1865 my $i; |
| 1866 my $num = br_ivec_len($brdata); |
| 1867 |
| 1868 for ($i = 0; $i < $num; $i++) { |
| 1869 my ($block, $branch, $taken) = br_ivec_get($brdata, $i); |
| 1870 |
| 1871 $db->{$line}->{$block}->{$branch} = $taken; |
| 1872 } |
| 1873 } |
| 1874 |
| 1875 return $db; |
| 1876 } |
| 1877 |
| 1878 |
| 1879 # |
| 1880 # db_to_brcount(db) |
| 1881 # |
| 1882 # Convert branch coverage data back to brcount format. |
| 1883 # |
| 1884 |
| 1885 sub db_to_brcount($) |
| 1886 { |
| 1887 my ($db) = @_; |
| 1888 my $line; |
| 1889 my $brcount = {}; |
| 1890 my $br_found = 0; |
| 1891 my $br_hit = 0; |
| 1892 |
| 1893 # Convert database back to brcount format |
| 1894 foreach $line (sort({$a <=> $b} keys(%{$db}))) { |
| 1895 my $ldata = $db->{$line}; |
| 1896 my $brdata; |
| 1897 my $block; |
| 1898 |
| 1899 foreach $block (sort({$a <=> $b} keys(%{$ldata}))) { |
| 1900 my $bdata = $ldata->{$block}; |
| 1901 my $branch; |
| 1902 |
| 1903 foreach $branch (sort({$a <=> $b} keys(%{$bdata}))) { |
| 1904 my $taken = $bdata->{$branch}; |
| 1905 |
| 1906 $br_found++; |
| 1907 $br_hit++ if ($taken ne "-" && $taken > 0); |
| 1908 $brdata = br_ivec_push($brdata, $block, |
| 1909 $branch, $taken); |
| 1910 } |
| 1911 } |
| 1912 $brcount->{$line} = $brdata; |
| 1913 } |
| 1914 |
| 1915 return ($brcount, $br_found, $br_hit); |
| 1916 } |
| 1917 |
| 1918 |
| 1919 # |
| 1920 # combine_brcount(brcount1, brcount2, type) |
| 1921 # |
| 1922 # If add is BR_ADD, add branch coverage data and return list (brcount_added, |
| 1923 # br_found, br_hit). If add is BR_SUB, subtract the taken values of brcount2 |
| 1924 # from brcount1 and return (brcount_sub, br_found, br_hit). |
| 1925 # |
| 1926 |
| 1927 sub combine_brcount($$$) |
| 1928 { |
| 1929 my ($brcount1, $brcount2, $type) = @_; |
| 1930 my $line; |
| 1931 my $block; |
| 1932 my $branch; |
| 1933 my $taken; |
| 1934 my $db; |
| 1935 my $br_found = 0; |
| 1936 my $br_hit = 0; |
| 1937 my $result; |
| 1938 |
| 1939 # Convert branches from first count to database |
| 1940 $db = brcount_to_db($brcount1); |
| 1941 # Combine values from database and second count |
| 1942 foreach $line (keys(%{$brcount2})) { |
| 1943 my $brdata = $brcount2->{$line}; |
| 1944 my $num = br_ivec_len($brdata); |
| 1945 my $i; |
| 1946 |
| 1947 for ($i = 0; $i < $num; $i++) { |
| 1948 ($block, $branch, $taken) = br_ivec_get($brdata, $i); |
| 1949 my $new_taken = $db->{$line}->{$block}->{$branch}; |
| 1950 |
| 1951 if ($type == $BR_ADD) { |
| 1952 $new_taken = br_taken_add($new_taken, $taken); |
| 1953 } elsif ($type == $BR_SUB) { |
| 1954 $new_taken = br_taken_sub($new_taken, $taken); |
| 1955 } |
| 1956 $db->{$line}->{$block}->{$branch} = $new_taken |
| 1957 if (defined($new_taken)); |
| 1958 } |
| 1959 } |
| 1960 # Convert database back to brcount format |
| 1961 ($result, $br_found, $br_hit) = db_to_brcount($db); |
| 1962 |
| 1963 return ($result, $br_found, $br_hit); |
| 1964 } |
| 1965 |
| 1966 |
| 1967 # |
| 1968 # add_testbrdata(testbrdata1, testbrdata2) |
| 1969 # |
| 1970 # Add branch coverage data for several tests. Return reference to |
| 1971 # added_testbrdata. |
| 1972 # |
| 1973 |
| 1974 sub add_testbrdata($$) |
| 1975 { |
| 1976 my ($testbrdata1, $testbrdata2) = @_; |
| 1977 my %result; |
| 1978 my $testname; |
| 1979 |
| 1980 foreach $testname (keys(%{$testbrdata1})) { |
| 1981 if (defined($testbrdata2->{$testname})) { |
| 1982 my $brcount; |
| 1983 |
| 1984 # Branch coverage data for this testname exists |
| 1985 # in both data sets: add |
| 1986 ($brcount) = combine_brcount($testbrdata1->{$testname}, |
| 1987 $testbrdata2->{$testname}, $BR_ADD); |
| 1988 $result{$testname} = $brcount; |
| 1989 next; |
| 1990 } |
| 1991 # Branch coverage data for this testname is unique to |
| 1992 # data set 1: copy |
| 1993 $result{$testname} = $testbrdata1->{$testname}; |
| 1994 } |
| 1995 |
| 1996 # Add count data for testnames unique to data set 2 |
| 1997 foreach $testname (keys(%{$testbrdata2})) { |
| 1998 if (!defined($result{$testname})) { |
| 1999 $result{$testname} = $testbrdata2->{$testname}; |
| 2000 } |
| 2001 } |
| 2002 return \%result; |
| 2003 } |
| 2004 |
| 2005 |
| 1592 # | 2006 # |
| 1593 # combine_info_entries(entry_ref1, entry_ref2, filename) | 2007 # combine_info_entries(entry_ref1, entry_ref2, filename) |
| 1594 # | 2008 # |
| 1595 # Combine .info data entry hashes referenced by ENTRY_REF1 and ENTRY_REF2. | 2009 # Combine .info data entry hashes referenced by ENTRY_REF1 and ENTRY_REF2. |
| 1596 # Return reference to resulting hash. | 2010 # Return reference to resulting hash. |
| 1597 # | 2011 # |
| 1598 | 2012 |
| 1599 sub combine_info_entries($$$) | 2013 sub combine_info_entries($$$) |
| 1600 { | 2014 { |
| 1601 my $entry1 = $_[0]; # Reference to hash containing first entry | 2015 my $entry1 = $_[0]; # Reference to hash containing first entry |
| 1602 my $testdata1; | 2016 my $testdata1; |
| 1603 my $sumcount1; | 2017 my $sumcount1; |
| 1604 my $funcdata1; | 2018 my $funcdata1; |
| 1605 my $checkdata1; | 2019 my $checkdata1; |
| 1606 my $testfncdata1; | 2020 my $testfncdata1; |
| 1607 my $sumfnccount1; | 2021 my $sumfnccount1; |
| 2022 my $testbrdata1; |
| 2023 my $sumbrcount1; |
| 1608 | 2024 |
| 1609 my $entry2 = $_[1]; # Reference to hash containing second entry | 2025 my $entry2 = $_[1]; # Reference to hash containing second entry |
| 1610 my $testdata2; | 2026 my $testdata2; |
| 1611 my $sumcount2; | 2027 my $sumcount2; |
| 1612 my $funcdata2; | 2028 my $funcdata2; |
| 1613 my $checkdata2; | 2029 my $checkdata2; |
| 1614 my $testfncdata2; | 2030 my $testfncdata2; |
| 1615 my $sumfnccount2; | 2031 my $sumfnccount2; |
| 2032 my $testbrdata2; |
| 2033 my $sumbrcount2; |
| 1616 | 2034 |
| 1617 my %result; # Hash containing combined entry | 2035 my %result; # Hash containing combined entry |
| 1618 my %result_testdata; | 2036 my %result_testdata; |
| 1619 my $result_sumcount = {}; | 2037 my $result_sumcount = {}; |
| 1620 my $result_funcdata; | 2038 my $result_funcdata; |
| 1621 my $result_testfncdata; | 2039 my $result_testfncdata; |
| 1622 my $result_sumfnccount; | 2040 my $result_sumfnccount; |
| 2041 my $result_testbrdata; |
| 2042 my $result_sumbrcount; |
| 1623 my $lines_found; | 2043 my $lines_found; |
| 1624 my $lines_hit; | 2044 my $lines_hit; |
| 1625 my $fn_found; | 2045 my $fn_found; |
| 1626 my $fn_hit; | 2046 my $fn_hit; |
| 2047 my $br_found; |
| 2048 my $br_hit; |
| 1627 | 2049 |
| 1628 my $testname; | 2050 my $testname; |
| 1629 my $filename = $_[2]; | 2051 my $filename = $_[2]; |
| 1630 | 2052 |
| 1631 # Retrieve data | 2053 # Retrieve data |
| 1632 ($testdata1, $sumcount1, $funcdata1, $checkdata1, $testfncdata1, | 2054 ($testdata1, $sumcount1, $funcdata1, $checkdata1, $testfncdata1, |
| 1633 » $sumfnccount1) = get_info_entry($entry1); | 2055 » $sumfnccount1, $testbrdata1, $sumbrcount1) = get_info_entry($entry1); |
| 1634 ($testdata2, $sumcount2, $funcdata2, $checkdata2, $testfncdata2, | 2056 ($testdata2, $sumcount2, $funcdata2, $checkdata2, $testfncdata2, |
| 1635 » $sumfnccount2) = get_info_entry($entry2); | 2057 » $sumfnccount2, $testbrdata2, $sumbrcount2) = get_info_entry($entry2); |
| 1636 | 2058 |
| 1637 # Merge checksums | 2059 # Merge checksums |
| 1638 $checkdata1 = merge_checksums($checkdata1, $checkdata2, $filename); | 2060 $checkdata1 = merge_checksums($checkdata1, $checkdata2, $filename); |
| 1639 | 2061 |
| 1640 # Combine funcdata | 2062 # Combine funcdata |
| 1641 $result_funcdata = merge_func_data($funcdata1, $funcdata2, $filename); | 2063 $result_funcdata = merge_func_data($funcdata1, $funcdata2, $filename); |
| 1642 | 2064 |
| 1643 # Combine function call count data | 2065 # Combine function call count data |
| 1644 $result_testfncdata = add_testfncdata($testfncdata1, $testfncdata2); | 2066 $result_testfncdata = add_testfncdata($testfncdata1, $testfncdata2); |
| 1645 ($result_sumfnccount, $fn_found, $fn_hit) = | 2067 ($result_sumfnccount, $fn_found, $fn_hit) = |
| 1646 add_fnccount($sumfnccount1, $sumfnccount2); | 2068 add_fnccount($sumfnccount1, $sumfnccount2); |
| 1647 | 2069 |
| 2070 # Combine branch coverage data |
| 2071 $result_testbrdata = add_testbrdata($testbrdata1, $testbrdata2); |
| 2072 ($result_sumbrcount, $br_found, $br_hit) = |
| 2073 combine_brcount($sumbrcount1, $sumbrcount2, $BR_ADD); |
| 2074 |
| 1648 # Combine testdata | 2075 # Combine testdata |
| 1649 foreach $testname (keys(%{$testdata1})) | 2076 foreach $testname (keys(%{$testdata1})) |
| 1650 { | 2077 { |
| 1651 if (defined($testdata2->{$testname})) | 2078 if (defined($testdata2->{$testname})) |
| 1652 { | 2079 { |
| 1653 # testname is present in both entries, requires | 2080 # testname is present in both entries, requires |
| 1654 # combination | 2081 # combination |
| 1655 ($result_testdata{$testname}) = | 2082 ($result_testdata{$testname}) = |
| 1656 add_counts($testdata1->{$testname}, | 2083 add_counts($testdata1->{$testname}, |
| 1657 $testdata2->{$testname}); | 2084 $testdata2->{$testname}); |
| (...skipping 22 matching lines...) Expand all Loading... |
| 1680 ($result_sumcount, $lines_found, $lines_hit) = | 2107 ($result_sumcount, $lines_found, $lines_hit) = |
| 1681 add_counts($result_sumcount, | 2108 add_counts($result_sumcount, |
| 1682 $result_testdata{$testname}); | 2109 $result_testdata{$testname}); |
| 1683 } | 2110 } |
| 1684 | 2111 |
| 1685 # Calculate resulting sumcount | 2112 # Calculate resulting sumcount |
| 1686 | 2113 |
| 1687 # Store result | 2114 # Store result |
| 1688 set_info_entry(\%result, \%result_testdata, $result_sumcount, | 2115 set_info_entry(\%result, \%result_testdata, $result_sumcount, |
| 1689 $result_funcdata, $checkdata1, $result_testfncdata, | 2116 $result_funcdata, $checkdata1, $result_testfncdata, |
| 1690 » » $result_sumfnccount, $lines_found, $lines_hit, | 2117 » » $result_sumfnccount, $result_testbrdata, |
| 1691 » » $fn_found, $fn_hit); | 2118 » » $result_sumbrcount, $lines_found, $lines_hit, |
| 2119 » » $fn_found, $fn_hit, $br_found, $br_hit); |
| 1692 | 2120 |
| 1693 return(\%result); | 2121 return(\%result); |
| 1694 } | 2122 } |
| 1695 | 2123 |
| 1696 | 2124 |
| 1697 # | 2125 # |
| 1698 # combine_info_files(info_ref1, info_ref2) | 2126 # combine_info_files(info_ref1, info_ref2) |
| 1699 # | 2127 # |
| 1700 # Combine .info data in hashes referenced by INFO_REF1 and INFO_REF2. Return | 2128 # Combine .info data in hashes referenced by INFO_REF1 and INFO_REF2. Return |
| 1701 # reference to resulting hash. | 2129 # reference to resulting hash. |
| (...skipping 21 matching lines...) Expand all Loading... |
| 1723 # resulting hash | 2151 # resulting hash |
| 1724 $hash1{$filename} = $hash2{$filename}; | 2152 $hash1{$filename} = $hash2{$filename}; |
| 1725 } | 2153 } |
| 1726 } | 2154 } |
| 1727 | 2155 |
| 1728 return(\%hash1); | 2156 return(\%hash1); |
| 1729 } | 2157 } |
| 1730 | 2158 |
| 1731 | 2159 |
| 1732 # | 2160 # |
| 1733 # get_prefix(filename_list) | 2161 # get_prefix(min_dir, filename_list) |
| 1734 # | 2162 # |
| 1735 # Search FILENAME_LIST for a directory prefix which is common to as many | 2163 # Search FILENAME_LIST for a directory prefix which is common to as many |
| 1736 # list entries as possible, so that removing this prefix will minimize the | 2164 # list entries as possible, so that removing this prefix will minimize the |
| 1737 # sum of the lengths of all resulting shortened filenames. | 2165 # sum of the lengths of all resulting shortened filenames while observing |
| 2166 # that no filename has less than MIN_DIR parent directories. |
| 1738 # | 2167 # |
| 1739 | 2168 |
| 1740 sub get_prefix(@) | 2169 sub get_prefix($@) |
| 1741 { | 2170 { |
| 1742 » my @filename_list = @_;»» # provided list of filenames | 2171 » my ($min_dir, @filename_list) = @_; |
| 1743 my %prefix; # mapping: prefix -> sum of lengths | 2172 my %prefix; # mapping: prefix -> sum of lengths |
| 1744 my $current; # Temporary iteration variable | 2173 my $current; # Temporary iteration variable |
| 1745 | 2174 |
| 1746 # Find list of prefixes | 2175 # Find list of prefixes |
| 1747 foreach (@filename_list) | 2176 foreach (@filename_list) |
| 1748 { | 2177 { |
| 1749 # Need explicit assignment to get a copy of $_ so that | 2178 # Need explicit assignment to get a copy of $_ so that |
| 1750 # shortening the contained prefix does not affect the list | 2179 # shortening the contained prefix does not affect the list |
| 1751 » » $current = shorten_prefix($_); | 2180 » » $current = $_; |
| 1752 while ($current = shorten_prefix($current)) | 2181 while ($current = shorten_prefix($current)) |
| 1753 { | 2182 { |
| 2183 $current .= "/"; |
| 2184 |
| 1754 # Skip rest if the remaining prefix has already been | 2185 # Skip rest if the remaining prefix has already been |
| 1755 # added to hash | 2186 # added to hash |
| 1756 » » » if ($prefix{$current}) { last; } | 2187 » » » if (exists($prefix{$current})) { last; } |
| 1757 | 2188 |
| 1758 # Initialize with 0 | 2189 # Initialize with 0 |
| 1759 $prefix{$current}="0"; | 2190 $prefix{$current}="0"; |
| 1760 } | 2191 } |
| 1761 | 2192 |
| 1762 } | 2193 } |
| 1763 | 2194 |
| 2195 # Remove all prefixes that would cause filenames to have less than |
| 2196 # the minimum number of parent directories |
| 2197 foreach my $filename (@filename_list) { |
| 2198 my $dir = dirname($filename); |
| 2199 |
| 2200 for (my $i = 0; $i < $min_dir; $i++) { |
| 2201 delete($prefix{$dir."/"}); |
| 2202 $dir = shorten_prefix($dir); |
| 2203 } |
| 2204 } |
| 2205 |
| 2206 # Check if any prefix remains |
| 2207 return undef if (!%prefix); |
| 2208 |
| 1764 # Calculate sum of lengths for all prefixes | 2209 # Calculate sum of lengths for all prefixes |
| 1765 foreach $current (keys(%prefix)) | 2210 foreach $current (keys(%prefix)) |
| 1766 { | 2211 { |
| 1767 foreach (@filename_list) | 2212 foreach (@filename_list) |
| 1768 { | 2213 { |
| 1769 # Add original length | 2214 # Add original length |
| 1770 $prefix{$current} += length($_); | 2215 $prefix{$current} += length($_); |
| 1771 | 2216 |
| 1772 # Check whether prefix matches | 2217 # Check whether prefix matches |
| 1773 if (substr($_, 0, length($current)) eq $current) | 2218 if (substr($_, 0, length($current)) eq $current) |
| 1774 { | 2219 { |
| 1775 # Subtract prefix length for this filename | 2220 # Subtract prefix length for this filename |
| 1776 $prefix{$current} -= length($current); | 2221 $prefix{$current} -= length($current); |
| 1777 } | 2222 } |
| 1778 } | 2223 } |
| 1779 } | 2224 } |
| 1780 | 2225 |
| 1781 # Find and return prefix with minimal sum | 2226 # Find and return prefix with minimal sum |
| 1782 $current = (keys(%prefix))[0]; | 2227 $current = (keys(%prefix))[0]; |
| 1783 | 2228 |
| 1784 foreach (keys(%prefix)) | 2229 foreach (keys(%prefix)) |
| 1785 { | 2230 { |
| 1786 if ($prefix{$_} < $prefix{$current}) | 2231 if ($prefix{$_} < $prefix{$current}) |
| 1787 { | 2232 { |
| 1788 $current = $_; | 2233 $current = $_; |
| 1789 } | 2234 } |
| 1790 } | 2235 } |
| 1791 | 2236 |
| 2237 $current =~ s/\/$//; |
| 2238 |
| 1792 return($current); | 2239 return($current); |
| 1793 } | 2240 } |
| 1794 | 2241 |
| 1795 | 2242 |
| 1796 # | 2243 # |
| 1797 # shorten_prefix(prefix) | 2244 # shorten_prefix(prefix) |
| 1798 # | 2245 # |
| 1799 # Return PREFIX shortened by last directory component. | 2246 # Return PREFIX shortened by last directory component. |
| 1800 # | 2247 # |
| 1801 | 2248 |
| (...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1873 # Die on error. | 2320 # Die on error. |
| 1874 # | 2321 # |
| 1875 | 2322 |
| 1876 sub read_testfile($) | 2323 sub read_testfile($) |
| 1877 { | 2324 { |
| 1878 my %result; | 2325 my %result; |
| 1879 my $test_name; | 2326 my $test_name; |
| 1880 my $changed_testname; | 2327 my $changed_testname; |
| 1881 local *TEST_HANDLE; | 2328 local *TEST_HANDLE; |
| 1882 | 2329 |
| 1883 » open(TEST_HANDLE, "<".$_[0]) | 2330 » open(TEST_HANDLE, "<", $_[0]) |
| 1884 or die("ERROR: cannot open $_[0]!\n"); | 2331 or die("ERROR: cannot open $_[0]!\n"); |
| 1885 | 2332 |
| 1886 while (<TEST_HANDLE>) | 2333 while (<TEST_HANDLE>) |
| 1887 { | 2334 { |
| 1888 chomp($_); | 2335 chomp($_); |
| 1889 | 2336 |
| 1890 # Match lines beginning with TN:<whitespace(s)> | 2337 # Match lines beginning with TN:<whitespace(s)> |
| 1891 if (/^TN:\s+(.*?)\s*$/) | 2338 if (/^TN:\s+(.*?)\s*$/) |
| 1892 { | 2339 { |
| 1893 # Store name for later use | 2340 # Store name for later use |
| (...skipping 85 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1979 # create_sub_dir(dir_name) | 2426 # create_sub_dir(dir_name) |
| 1980 # | 2427 # |
| 1981 # Create subdirectory DIR_NAME if it does not already exist, including all its | 2428 # Create subdirectory DIR_NAME if it does not already exist, including all its |
| 1982 # parent directories. | 2429 # parent directories. |
| 1983 # | 2430 # |
| 1984 # Die on error. | 2431 # Die on error. |
| 1985 # | 2432 # |
| 1986 | 2433 |
| 1987 sub create_sub_dir($) | 2434 sub create_sub_dir($) |
| 1988 { | 2435 { |
| 1989 » system("mkdir", "-p" ,$_[0]) | 2436 » my ($dir) = @_; |
| 1990 » » and die("ERROR: cannot create directory $_!\n"); | 2437 |
| 2438 » system("mkdir", "-p" ,$dir) |
| 2439 » » and die("ERROR: cannot create directory $dir!\n"); |
| 1991 } | 2440 } |
| 1992 | 2441 |
| 1993 | 2442 |
| 1994 # | 2443 # |
| 1995 # write_description_file(descriptions, overall_found, overall_hit, | 2444 # write_description_file(descriptions, overall_found, overall_hit, |
| 1996 # total_fn_found, total_fn_hit) | 2445 # total_fn_found, total_fn_hit, total_br_found, |
| 2446 # total_br_hit) |
| 1997 # | 2447 # |
| 1998 # Write HTML file containing all test case descriptions. DESCRIPTIONS is a | 2448 # Write HTML file containing all test case descriptions. DESCRIPTIONS is a |
| 1999 # reference to a hash containing a mapping | 2449 # reference to a hash containing a mapping |
| 2000 # | 2450 # |
| 2001 # test case name -> test case description | 2451 # test case name -> test case description |
| 2002 # | 2452 # |
| 2003 # Die on error. | 2453 # Die on error. |
| 2004 # | 2454 # |
| 2005 | 2455 |
| 2006 sub write_description_file($$$$$) | 2456 sub write_description_file($$$$$$$) |
| 2007 { | 2457 { |
| 2008 my %description = %{$_[0]}; | 2458 my %description = %{$_[0]}; |
| 2009 my $found = $_[1]; | 2459 my $found = $_[1]; |
| 2010 my $hit = $_[2]; | 2460 my $hit = $_[2]; |
| 2011 my $fn_found = $_[3]; | 2461 my $fn_found = $_[3]; |
| 2012 my $fn_hit = $_[4]; | 2462 my $fn_hit = $_[4]; |
| 2463 my $br_found = $_[5]; |
| 2464 my $br_hit = $_[6]; |
| 2013 my $test_name; | 2465 my $test_name; |
| 2014 local *HTML_HANDLE; | 2466 local *HTML_HANDLE; |
| 2015 | 2467 |
| 2016 html_create(*HTML_HANDLE,"descriptions.$html_ext"); | 2468 html_create(*HTML_HANDLE,"descriptions.$html_ext"); |
| 2017 write_html_prolog(*HTML_HANDLE, "", "LCOV - test case descriptions"); | 2469 write_html_prolog(*HTML_HANDLE, "", "LCOV - test case descriptions"); |
| 2018 write_header(*HTML_HANDLE, 3, "", "", $found, $hit, $fn_found, | 2470 write_header(*HTML_HANDLE, 3, "", "", $found, $hit, $fn_found, |
| 2019 » » $fn_hit, 0); | 2471 » » $fn_hit, $br_found, $br_hit, 0); |
| 2020 | 2472 |
| 2021 write_test_table_prolog(*HTML_HANDLE, | 2473 write_test_table_prolog(*HTML_HANDLE, |
| 2022 "Test case descriptions - alphabetical list"); | 2474 "Test case descriptions - alphabetical list"); |
| 2023 | 2475 |
| 2024 foreach $test_name (sort(keys(%description))) | 2476 foreach $test_name (sort(keys(%description))) |
| 2025 { | 2477 { |
| 2026 write_test_table_entry(*HTML_HANDLE, $test_name, | 2478 write_test_table_entry(*HTML_HANDLE, $test_name, |
| 2027 escape_html($description{$test_name})); | 2479 escape_html($description{$test_name})); |
| 2028 } | 2480 } |
| 2029 | 2481 |
| (...skipping 109 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2139 0x54, 0x28, 0xcf, 0x63, 0x60, 0x40, 0x03, 0xff, 0xa1, 0x00, | 2591 0x54, 0x28, 0xcf, 0x63, 0x60, 0x40, 0x03, 0xff, 0xa1, 0x00, |
| 2140 0x5d, 0x9c, 0x11, 0x5d, 0x11, 0x8a, 0x24, 0x23, 0x23, 0x23, | 2592 0x5d, 0x9c, 0x11, 0x5d, 0x11, 0x8a, 0x24, 0x23, 0x23, 0x23, |
| 2141 0x86, 0x42, 0x6c, 0xa6, 0x20, 0x2b, 0x66, 0xc4, 0xa7, 0x08, | 2593 0x86, 0x42, 0x6c, 0xa6, 0x20, 0x2b, 0x66, 0xc4, 0xa7, 0x08, |
| 2142 0x59, 0x31, 0x23, 0x21, 0x45, 0x30, 0xc0, 0xc4, 0x30, 0x60, | 2594 0x59, 0x31, 0x23, 0x21, 0x45, 0x30, 0xc0, 0xc4, 0x30, 0x60, |
| 2143 0x80, 0xfa, 0x6e, 0x24, 0x3e, 0x78, 0x48, 0x0a, 0x70, 0x62, | 2595 0x80, 0xfa, 0x6e, 0x24, 0x3e, 0x78, 0x48, 0x0a, 0x70, 0x62, |
| 2144 0xa2, 0x90, 0x81, 0xd8, 0x44, 0x01, 0x00, 0xe9, 0x5c, 0x2f, | 2596 0xa2, 0x90, 0x81, 0xd8, 0x44, 0x01, 0x00, 0xe9, 0x5c, 0x2f, |
| 2145 0xf5, 0xe2, 0x9d, 0x0f, 0xf9, 0x00, 0x00, 0x00, 0x00, 0x49, | 2597 0xf5, 0xe2, 0x9d, 0x0f, 0xf9, 0x00, 0x00, 0x00, 0x00, 0x49, |
| 2146 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, 0x82] if ($sort); | 2598 0x45, 0x4e, 0x44, 0xae, 0x42, 0x60, 0x82] if ($sort); |
| 2147 foreach (keys(%data)) | 2599 foreach (keys(%data)) |
| 2148 { | 2600 { |
| 2149 » » open(PNG_HANDLE, ">".$_) | 2601 » » open(PNG_HANDLE, ">", $_) |
| 2150 or die("ERROR: cannot create $_!\n"); | 2602 or die("ERROR: cannot create $_!\n"); |
| 2151 binmode(PNG_HANDLE); | 2603 binmode(PNG_HANDLE); |
| 2152 print(PNG_HANDLE map(chr,@{$data{$_}})); | 2604 print(PNG_HANDLE map(chr,@{$data{$_}})); |
| 2153 close(PNG_HANDLE); | 2605 close(PNG_HANDLE); |
| 2154 } | 2606 } |
| 2155 } | 2607 } |
| 2156 | 2608 |
| 2157 | 2609 |
| 2158 # | 2610 # |
| 2159 # write_htaccess_file() | 2611 # write_htaccess_file() |
| 2160 # | 2612 # |
| 2161 | 2613 |
| 2162 sub write_htaccess_file() | 2614 sub write_htaccess_file() |
| 2163 { | 2615 { |
| 2164 local *HTACCESS_HANDLE; | 2616 local *HTACCESS_HANDLE; |
| 2165 my $htaccess_data; | 2617 my $htaccess_data; |
| 2166 | 2618 |
| 2167 » open(*HTACCESS_HANDLE, ">.htaccess") | 2619 » open(*HTACCESS_HANDLE, ">", ".htaccess") |
| 2168 or die("ERROR: cannot open .htaccess for writing!\n"); | 2620 or die("ERROR: cannot open .htaccess for writing!\n"); |
| 2169 | 2621 |
| 2170 $htaccess_data = (<<"END_OF_HTACCESS") | 2622 $htaccess_data = (<<"END_OF_HTACCESS") |
| 2171 AddEncoding x-gzip .html | 2623 AddEncoding x-gzip .html |
| 2172 END_OF_HTACCESS | 2624 END_OF_HTACCESS |
| 2173 ; | 2625 ; |
| 2174 | 2626 |
| 2175 print(HTACCESS_HANDLE $htaccess_data); | 2627 print(HTACCESS_HANDLE $htaccess_data); |
| 2176 close(*HTACCESS_HANDLE); | 2628 close(*HTACCESS_HANDLE); |
| 2177 } | 2629 } |
| (...skipping 12 matching lines...) Expand all Loading... |
| 2190 | 2642 |
| 2191 # Check for a specified external style sheet file | 2643 # Check for a specified external style sheet file |
| 2192 if ($css_filename) | 2644 if ($css_filename) |
| 2193 { | 2645 { |
| 2194 # Simply copy that file | 2646 # Simply copy that file |
| 2195 system("cp", $css_filename, "gcov.css") | 2647 system("cp", $css_filename, "gcov.css") |
| 2196 and die("ERROR: cannot copy file $css_filename!\n"); | 2648 and die("ERROR: cannot copy file $css_filename!\n"); |
| 2197 return; | 2649 return; |
| 2198 } | 2650 } |
| 2199 | 2651 |
| 2200 » open(CSS_HANDLE, ">gcov.css") | 2652 » open(CSS_HANDLE, ">", "gcov.css") |
| 2201 or die ("ERROR: cannot open gcov.css for writing!\n"); | 2653 or die ("ERROR: cannot open gcov.css for writing!\n"); |
| 2202 | 2654 |
| 2203 | 2655 |
| 2204 # ************************************************************* | 2656 # ************************************************************* |
| 2205 | 2657 |
| 2206 my $css_data = ($_=<<"END_OF_CSS") | 2658 my $css_data = ($_=<<"END_OF_CSS") |
| 2207 /* All views: initial background and text color */ | 2659 /* All views: initial background and text color */ |
| 2208 body | 2660 body |
| 2209 { | 2661 { |
| 2210 color: #000000; | 2662 color: #000000; |
| (...skipping 110 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2321 text-align: right; | 2773 text-align: right; |
| 2322 color: #000000; | 2774 color: #000000; |
| 2323 font-family: sans-serif; | 2775 font-family: sans-serif; |
| 2324 font-weight: bold; | 2776 font-weight: bold; |
| 2325 white-space: nowrap; | 2777 white-space: nowrap; |
| 2326 padding-left: 12px; | 2778 padding-left: 12px; |
| 2327 padding-right: 4px; | 2779 padding-right: 4px; |
| 2328 background-color: #FF0000; | 2780 background-color: #FF0000; |
| 2329 } | 2781 } |
| 2330 | 2782 |
| 2331 /* All views: header legend item for legend entry */ | |
| 2332 td.headerItemLeg | |
| 2333 { | |
| 2334 text-align: right; | |
| 2335 padding-right: 6px; | |
| 2336 font-family: sans-serif; | |
| 2337 font-weight: bold; | |
| 2338 vertical-align: bottom; | |
| 2339 white-space: nowrap; | |
| 2340 } | |
| 2341 | |
| 2342 /* All views: header legend value for legend entry */ | 2783 /* All views: header legend value for legend entry */ |
| 2343 td.headerValueLeg | 2784 td.headerValueLeg |
| 2344 { | 2785 { |
| 2345 text-align: left; | 2786 text-align: left; |
| 2346 color: #000000; | 2787 color: #000000; |
| 2347 font-family: sans-serif; | 2788 font-family: sans-serif; |
| 2348 font-size: 80%; | 2789 font-size: 80%; |
| 2349 white-space: nowrap; | 2790 white-space: nowrap; |
| 2350 padding-top: 4px; | 2791 padding-top: 4px; |
| 2351 } | 2792 } |
| (...skipping 60 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2412 | 2853 |
| 2413 /* Directory view/File view (all): percentage entry for files with | 2854 /* Directory view/File view (all): percentage entry for files with |
| 2414 high coverage rate */ | 2855 high coverage rate */ |
| 2415 td.coverPerHi | 2856 td.coverPerHi |
| 2416 { | 2857 { |
| 2417 text-align: right; | 2858 text-align: right; |
| 2418 padding-left: 10px; | 2859 padding-left: 10px; |
| 2419 padding-right: 10px; | 2860 padding-right: 10px; |
| 2420 background-color: #A7FC9D; | 2861 background-color: #A7FC9D; |
| 2421 font-weight: bold; | 2862 font-weight: bold; |
| 2863 font-family: sans-serif; |
| 2422 } | 2864 } |
| 2423 | 2865 |
| 2424 /* Directory view/File view (all): line count entry for files with | 2866 /* Directory view/File view (all): line count entry for files with |
| 2425 high coverage rate */ | 2867 high coverage rate */ |
| 2426 td.coverNumHi | 2868 td.coverNumHi |
| 2427 { | 2869 { |
| 2428 text-align: right; | 2870 text-align: right; |
| 2429 padding-left: 10px; | 2871 padding-left: 10px; |
| 2430 padding-right: 10px; | 2872 padding-right: 10px; |
| 2431 background-color: #A7FC9D; | 2873 background-color: #A7FC9D; |
| 2432 white-space: nowrap; | 2874 white-space: nowrap; |
| 2433 » } | 2875 » font-family: sans-serif; |
| 2434 » | |
| 2435 » /* Directory view/File view (all): legend entry for high coverage | |
| 2436 » rate */ | |
| 2437 » span.coverLegendHi | |
| 2438 » { | |
| 2439 » padding-left: 10px; | |
| 2440 » padding-right: 10px; | |
| 2441 » padding-bottom: 2px; | |
| 2442 » background-color: #A7FC9D; | |
| 2443 } | 2876 } |
| 2444 | 2877 |
| 2445 /* Directory view/File view (all): percentage entry for files with | 2878 /* Directory view/File view (all): percentage entry for files with |
| 2446 medium coverage rate */ | 2879 medium coverage rate */ |
| 2447 td.coverPerMed | 2880 td.coverPerMed |
| 2448 { | 2881 { |
| 2449 text-align: right; | 2882 text-align: right; |
| 2450 padding-left: 10px; | 2883 padding-left: 10px; |
| 2451 padding-right: 10px; | 2884 padding-right: 10px; |
| 2452 background-color: #FFEA20; | 2885 background-color: #FFEA20; |
| 2453 font-weight: bold; | 2886 font-weight: bold; |
| 2887 font-family: sans-serif; |
| 2454 } | 2888 } |
| 2455 | 2889 |
| 2456 /* Directory view/File view (all): line count entry for files with | 2890 /* Directory view/File view (all): line count entry for files with |
| 2457 medium coverage rate */ | 2891 medium coverage rate */ |
| 2458 td.coverNumMed | 2892 td.coverNumMed |
| 2459 { | 2893 { |
| 2460 text-align: right; | 2894 text-align: right; |
| 2461 padding-left: 10px; | 2895 padding-left: 10px; |
| 2462 padding-right: 10px; | 2896 padding-right: 10px; |
| 2463 background-color: #FFEA20; | 2897 background-color: #FFEA20; |
| 2464 white-space: nowrap; | 2898 white-space: nowrap; |
| 2465 » } | 2899 » font-family: sans-serif; |
| 2466 » | |
| 2467 » /* Directory view/File view (all): legend entry for medium coverage | |
| 2468 » rate */ | |
| 2469 » span.coverLegendMed | |
| 2470 » { | |
| 2471 » padding-left: 10px; | |
| 2472 » padding-right: 10px; | |
| 2473 » padding-bottom: 2px; | |
| 2474 » background-color: #FFEA20; | |
| 2475 } | 2900 } |
| 2476 | 2901 |
| 2477 /* Directory view/File view (all): percentage entry for files with | 2902 /* Directory view/File view (all): percentage entry for files with |
| 2478 low coverage rate */ | 2903 low coverage rate */ |
| 2479 td.coverPerLo | 2904 td.coverPerLo |
| 2480 { | 2905 { |
| 2481 text-align: right; | 2906 text-align: right; |
| 2482 padding-left: 10px; | 2907 padding-left: 10px; |
| 2483 padding-right: 10px; | 2908 padding-right: 10px; |
| 2484 background-color: #FF0000; | 2909 background-color: #FF0000; |
| 2485 font-weight: bold; | 2910 font-weight: bold; |
| 2911 font-family: sans-serif; |
| 2486 } | 2912 } |
| 2487 | 2913 |
| 2488 /* Directory view/File view (all): line count entry for files with | 2914 /* Directory view/File view (all): line count entry for files with |
| 2489 low coverage rate */ | 2915 low coverage rate */ |
| 2490 td.coverNumLo | 2916 td.coverNumLo |
| 2491 { | 2917 { |
| 2492 text-align: right; | 2918 text-align: right; |
| 2493 padding-left: 10px; | 2919 padding-left: 10px; |
| 2494 padding-right: 10px; | 2920 padding-right: 10px; |
| 2495 background-color: #FF0000; | 2921 background-color: #FF0000; |
| 2496 white-space: nowrap; | 2922 white-space: nowrap; |
| 2497 » } | 2923 » font-family: sans-serif; |
| 2498 » | |
| 2499 » /* Directory view/File view (all): legend entry for low coverage | |
| 2500 » rate */ | |
| 2501 » span.coverLegendLo | |
| 2502 » { | |
| 2503 » padding-left: 10px; | |
| 2504 » padding-right: 10px; | |
| 2505 » padding-bottom: 2px; | |
| 2506 » background-color: #FF0000; | |
| 2507 } | 2924 } |
| 2508 | 2925 |
| 2509 /* File view (all): "show/hide details" link format */ | 2926 /* File view (all): "show/hide details" link format */ |
| 2510 a.detail:link | 2927 a.detail:link |
| 2511 { | 2928 { |
| 2512 color: #B8D0FF; | 2929 color: #B8D0FF; |
| 2930 font-size:80%; |
| 2513 } | 2931 } |
| 2514 | 2932 |
| 2515 /* File view (all): "show/hide details" link - visited format */ | 2933 /* File view (all): "show/hide details" link - visited format */ |
| 2516 a.detail:visited | 2934 a.detail:visited |
| 2517 { | 2935 { |
| 2518 color: #B8D0FF; | 2936 color: #B8D0FF; |
| 2937 font-size:80%; |
| 2519 } | 2938 } |
| 2520 | 2939 |
| 2521 /* File view (all): "show/hide details" link - activated format */ | 2940 /* File view (all): "show/hide details" link - activated format */ |
| 2522 a.detail:active | 2941 a.detail:active |
| 2523 { | 2942 { |
| 2524 color: #FFFFFF; | 2943 color: #FFFFFF; |
| 2525 » } | 2944 » font-size:80%; |
| 2526 » | |
| 2527 » /* File view (detail): test name table headline format */ | |
| 2528 » td.testNameHead | |
| 2529 » { | |
| 2530 » text-align: right; | |
| 2531 » padding-right: 10px; | |
| 2532 » background-color: #DAE7FE; | |
| 2533 » font-family: sans-serif; | |
| 2534 » font-weight: bold; | |
| 2535 » } | |
| 2536 » | |
| 2537 » /* File view (detail): test lines table headline format */ | |
| 2538 » td.testLinesHead | |
| 2539 » { | |
| 2540 » text-align: center; | |
| 2541 » background-color: #DAE7FE; | |
| 2542 » font-family: sans-serif; | |
| 2543 » font-weight: bold; | |
| 2544 } | 2945 } |
| 2545 | 2946 |
| 2546 /* File view (detail): test name entry */ | 2947 /* File view (detail): test name entry */ |
| 2547 td.testName | 2948 td.testName |
| 2548 { | 2949 { |
| 2549 text-align: right; | 2950 text-align: right; |
| 2550 padding-right: 10px; | 2951 padding-right: 10px; |
| 2551 background-color: #DAE7FE; | 2952 background-color: #DAE7FE; |
| 2953 font-family: sans-serif; |
| 2552 } | 2954 } |
| 2553 | 2955 |
| 2554 /* File view (detail): test percentage entry */ | 2956 /* File view (detail): test percentage entry */ |
| 2555 td.testPer | 2957 td.testPer |
| 2556 { | 2958 { |
| 2557 text-align: right; | 2959 text-align: right; |
| 2558 padding-left: 10px; | 2960 padding-left: 10px; |
| 2559 padding-right: 10px; | 2961 padding-right: 10px; |
| 2560 background-color: #DAE7FE; | 2962 background-color: #DAE7FE; |
| 2963 font-family: sans-serif; |
| 2561 } | 2964 } |
| 2562 | 2965 |
| 2563 /* File view (detail): test lines count entry */ | 2966 /* File view (detail): test lines count entry */ |
| 2564 td.testNum | 2967 td.testNum |
| 2565 { | 2968 { |
| 2566 text-align: right; | 2969 text-align: right; |
| 2567 padding-left: 10px; | 2970 padding-left: 10px; |
| 2568 padding-right: 10px; | 2971 padding-right: 10px; |
| 2569 background-color: #DAE7FE; | 2972 background-color: #DAE7FE; |
| 2973 font-family: sans-serif; |
| 2570 } | 2974 } |
| 2571 | 2975 |
| 2572 /* Test case descriptions: test name format*/ | 2976 /* Test case descriptions: test name format*/ |
| 2573 dt | 2977 dt |
| 2574 { | 2978 { |
| 2575 font-family: sans-serif; | 2979 font-family: sans-serif; |
| 2576 font-weight: bold; | 2980 font-weight: bold; |
| 2577 } | 2981 } |
| 2578 | 2982 |
| 2579 /* Test case descriptions: description table body */ | 2983 /* Test case descriptions: description table body */ |
| (...skipping 18 matching lines...) Expand all Loading... |
| 2598 } | 3002 } |
| 2599 | 3003 |
| 2600 /* Source code view: function entry zero count*/ | 3004 /* Source code view: function entry zero count*/ |
| 2601 td.coverFnLo | 3005 td.coverFnLo |
| 2602 { | 3006 { |
| 2603 text-align: right; | 3007 text-align: right; |
| 2604 padding-left: 10px; | 3008 padding-left: 10px; |
| 2605 padding-right: 10px; | 3009 padding-right: 10px; |
| 2606 background-color: #FF0000; | 3010 background-color: #FF0000; |
| 2607 font-weight: bold; | 3011 font-weight: bold; |
| 3012 font-family: sans-serif; |
| 2608 } | 3013 } |
| 2609 | 3014 |
| 2610 /* Source code view: function entry nonzero count*/ | 3015 /* Source code view: function entry nonzero count*/ |
| 2611 td.coverFnHi | 3016 td.coverFnHi |
| 2612 { | 3017 { |
| 2613 text-align: right; | 3018 text-align: right; |
| 2614 padding-left: 10px; | 3019 padding-left: 10px; |
| 2615 padding-right: 10px; | 3020 padding-right: 10px; |
| 2616 background-color: #DAE7FE; | 3021 background-color: #DAE7FE; |
| 2617 font-weight: bold; | 3022 font-weight: bold; |
| 3023 font-family: sans-serif; |
| 2618 } | 3024 } |
| 2619 | 3025 |
| 2620 /* Source code view: source code format */ | 3026 /* Source code view: source code format */ |
| 2621 /* Source code view: source code format */ | |
| 2622 pre.source | 3027 pre.source |
| 2623 { | 3028 { |
| 2624 font-family: monospace; | 3029 font-family: monospace; |
| 2625 white-space: pre; | 3030 white-space: pre; |
| 3031 margin-top: 2px; |
| 2626 } | 3032 } |
| 2627 | 3033 |
| 2628 /* Source code view: line number format */ | 3034 /* Source code view: line number format */ |
| 2629 span.lineNum | 3035 span.lineNum |
| 2630 { | 3036 { |
| 2631 background-color: #EFE383; | 3037 background-color: #EFE383; |
| 2632 } | 3038 } |
| 2633 | 3039 |
| 2634 /* Source code view: format for lines which were executed */ | 3040 /* Source code view: format for lines which were executed */ |
| 2635 td.lineCov, | 3041 td.lineCov, |
| (...skipping 17 matching lines...) Expand all Loading... |
| 2653 { | 3059 { |
| 2654 background-color: #FF6230; | 3060 background-color: #FF6230; |
| 2655 } | 3061 } |
| 2656 | 3062 |
| 2657 /* Source code view: format for NoCov legend */ | 3063 /* Source code view: format for NoCov legend */ |
| 2658 span.coverLegendNoCov | 3064 span.coverLegendNoCov |
| 2659 { | 3065 { |
| 2660 padding-left: 10px; | 3066 padding-left: 10px; |
| 2661 padding-right: 10px; | 3067 padding-right: 10px; |
| 2662 padding-bottom: 2px; | 3068 padding-bottom: 2px; |
| 2663 » background-color: #FF0000; | 3069 » background-color: #FF6230; |
| 2664 } | 3070 } |
| 2665 | 3071 |
| 2666 /* Source code view (function table): standard link - visited format */ | 3072 /* Source code view (function table): standard link - visited format */ |
| 2667 td.lineNoCov > a:visited, | 3073 td.lineNoCov > a:visited, |
| 2668 td.lineCov > a:visited | 3074 td.lineCov > a:visited |
| 2669 { | 3075 { |
| 2670 color: black; | 3076 color: black; |
| 2671 text-decoration: underline; | 3077 text-decoration: underline; |
| 2672 } | 3078 } |
| 2673 | 3079 |
| 2674 /* Source code view: format for lines which were executed only in a | 3080 /* Source code view: format for lines which were executed only in a |
| 2675 previous version */ | 3081 previous version */ |
| 2676 span.lineDiffCov | 3082 span.lineDiffCov |
| 2677 { | 3083 { |
| 2678 background-color: #B5F7AF; | 3084 background-color: #B5F7AF; |
| 2679 } | 3085 } |
| 2680 | 3086 |
| 2681 » /* Source code view: format for DiffCov legend */ | 3087 » /* Source code view: format for branches which were executed |
| 2682 » span.LegendDiffCov | 3088 » * and taken */ |
| 3089 » span.branchCov |
| 2683 { | 3090 { |
| 3091 background-color: #CAD7FE; |
| 3092 } |
| 3093 |
| 3094 /* Source code view: format for branches which were executed |
| 3095 * but not taken */ |
| 3096 span.branchNoCov |
| 3097 { |
| 3098 background-color: #FF6230; |
| 3099 } |
| 3100 |
| 3101 /* Source code view: format for branches which were not executed */ |
| 3102 span.branchNoExec |
| 3103 { |
| 3104 background-color: #FF6230; |
| 3105 } |
| 3106 |
| 3107 /* Source code view: format for the source code heading line */ |
| 3108 pre.sourceHeading |
| 3109 { |
| 3110 white-space: pre; |
| 3111 font-family: monospace; |
| 3112 font-weight: bold; |
| 3113 margin: 0px; |
| 3114 } |
| 3115 |
| 3116 /* All views: header legend value for low rate */ |
| 3117 td.headerValueLegL |
| 3118 { |
| 3119 font-family: sans-serif; |
| 2684 text-align: center; | 3120 text-align: center; |
| 3121 white-space: nowrap; |
| 3122 padding-left: 4px; |
| 3123 padding-right: 2px; |
| 3124 background-color: #FF0000; |
| 3125 font-size: 80%; |
| 3126 } |
| 3127 |
| 3128 /* All views: header legend value for med rate */ |
| 3129 td.headerValueLegM |
| 3130 { |
| 3131 font-family: sans-serif; |
| 3132 text-align: center; |
| 3133 white-space: nowrap; |
| 3134 padding-left: 2px; |
| 3135 padding-right: 2px; |
| 3136 background-color: #FFEA20; |
| 3137 font-size: 80%; |
| 3138 } |
| 3139 |
| 3140 /* All views: header legend value for hi rate */ |
| 3141 td.headerValueLegH |
| 3142 { |
| 3143 font-family: sans-serif; |
| 3144 text-align: center; |
| 3145 white-space: nowrap; |
| 3146 padding-left: 2px; |
| 3147 padding-right: 4px; |
| 3148 background-color: #A7FC9D; |
| 3149 font-size: 80%; |
| 3150 } |
| 3151 |
| 3152 /* All views except source code view: legend format for low coverage */ |
| 3153 span.coverLegendCovLo |
| 3154 { |
| 2685 padding-left: 10px; | 3155 padding-left: 10px; |
| 2686 padding-right: 10px; | 3156 padding-right: 10px; |
| 2687 » background-color: #B5F7AF; | 3157 » padding-top: 2px; |
| 3158 » background-color: #FF0000; |
| 3159 » } |
| 3160 |
| 3161 » /* All views except source code view: legend format for med coverage */ |
| 3162 » span.coverLegendCovMed |
| 3163 » { |
| 3164 » padding-left: 10px; |
| 3165 » padding-right: 10px; |
| 3166 » padding-top: 2px; |
| 3167 » background-color: #FFEA20; |
| 3168 » } |
| 3169 |
| 3170 » /* All views except source code view: legend format for hi coverage */ |
| 3171 » span.coverLegendCovHi |
| 3172 » { |
| 3173 » padding-left: 10px; |
| 3174 » padding-right: 10px; |
| 3175 » padding-top: 2px; |
| 3176 » background-color: #A7FC9D; |
| 2688 } | 3177 } |
| 2689 END_OF_CSS | 3178 END_OF_CSS |
| 2690 ; | 3179 ; |
| 2691 | 3180 |
| 2692 # ************************************************************* | 3181 # ************************************************************* |
| 2693 | 3182 |
| 2694 | 3183 |
| 2695 # Remove leading tab from all lines | 3184 # Remove leading tab from all lines |
| 2696 $css_data =~ s/^\t//gm; | 3185 $css_data =~ s/^\t//gm; |
| 2697 | 3186 |
| 2698 print(CSS_HANDLE $css_data); | 3187 print(CSS_HANDLE $css_data); |
| 2699 | 3188 |
| 2700 close(CSS_HANDLE); | 3189 close(CSS_HANDLE); |
| 2701 } | 3190 } |
| 2702 | 3191 |
| 2703 | 3192 |
| 2704 # | 3193 # |
| 2705 # get_bar_graph_code(base_dir, cover_found, cover_hit) | 3194 # get_bar_graph_code(base_dir, cover_found, cover_hit) |
| 2706 # | 3195 # |
| 2707 # Return a string containing HTML code which implements a bar graph display | 3196 # Return a string containing HTML code which implements a bar graph display |
| 2708 # for a coverage rate of cover_hit * 100 / cover_found. | 3197 # for a coverage rate of cover_hit * 100 / cover_found. |
| 2709 # | 3198 # |
| 2710 | 3199 |
| 2711 sub get_bar_graph_code($$$) | 3200 sub get_bar_graph_code($$$) |
| 2712 { | 3201 { |
| 3202 my ($base_dir, $found, $hit) = @_; |
| 2713 my $rate; | 3203 my $rate; |
| 2714 my $alt; | 3204 my $alt; |
| 2715 my $width; | 3205 my $width; |
| 2716 my $remainder; | 3206 my $remainder; |
| 2717 my $png_name; | 3207 my $png_name; |
| 2718 my $graph_code; | 3208 my $graph_code; |
| 2719 | 3209 |
| 2720 # Check number of instrumented lines | 3210 # Check number of instrumented lines |
| 2721 if ($_[1] == 0) { return ""; } | 3211 if ($_[1] == 0) { return ""; } |
| 2722 | 3212 |
| 2723 » $rate» » = $_[2] * 100 / $_[1]; | 3213 » $alt» » = rate($hit, $found, "%"); |
| 2724 » $alt» » = sprintf("%.1f", $rate)."%"; | 3214 » $width» » = rate($hit, $found, undef, 0); |
| 2725 » $width» » = sprintf("%.0f", $rate); | 3215 » $remainder» = 100 - $width; |
| 2726 » $remainder» = sprintf("%d", 100-$width); | |
| 2727 | 3216 |
| 2728 # Decide which .png file to use | 3217 # Decide which .png file to use |
| 2729 » $png_name = $rate_png[classify_rate($_[1], $_[2], $med_limit, | 3218 » $png_name = $rate_png[classify_rate($found, $hit, $med_limit, |
| 2730 $hi_limit)]; | 3219 $hi_limit)]; |
| 2731 | 3220 |
| 2732 if ($width == 0) | 3221 if ($width == 0) |
| 2733 { | 3222 { |
| 2734 # Zero coverage | 3223 # Zero coverage |
| 2735 $graph_code = (<<END_OF_HTML) | 3224 $graph_code = (<<END_OF_HTML) |
| 2736 <table border=0 cellspacing=0 cellpadding=1><tr><td class="cover
BarOutline"><img src="$_[0]snow.png" width=100 height=10 alt="$alt"></td></tr></
table> | 3225 <table border=0 cellspacing=0 cellpadding=1><tr><td class="cover
BarOutline"><img src="$_[0]snow.png" width=100 height=10 alt="$alt"></td></tr></
table> |
| 2737 END_OF_HTML | 3226 END_OF_HTML |
| 2738 ; | 3227 ; |
| 2739 } | 3228 } |
| (...skipping 28 matching lines...) Expand all Loading... |
| 2768 # | 3257 # |
| 2769 | 3258 |
| 2770 sub classify_rate($$$$) | 3259 sub classify_rate($$$$) |
| 2771 { | 3260 { |
| 2772 my ($found, $hit, $med, $hi) = @_; | 3261 my ($found, $hit, $med, $hi) = @_; |
| 2773 my $rate; | 3262 my $rate; |
| 2774 | 3263 |
| 2775 if ($found == 0) { | 3264 if ($found == 0) { |
| 2776 return 2; | 3265 return 2; |
| 2777 } | 3266 } |
| 2778 » $rate = $hit * 100 / $found; | 3267 » $rate = rate($hit, $found); |
| 2779 if ($rate < $med) { | 3268 if ($rate < $med) { |
| 2780 return 0; | 3269 return 0; |
| 2781 } elsif ($rate < $hi) { | 3270 } elsif ($rate < $hi) { |
| 2782 return 1; | 3271 return 1; |
| 2783 } | 3272 } |
| 2784 return 2; | 3273 return 2; |
| 2785 } | 3274 } |
| 2786 | 3275 |
| 2787 | 3276 |
| 2788 # | 3277 # |
| (...skipping 57 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2846 <td width="100%"> | 3335 <td width="100%"> |
| 2847 <table cellpadding=1 border=0 width="100%"> | 3336 <table cellpadding=1 border=0 width="100%"> |
| 2848 END_OF_HTML | 3337 END_OF_HTML |
| 2849 ; | 3338 ; |
| 2850 | 3339 |
| 2851 # ************************************************************* | 3340 # ************************************************************* |
| 2852 } | 3341 } |
| 2853 | 3342 |
| 2854 | 3343 |
| 2855 # | 3344 # |
| 2856 # write_header_line(filehandle, type, additional params..) | 3345 # write_header_line(handle, content) |
| 2857 # | 3346 # |
| 2858 # Write a header line. | 3347 # Write a header line with the specified table contents. |
| 2859 # | 3348 # |
| 2860 | 3349 |
| 2861 sub write_header_line(*$@) | 3350 sub write_header_line(*@) |
| 2862 { | 3351 { |
| 2863 » my $HANDLE = shift; | 3352 » my ($handle, @content) = @_; |
| 2864 » my $type = shift; | 3353 » my $entry; |
| 2865 » my @args = @_; | |
| 2866 | 3354 |
| 2867 » # Reduce indentation by using gotos | 3355 » write_html($handle, " <tr>\n"); |
| 2868 » if ($type eq 0) { | 3356 » foreach $entry (@content) { |
| 2869 » » goto header; | 3357 » » my ($width, $class, $text, $colspan) = @{$entry}; |
| 2870 » } elsif ($type eq 1) { | 3358 |
| 2871 » » goto body; | 3359 » » if (defined($width)) { |
| 2872 » } elsif ($type eq 2) { | 3360 » » » $width = " width=\"$width\""; |
| 2873 » » goto legend_dir; | 3361 » » } else { |
| 2874 » } elsif ($type eq 3) { | 3362 » » » $width = ""; |
| 2875 » » goto legend_source; | 3363 » » } |
| 2876 » } elsif ($type eq 4) { | 3364 » » if (defined($class)) { |
| 2877 » » goto half_body; | 3365 » » » $class = " class=\"$class\""; |
| 3366 » » } else { |
| 3367 » » » $class = ""; |
| 3368 » » } |
| 3369 » » if (defined($colspan)) { |
| 3370 » » » $colspan = " colspan=\"$colspan\""; |
| 3371 » » } else { |
| 3372 » » » $colspan = ""; |
| 3373 » » } |
| 3374 » » $text = "" if (!defined($text)); |
| 3375 » » write_html($handle, |
| 3376 » » » " <td$width$class$colspan>$text</td>\n"); |
| 2878 } | 3377 } |
| 2879 | 3378 » write_html($handle, " </tr>\n"); |
| 2880 header: | |
| 2881 » # ************************************************************* | |
| 2882 » write_html($HANDLE, <<END_OF_HTML); | |
| 2883 <tr> | |
| 2884 <td width="5%"></td> | |
| 2885 <td width="10%" class="headerItem">$args[0]</td> | |
| 2886 <td width="35%" class="headerValue">$args[1]</td> | |
| 2887 <td width="10%"></td> | |
| 2888 <td width="10%" class="headerCovTableHead">$args[2]</td> | |
| 2889 <td width="10%" class="headerCovTableHead">$args[3]</td> | |
| 2890 <td width="15%" class="headerCovTableHead">$args[4]</td> | |
| 2891 <td width="5%"></td> | |
| 2892 </tr> | |
| 2893 END_OF_HTML | |
| 2894 » # ************************************************************* | |
| 2895 » return; | |
| 2896 | |
| 2897 body: | |
| 2898 » # ************************************************************* | |
| 2899 » write_html($HANDLE, <<END_OF_HTML); | |
| 2900 <tr> | |
| 2901 <td></td> | |
| 2902 <td class="headerItem">$args[0]</td> | |
| 2903 <td class="headerValue">$args[1]</td> | |
| 2904 <td class="headerItem">$args[2]</td> | |
| 2905 <td class="headerCovTableEntry">$args[3]</td> | |
| 2906 <td class="headerCovTableEntry">$args[4]</td> | |
| 2907 <td class="headerCovTableEntry$args[5]">$args[6]</td> | |
| 2908 </tr> | |
| 2909 END_OF_HTML | |
| 2910 » # ************************************************************* | |
| 2911 » return; | |
| 2912 | |
| 2913 half_body: | |
| 2914 » # ************************************************************* | |
| 2915 » write_html($HANDLE, <<END_OF_HTML); | |
| 2916 <tr> | |
| 2917 <td></td> | |
| 2918 <td class="headerItem">$args[0]</td> | |
| 2919 <td class="headerValue">$args[1]</td> | |
| 2920 </tr> | |
| 2921 END_OF_HTML | |
| 2922 » # ************************************************************* | |
| 2923 » return; | |
| 2924 | |
| 2925 legend_dir: | |
| 2926 » # ************************************************************* | |
| 2927 » write_html($HANDLE, <<END_OF_HTML); | |
| 2928 <tr> | |
| 2929 <td></td> | |
| 2930 <td class="headerItemLeg">$args[0]</td> | |
| 2931 <td class="headerValueLeg"> | |
| 2932 $args[1] </td> | |
| 2933 <td></td> | |
| 2934 <td class="headerValueLeg" colspan=3> | |
| 2935 $args[2] </td> | |
| 2936 </tr> | |
| 2937 END_OF_HTML | |
| 2938 » # ************************************************************* | |
| 2939 » return; | |
| 2940 | |
| 2941 legend_source: | |
| 2942 » # ************************************************************* | |
| 2943 » write_html($HANDLE, <<END_OF_HTML); | |
| 2944 <tr> | |
| 2945 <td></td> | |
| 2946 <td class="headerItem">$args[0]</td> | |
| 2947 <td class="headerValueLeg" colspan=5> | |
| 2948 <span class="coverLegendNoCov">$args[1]</span> | |
| 2949 <span class="coverLegendCov">$args[2]</span> | |
| 2950 </td> | |
| 2951 </tr> | |
| 2952 END_OF_HTML | |
| 2953 » # ************************************************************* | |
| 2954 } | 3379 } |
| 2955 | 3380 |
| 2956 | 3381 |
| 2957 # | 3382 # |
| 2958 # write_header_epilog(filehandle, base_dir) | 3383 # write_header_epilog(filehandle, base_dir) |
| 2959 # | 3384 # |
| 2960 # Write end of page header HTML code. | 3385 # Write end of page header HTML code. |
| 2961 # | 3386 # |
| 2962 | 3387 |
| 2963 sub write_header_epilog(*$) | 3388 sub write_header_epilog(*$) |
| 2964 { | 3389 { |
| 2965 # ************************************************************* | 3390 # ************************************************************* |
| 2966 | 3391 |
| 2967 write_html($_[0], <<END_OF_HTML) | 3392 write_html($_[0], <<END_OF_HTML) |
| 2968 <tr><td><img src="$_[1]glass.png" width=3 height=3 alt=""></td><
/tr> | 3393 » <tr><td><img src="$_[1]glass.png" width=3 height=3 alt=""></td
></tr> |
| 2969 </table> | 3394 </table> |
| 2970 </td> | 3395 </td> |
| 2971 </tr> | 3396 </tr> |
| 3397 |
| 2972 <tr><td class="ruler"><img src="$_[1]glass.png" width=3 height=3 alt
=""></td></tr> | 3398 <tr><td class="ruler"><img src="$_[1]glass.png" width=3 height=3 alt
=""></td></tr> |
| 2973 </table> | 3399 </table> |
| 2974 | 3400 |
| 2975 END_OF_HTML | 3401 END_OF_HTML |
| 2976 ; | 3402 ; |
| 2977 | 3403 |
| 2978 # ************************************************************* | 3404 # ************************************************************* |
| 2979 } | 3405 } |
| 2980 | 3406 |
| 2981 | 3407 |
| 2982 # | 3408 # |
| 2983 # write_file_table_prolog(filehandle, file_heading, lines_heading, func_heading) | 3409 # write_file_table_prolog(handle, file_heading, ([heading, num_cols], ...)) |
| 2984 # | 3410 # |
| 2985 # Write heading for file table. | 3411 # Write heading for file table. |
| 2986 # | 3412 # |
| 2987 | 3413 |
| 2988 sub write_file_table_prolog(*$$$) | 3414 sub write_file_table_prolog(*$@) |
| 2989 { | 3415 { |
| 2990 » # ************************************************************* | 3416 » my ($handle, $file_heading, @columns) = @_; |
| 3417 » my $num_columns = 0; |
| 3418 » my $file_width; |
| 3419 » my $col; |
| 3420 » my $width; |
| 2991 | 3421 |
| 2992 if ($func_coverage) | 3422 » $width = 20 if (scalar(@columns) == 1); |
| 2993 { | 3423 » $width = 10 if (scalar(@columns) == 2); |
| 2994 write_html($_[0], <<END_OF_HTML) | 3424 » $width = 8 if (scalar(@columns) > 2); |
| 3425 |
| 3426 » foreach $col (@columns) { |
| 3427 » » my ($heading, $cols) = @{$col}; |
| 3428 |
| 3429 » » $num_columns += $cols; |
| 3430 » } |
| 3431 » $file_width = 100 - $num_columns * $width; |
| 3432 |
| 3433 » # Table definition |
| 3434 » write_html($handle, <<END_OF_HTML); |
| 2995 <center> | 3435 <center> |
| 2996 <table width="80%" cellpadding=1 cellspacing=1 border=0> | 3436 <table width="80%" cellpadding=1 cellspacing=1 border=0> |
| 2997 | 3437 |
| 2998 <tr> | 3438 <tr> |
| 2999 » <td width="45%"><br></td> | 3439 » <td width="$file_width%"><br></td> |
| 3000 » <td width="15%"></td> | 3440 END_OF_HTML |
| 3001 » <td width="10%"></td> | 3441 » # Empty first row |
| 3002 » <td width="10%"></td> | 3442 » foreach $col (@columns) { |
| 3003 » <td width="10%"></td> | 3443 » » my ($heading, $cols) = @{$col}; |
| 3004 » <td width="10%"></td> | 3444 |
| 3445 » » while ($cols-- > 0) { |
| 3446 » » » write_html($handle, <<END_OF_HTML); |
| 3447 » <td width="$width%"></td> |
| 3448 END_OF_HTML |
| 3449 » » } |
| 3450 » } |
| 3451 » # Next row |
| 3452 » write_html($handle, <<END_OF_HTML); |
| 3005 </tr> | 3453 </tr> |
| 3006 | 3454 |
| 3007 <tr> | 3455 <tr> |
| 3008 » <td class="tableHead">$_[1]</td> | 3456 » <td class="tableHead">$file_heading</td> |
| 3009 » <td class="tableHead" colspan=3>$_[2]</td> | 3457 END_OF_HTML |
| 3010 » <td class="tableHead" colspan=2>$_[3]</td> | 3458 » # Heading row |
| 3459 » foreach $col (@columns) { |
| 3460 » » my ($heading, $cols) = @{$col}; |
| 3461 » » my $colspan = ""; |
| 3462 |
| 3463 » » $colspan = " colspan=$cols" if ($cols > 1); |
| 3464 » » write_html($handle, <<END_OF_HTML); |
| 3465 » <td class="tableHead"$colspan>$heading</td> |
| 3466 END_OF_HTML |
| 3467 » } |
| 3468 » write_html($handle, <<END_OF_HTML); |
| 3011 </tr> | 3469 </tr> |
| 3470 END_OF_HTML |
| 3471 } |
| 3012 | 3472 |
| 3473 |
| 3474 # write_file_table_entry(handle, base_dir, filename, page_link, |
| 3475 # ([ found, hit, med_limit, hi_limit, graph ], ..) |
| 3476 # |
| 3477 # Write an entry of the file table. |
| 3478 # |
| 3479 |
| 3480 sub write_file_table_entry(*$$$@) |
| 3481 { |
| 3482 my ($handle, $base_dir, $filename, $page_link, @entries) = @_; |
| 3483 my $file_code; |
| 3484 my $entry; |
| 3485 my $esc_filename = escape_html($filename); |
| 3486 |
| 3487 # Add link to source if provided |
| 3488 if (defined($page_link) && $page_link ne "") { |
| 3489 $file_code = "<a href=\"$page_link\">$esc_filename</a>"; |
| 3490 } else { |
| 3491 $file_code = $esc_filename; |
| 3492 } |
| 3493 |
| 3494 # First column: filename |
| 3495 write_html($handle, <<END_OF_HTML); |
| 3496 <tr> |
| 3497 <td class="coverFile">$file_code</td> |
| 3013 END_OF_HTML | 3498 END_OF_HTML |
| 3014 ; | 3499 » # Columns as defined |
| 3015 } | 3500 » foreach $entry (@entries) { |
| 3016 else | 3501 » » my ($found, $hit, $med, $hi, $graph) = @{$entry}; |
| 3017 { | 3502 » » my $bar_graph; |
| 3018 write_html($_[0], <<END_OF_HTML) | 3503 » » my $class; |
| 3019 » <center> | 3504 » » my $rate; |
| 3020 » <table width="80%" cellpadding=1 cellspacing=1 border=0> | |
| 3021 | 3505 |
| 3022 » <tr> | 3506 » » # Generate bar graph if requested |
| 3023 » <td width="50%"><br></td> | 3507 » » if ($graph) { |
| 3024 » <td width="15%"></td> | 3508 » » » $bar_graph = get_bar_graph_code($base_dir, $found, |
| 3025 » <td width="15%"></td> | 3509 » » » » » » » $hit); |
| 3026 » <td width="20%"></td> | 3510 » » » write_html($handle, <<END_OF_HTML); |
| 3511 » <td class="coverBar" align="center"> |
| 3512 » $bar_graph |
| 3513 » </td> |
| 3514 END_OF_HTML |
| 3515 » » } |
| 3516 » » # Get rate color and text |
| 3517 » » if ($found == 0) { |
| 3518 » » » $rate = "-"; |
| 3519 » » » $class = "Hi"; |
| 3520 » » } else { |
| 3521 » » » $rate = rate($hit, $found, " %"); |
| 3522 » » » $class = $rate_name[classify_rate($found, $hit, |
| 3523 » » » » » $med, $hi)]; |
| 3524 » » } |
| 3525 » » write_html($handle, <<END_OF_HTML); |
| 3526 » <td class="coverPer$class">$rate</td> |
| 3527 » <td class="coverNum$class">$hit / $found</td> |
| 3528 END_OF_HTML |
| 3529 » } |
| 3530 » # End of row |
| 3531 write_html($handle, <<END_OF_HTML); |
| 3027 </tr> | 3532 </tr> |
| 3028 | |
| 3029 <tr> | |
| 3030 <td class="tableHead">$_[1]</td> | |
| 3031 <td class="tableHead" colspan=3>$_[2]</td> | |
| 3032 </tr> | |
| 3033 | |
| 3034 END_OF_HTML | 3533 END_OF_HTML |
| 3035 ; | |
| 3036 } | |
| 3037 | |
| 3038 # ************************************************************* | |
| 3039 } | 3534 } |
| 3040 | 3535 |
| 3041 | 3536 |
| 3042 # | 3537 # |
| 3043 # write_file_table_entry(filehandle, cover_filename, cover_bar_graph, | 3538 # write_file_table_detail_entry(filehandle, test_name, ([found, hit], ...)) |
| 3044 # cover_found, cover_hit, fn_found, fn_hit, | |
| 3045 #» » » page_link, func_link) | |
| 3046 # | |
| 3047 # Write an entry of the file table. | |
| 3048 # | |
| 3049 | |
| 3050 sub write_file_table_entry(*$$$$$$$) | |
| 3051 { | |
| 3052 » local *HANDLE = shift; | |
| 3053 » my ($filename, $bar_graph, $found, $hit, $fn_found, $fn_hit, | |
| 3054 » $page_link) = @_; | |
| 3055 » my $rate; | |
| 3056 » my $rate_string; | |
| 3057 » my $funcs_string; | |
| 3058 » my $class_lines = "Lo"; | |
| 3059 » my $class_funcs = "Hi"; | |
| 3060 » my $file_code; | |
| 3061 | |
| 3062 » # Add link to source if provided | |
| 3063 » if (defined($page_link) && $page_link ne "") { | |
| 3064 » » $file_code = "<a href=\"$page_link\">$filename</a>"; | |
| 3065 » } else { | |
| 3066 » » $file_code = $filename; | |
| 3067 » } | |
| 3068 | |
| 3069 » # Get line coverage rate | |
| 3070 » if ($found > 0) | |
| 3071 » { | |
| 3072 » » $rate = $hit * 100 / $found; | |
| 3073 » » $rate_string = sprintf("%.1f", $rate)." %"; | |
| 3074 » » | |
| 3075 » » $class_lines = $rate_name[classify_rate($found, $hit, | |
| 3076 » » » » » $med_limit, $hi_limit)]; | |
| 3077 » } | |
| 3078 » else | |
| 3079 » { | |
| 3080 » » $rate_string = "-"; | |
| 3081 » } | |
| 3082 | |
| 3083 » # Get function coverage rate | |
| 3084 » if ($fn_found > 0) | |
| 3085 » { | |
| 3086 » » $rate = $fn_hit * 100 / $fn_found; | |
| 3087 » » $class_funcs = $rate_name[classify_rate($fn_found, $fn_hit, | |
| 3088 » » » » » $fn_med_limit, $fn_hi_limit)]; | |
| 3089 » » $funcs_string = sprintf("%.1f", $rate)." %";» » | |
| 3090 » } | |
| 3091 » else | |
| 3092 » { | |
| 3093 » » # Define 0 of 0 functions as 100% | |
| 3094 » » $rate = 100; | |
| 3095 » » $funcs_string = "-"; | |
| 3096 » } | |
| 3097 | |
| 3098 » # ************************************************************* | |
| 3099 | |
| 3100 » write_html(*HANDLE, <<END_OF_HTML) | |
| 3101 » <tr> | |
| 3102 » <td class="coverFile">$file_code</td> | |
| 3103 » <td class="coverBar" align="center"> | |
| 3104 » $bar_graph | |
| 3105 » </td> | |
| 3106 » <td class="coverPer$class_lines">$rate_string</td> | |
| 3107 » <td class="coverNum$class_lines">$hit / $found</td> | |
| 3108 END_OF_HTML | |
| 3109 » ; | |
| 3110 | |
| 3111 if ($func_coverage) | |
| 3112 { | |
| 3113 write_html(*HANDLE, <<END_OF_HTML) | |
| 3114 » <td class="coverPer$class_funcs">$funcs_string</td> | |
| 3115 » <td class="coverNum$class_funcs">$fn_hit / $fn_found</td> | |
| 3116 END_OF_HTML | |
| 3117 » ; | |
| 3118 } | |
| 3119 write_html(*HANDLE, <<END_OF_HTML) | |
| 3120 » </tr> | |
| 3121 END_OF_HTML | |
| 3122 ; | |
| 3123 | |
| 3124 » # ************************************************************* | |
| 3125 } | |
| 3126 | |
| 3127 | |
| 3128 # | |
| 3129 # write_file_table_detail_heading(filehandle, left_heading, right_heading) | |
| 3130 # | |
| 3131 # Write heading for detail section in file table. | |
| 3132 # | |
| 3133 | |
| 3134 sub write_file_table_detail_heading(*$$$) | |
| 3135 { | |
| 3136 my $func_rows = ""; | |
| 3137 | |
| 3138 if ($func_coverage) | |
| 3139 { | |
| 3140 $func_rows = "<td class=\"testLinesHead\" colspan=2>$_[3]</td>"; | |
| 3141 } | |
| 3142 | |
| 3143 » # ************************************************************* | |
| 3144 » write_html($_[0], <<END_OF_HTML) | |
| 3145 » <tr> | |
| 3146 » <td class="testNameHead" colspan=2>$_[1]</td> | |
| 3147 » <td class="testLinesHead" colspan=2>$_[2]</td> | |
| 3148 $func_rows | |
| 3149 » </tr> | |
| 3150 | |
| 3151 END_OF_HTML | |
| 3152 » ; | |
| 3153 | |
| 3154 » # ************************************************************* | |
| 3155 } | |
| 3156 | |
| 3157 | |
| 3158 # | |
| 3159 # write_file_table_detail_entry(filehandle, test_name, | |
| 3160 # cover_found, cover_hit, func_found, func_hit) | |
| 3161 # | 3539 # |
| 3162 # Write entry for detail section in file table. | 3540 # Write entry for detail section in file table. |
| 3163 # | 3541 # |
| 3164 | 3542 |
| 3165 sub write_file_table_detail_entry(*$$$$$) | 3543 sub write_file_table_detail_entry(*$@) |
| 3166 { | 3544 { |
| 3167 » my $rate; | 3545 » my ($handle, $test, @entries) = @_; |
| 3168 » my $func_rate; | 3546 » my $entry; |
| 3169 » my $name = $_[1]; | 3547 |
| 3170 » | 3548 » if ($test eq "") { |
| 3171 » if ($_[2]>0) | 3549 » » $test = "<span style=\"font-style:italic\"><unnamed></span
>"; |
| 3172 » { | 3550 » } elsif ($test =~ /^(.*),diff$/) { |
| 3173 » » $rate = sprintf("%.1f", $_[3]*100/$_[2])." %"; | 3551 » » $test = $1." (converted)"; |
| 3174 } | 3552 } |
| 3175 » else | 3553 » # Testname |
| 3176 » { | 3554 » write_html($handle, <<END_OF_HTML); |
| 3177 » » $rate = "-"; | 3555 » <tr> |
| 3556 » <td class="testName" colspan=2>$test</td> |
| 3557 END_OF_HTML |
| 3558 » # Test data |
| 3559 » foreach $entry (@entries) { |
| 3560 » » my ($found, $hit) = @{$entry}; |
| 3561 » » my $rate = rate($hit, $found, " %"); |
| 3562 |
| 3563 » » write_html($handle, <<END_OF_HTML); |
| 3564 » <td class="testPer">$rate</td> |
| 3565 » <td class="testNum">$hit / $found</td> |
| 3566 END_OF_HTML |
| 3178 } | 3567 } |
| 3179 | 3568 |
| 3180 » if ($_[4]>0) | 3569 write_html($handle, <<END_OF_HTML); |
| 3181 » { | |
| 3182 » » $func_rate = sprintf("%.1f", $_[5]*100/$_[4])." %"; | |
| 3183 » } | |
| 3184 » else | |
| 3185 » { | |
| 3186 » » $func_rate = "-"; | |
| 3187 » } | |
| 3188 | |
| 3189 » if ($name =~ /^(.*),diff$/) | |
| 3190 » { | |
| 3191 » » $name = $1." (converted)"; | |
| 3192 » } | |
| 3193 | |
| 3194 » if ($name eq "") | |
| 3195 » { | |
| 3196 » » $name = "<span style=\"font-style:italic\"><unnamed></span
>"; | |
| 3197 » } | |
| 3198 | |
| 3199 » # ************************************************************* | |
| 3200 | |
| 3201 » write_html($_[0], <<END_OF_HTML) | |
| 3202 » <tr> | |
| 3203 » <td class="testName" colspan=2>$name</td> | |
| 3204 » <td class="testPer">$rate</td> | |
| 3205 » <td class="testNum">$_[3] / $_[2] lines</td> | |
| 3206 END_OF_HTML | |
| 3207 » ; | |
| 3208 if ($func_coverage) | |
| 3209 { | |
| 3210 write_html($_[0], <<END_OF_HTML) | |
| 3211 » <td class="testPer">$func_rate</td> | |
| 3212 » <td class="testNum">$_[5] / $_[4]</td> | |
| 3213 END_OF_HTML | |
| 3214 » ; | |
| 3215 } | |
| 3216 write_html($_[0], <<END_OF_HTML) | |
| 3217 </tr> | 3570 </tr> |
| 3218 | 3571 |
| 3219 END_OF_HTML | 3572 END_OF_HTML |
| 3220 ; | |
| 3221 | 3573 |
| 3222 # ************************************************************* | 3574 # ************************************************************* |
| 3223 } | 3575 } |
| 3224 | 3576 |
| 3225 | 3577 |
| 3226 # | 3578 # |
| 3227 # write_file_table_epilog(filehandle) | 3579 # write_file_table_epilog(filehandle) |
| 3228 # | 3580 # |
| 3229 # Write end of file table HTML code. | 3581 # Write end of file table HTML code. |
| 3230 # | 3582 # |
| (...skipping 84 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 3315 </center> | 3667 </center> |
| 3316 <br> | 3668 <br> |
| 3317 | 3669 |
| 3318 END_OF_HTML | 3670 END_OF_HTML |
| 3319 ; | 3671 ; |
| 3320 | 3672 |
| 3321 # ************************************************************* | 3673 # ************************************************************* |
| 3322 } | 3674 } |
| 3323 | 3675 |
| 3324 | 3676 |
| 3677 sub fmt_centered($$) |
| 3678 { |
| 3679 my ($width, $text) = @_; |
| 3680 my $w0 = length($text); |
| 3681 my $w1 = int(($width - $w0) / 2); |
| 3682 my $w2 = $width - $w0 - $w1; |
| 3683 |
| 3684 return (" "x$w1).$text.(" "x$w2); |
| 3685 } |
| 3686 |
| 3687 |
| 3325 # | 3688 # |
| 3326 # write_source_prolog(filehandle) | 3689 # write_source_prolog(filehandle) |
| 3327 # | 3690 # |
| 3328 # Write start of source code table. | 3691 # Write start of source code table. |
| 3329 # | 3692 # |
| 3330 | 3693 |
| 3331 sub write_source_prolog(*) | 3694 sub write_source_prolog(*) |
| 3332 { | 3695 { |
| 3696 my $lineno_heading = " "; |
| 3697 my $branch_heading = ""; |
| 3698 my $line_heading = fmt_centered($line_field_width, "Line data"); |
| 3699 my $source_heading = " Source code"; |
| 3700 |
| 3701 if ($br_coverage) { |
| 3702 $branch_heading = fmt_centered($br_field_width, "Branch data"). |
| 3703 " "; |
| 3704 } |
| 3333 # ************************************************************* | 3705 # ************************************************************* |
| 3334 | 3706 |
| 3335 write_html($_[0], <<END_OF_HTML) | 3707 write_html($_[0], <<END_OF_HTML) |
| 3336 <table cellpadding=0 cellspacing=0 border=0> | 3708 <table cellpadding=0 cellspacing=0 border=0> |
| 3337 <tr> | 3709 <tr> |
| 3338 <td><br></td> | 3710 <td><br></td> |
| 3339 </tr> | 3711 </tr> |
| 3340 <tr> | 3712 <tr> |
| 3341 » <td><pre class="source"> | 3713 » <td> |
| 3714 <pre class="sourceHeading">${lineno_heading}${branch_heading}${line_heading} ${s
ource_heading}</pre> |
| 3715 <pre class="source"> |
| 3342 END_OF_HTML | 3716 END_OF_HTML |
| 3343 ; | 3717 ; |
| 3344 | 3718 |
| 3345 # ************************************************************* | 3719 # ************************************************************* |
| 3346 } | 3720 } |
| 3347 | 3721 |
| 3348 | 3722 |
| 3349 # | 3723 # |
| 3724 # get_branch_blocks(brdata) |
| 3725 # |
| 3726 # Group branches that belong to the same basic block. |
| 3727 # |
| 3728 # Returns: [block1, block2, ...] |
| 3729 # block: [branch1, branch2, ...] |
| 3730 # branch: [block_num, branch_num, taken_count, text_length, open, close] |
| 3731 # |
| 3732 |
| 3733 sub get_branch_blocks($) |
| 3734 { |
| 3735 my ($brdata) = @_; |
| 3736 my $last_block_num; |
| 3737 my $block = []; |
| 3738 my @blocks; |
| 3739 my $i; |
| 3740 my $num = br_ivec_len($brdata); |
| 3741 |
| 3742 # Group branches |
| 3743 for ($i = 0; $i < $num; $i++) { |
| 3744 my ($block_num, $branch, $taken) = br_ivec_get($brdata, $i); |
| 3745 my $br; |
| 3746 |
| 3747 if (defined($last_block_num) && $block_num != $last_block_num) { |
| 3748 push(@blocks, $block); |
| 3749 $block = []; |
| 3750 } |
| 3751 $br = [$block_num, $branch, $taken, 3, 0, 0]; |
| 3752 push(@{$block}, $br); |
| 3753 $last_block_num = $block_num; |
| 3754 } |
| 3755 push(@blocks, $block) if (scalar(@{$block}) > 0); |
| 3756 |
| 3757 # Add braces to first and last branch in group |
| 3758 foreach $block (@blocks) { |
| 3759 $block->[0]->[$BR_OPEN] = 1; |
| 3760 $block->[0]->[$BR_LEN]++; |
| 3761 $block->[scalar(@{$block}) - 1]->[$BR_CLOSE] = 1; |
| 3762 $block->[scalar(@{$block}) - 1]->[$BR_LEN]++; |
| 3763 } |
| 3764 |
| 3765 return @blocks; |
| 3766 } |
| 3767 |
| 3768 # |
| 3769 # get_block_len(block) |
| 3770 # |
| 3771 # Calculate total text length of all branches in a block of branches. |
| 3772 # |
| 3773 |
| 3774 sub get_block_len($) |
| 3775 { |
| 3776 my ($block) = @_; |
| 3777 my $len = 0; |
| 3778 my $branch; |
| 3779 |
| 3780 foreach $branch (@{$block}) { |
| 3781 $len += $branch->[$BR_LEN]; |
| 3782 } |
| 3783 |
| 3784 return $len; |
| 3785 } |
| 3786 |
| 3787 |
| 3788 # |
| 3789 # get_branch_html(brdata) |
| 3790 # |
| 3791 # Return a list of HTML lines which represent the specified branch coverage |
| 3792 # data in source code view. |
| 3793 # |
| 3794 |
| 3795 sub get_branch_html($) |
| 3796 { |
| 3797 my ($brdata) = @_; |
| 3798 my @blocks = get_branch_blocks($brdata); |
| 3799 my $block; |
| 3800 my $branch; |
| 3801 my $line_len = 0; |
| 3802 my $line = []; # [branch2|" ", branch|" ", ...] |
| 3803 my @lines; # [line1, line2, ...] |
| 3804 my @result; |
| 3805 |
| 3806 # Distribute blocks to lines |
| 3807 foreach $block (@blocks) { |
| 3808 my $block_len = get_block_len($block); |
| 3809 |
| 3810 # Does this block fit into the current line? |
| 3811 if ($line_len + $block_len <= $br_field_width) { |
| 3812 # Add it |
| 3813 $line_len += $block_len; |
| 3814 push(@{$line}, @{$block}); |
| 3815 next; |
| 3816 } elsif ($block_len <= $br_field_width) { |
| 3817 # It would fit if the line was empty - add it to new |
| 3818 # line |
| 3819 push(@lines, $line); |
| 3820 $line_len = $block_len; |
| 3821 $line = [ @{$block} ]; |
| 3822 next; |
| 3823 } |
| 3824 # Split the block into several lines |
| 3825 foreach $branch (@{$block}) { |
| 3826 if ($line_len + $branch->[$BR_LEN] >= $br_field_width) { |
| 3827 # Start a new line |
| 3828 if (($line_len + 1 <= $br_field_width) && |
| 3829 scalar(@{$line}) > 0 && |
| 3830 !$line->[scalar(@$line) - 1]->[$BR_CLOSE]) { |
| 3831 # Try to align branch symbols to be in |
| 3832 # one # row |
| 3833 push(@{$line}, " "); |
| 3834 } |
| 3835 push(@lines, $line); |
| 3836 $line_len = 0; |
| 3837 $line = []; |
| 3838 } |
| 3839 push(@{$line}, $branch); |
| 3840 $line_len += $branch->[$BR_LEN]; |
| 3841 } |
| 3842 } |
| 3843 push(@lines, $line); |
| 3844 |
| 3845 # Convert to HTML |
| 3846 foreach $line (@lines) { |
| 3847 my $current = ""; |
| 3848 my $current_len = 0; |
| 3849 |
| 3850 foreach $branch (@$line) { |
| 3851 # Skip alignment space |
| 3852 if ($branch eq " ") { |
| 3853 $current .= " "; |
| 3854 $current_len++; |
| 3855 next; |
| 3856 } |
| 3857 |
| 3858 my ($block_num, $br_num, $taken, $len, $open, $close) = |
| 3859 @{$branch}; |
| 3860 my $class; |
| 3861 my $title; |
| 3862 my $text; |
| 3863 |
| 3864 if ($taken eq '-') { |
| 3865 $class = "branchNoExec"; |
| 3866 $text = " # "; |
| 3867 $title = "Branch $br_num was not executed"; |
| 3868 } elsif ($taken == 0) { |
| 3869 $class = "branchNoCov"; |
| 3870 $text = " - "; |
| 3871 $title = "Branch $br_num was not taken"; |
| 3872 } else { |
| 3873 $class = "branchCov"; |
| 3874 $text = " + "; |
| 3875 $title = "Branch $br_num was taken $taken ". |
| 3876 "time"; |
| 3877 $title .= "s" if ($taken > 1); |
| 3878 } |
| 3879 $current .= "[" if ($open); |
| 3880 $current .= "<span class=\"$class\" title=\"$title\">"; |
| 3881 $current .= $text."</span>"; |
| 3882 $current .= "]" if ($close); |
| 3883 $current_len += $len; |
| 3884 } |
| 3885 |
| 3886 # Right-align result text |
| 3887 if ($current_len < $br_field_width) { |
| 3888 $current = (" "x($br_field_width - $current_len)). |
| 3889 $current; |
| 3890 } |
| 3891 push(@result, $current); |
| 3892 } |
| 3893 |
| 3894 return @result; |
| 3895 } |
| 3896 |
| 3897 |
| 3898 # |
| 3899 # format_count(count, width) |
| 3900 # |
| 3901 # Return a right-aligned representation of count that fits in width characters. |
| 3902 # |
| 3903 |
| 3904 sub format_count($$) |
| 3905 { |
| 3906 my ($count, $width) = @_; |
| 3907 my $result; |
| 3908 my $exp; |
| 3909 |
| 3910 $result = sprintf("%*.0f", $width, $count); |
| 3911 while (length($result) > $width) { |
| 3912 last if ($count < 10); |
| 3913 $exp++; |
| 3914 $count = int($count/10); |
| 3915 $result = sprintf("%*s", $width, ">$count*10^$exp"); |
| 3916 } |
| 3917 return $result; |
| 3918 } |
| 3919 |
| 3920 # |
| 3350 # write_source_line(filehandle, line_num, source, hit_count, converted, | 3921 # write_source_line(filehandle, line_num, source, hit_count, converted, |
| 3351 # add_anchor) | 3922 # brdata, add_anchor) |
| 3352 # | 3923 # |
| 3353 # Write formatted source code line. Return a line in a format as needed | 3924 # Write formatted source code line. Return a line in a format as needed |
| 3354 # by gen_png() | 3925 # by gen_png() |
| 3355 # | 3926 # |
| 3356 | 3927 |
| 3357 sub write_source_line(*$$$$$) | 3928 sub write_source_line(*$$$$$$) |
| 3358 { | 3929 { |
| 3930 » my ($handle, $line, $source, $count, $converted, $brdata, |
| 3931 » $add_anchor) = @_; |
| 3359 my $source_format; | 3932 my $source_format; |
| 3360 » my $count; | 3933 » my $count_format; |
| 3361 my $result; | 3934 my $result; |
| 3362 my $anchor_start = ""; | 3935 my $anchor_start = ""; |
| 3363 my $anchor_end = ""; | 3936 my $anchor_end = ""; |
| 3364 | 3937 » my $count_field_width = $line_field_width - 1; |
| 3365 » if (!(defined$_[3])) | 3938 » my @br_html; |
| 3366 » { | 3939 » my $html; |
| 3940 |
| 3941 » # Get branch HTML data for this line |
| 3942 » @br_html = get_branch_html($brdata) if ($br_coverage); |
| 3943 |
| 3944 » if (!defined($count)) { |
| 3367 $result = ""; | 3945 $result = ""; |
| 3368 $source_format = ""; | 3946 $source_format = ""; |
| 3369 » » $count» » = " "x15; | 3947 » » $count_format» = " "x$count_field_width; |
| 3370 » } | 3948 » } |
| 3371 » elsif ($_[3] == 0) | 3949 » elsif ($count == 0) { |
| 3372 » { | 3950 » » $result»» = $count; |
| 3373 » » $result»» = $_[3]; | |
| 3374 $source_format = '<span class="lineNoCov">'; | 3951 $source_format = '<span class="lineNoCov">'; |
| 3375 » » $count» » = sprintf("%15d", $_[3]); | 3952 » » $count_format» = format_count($count, $count_field_width); |
| 3376 » } | 3953 » } |
| 3377 » elsif ($_[4] && defined($highlight)) | 3954 » elsif ($converted && defined($highlight)) { |
| 3378 » { | 3955 » » $result»» = "*".$count; |
| 3379 » » $result»» = "*".$_[3]; | |
| 3380 $source_format = '<span class="lineDiffCov">'; | 3956 $source_format = '<span class="lineDiffCov">'; |
| 3381 » » $count» » = sprintf("%15d", $_[3]); | 3957 » » $count_format» = format_count($count, $count_field_width); |
| 3382 » } | 3958 » } |
| 3383 » else | 3959 » else { |
| 3384 » { | 3960 » » $result»» = $count; |
| 3385 » » $result»» = $_[3]; | |
| 3386 $source_format = '<span class="lineCov">'; | 3961 $source_format = '<span class="lineCov">'; |
| 3387 » » $count» » = sprintf("%15d", $_[3]); | 3962 » » $count_format» = format_count($count, $count_field_width); |
| 3388 » } | 3963 » } |
| 3389 | 3964 » $result .= ":".$source; |
| 3390 » $result .= ":".$_[2]; | |
| 3391 | 3965 |
| 3392 # Write out a line number navigation anchor every $nav_resolution | 3966 # Write out a line number navigation anchor every $nav_resolution |
| 3393 # lines if necessary | 3967 # lines if necessary |
| 3394 » if ($_[5]) | 3968 » if ($add_anchor) |
| 3395 { | 3969 { |
| 3396 $anchor_start = "<a name=\"$_[1]\">"; | 3970 $anchor_start = "<a name=\"$_[1]\">"; |
| 3397 $anchor_end = "</a>"; | 3971 $anchor_end = "</a>"; |
| 3398 } | 3972 } |
| 3399 | 3973 |
| 3400 | 3974 |
| 3401 # ************************************************************* | 3975 # ************************************************************* |
| 3402 | 3976 |
| 3403 » write_html($_[0], | 3977 » $html = $anchor_start; |
| 3404 » » $anchor_start. | 3978 » $html .= "<span class=\"lineNum\">".sprintf("%8d", $line)." </span>"; |
| 3405 » » '<span class="lineNum">'.sprintf("%8d", $_[1]). | 3979 » $html .= shift(@br_html).":" if ($br_coverage); |
| 3406 » » " </span>$source_format$count : ". | 3980 » $html .= "$source_format$count_format : "; |
| 3407 » » escape_html($_[2]).($source_format?"</span>":""). | 3981 » $html .= escape_html($source); |
| 3408 » » $anchor_end."\n"); | 3982 » $html .= "</span>" if ($source_format); |
| 3983 » $html .= $anchor_end."\n"; |
| 3409 | 3984 |
| 3985 write_html($handle, $html); |
| 3986 |
| 3987 if ($br_coverage) { |
| 3988 # Add lines for overlong branch information |
| 3989 foreach (@br_html) { |
| 3990 write_html($handle, "<span class=\"lineNum\">". |
| 3991 " </span>$_\n"); |
| 3992 } |
| 3993 } |
| 3410 # ************************************************************* | 3994 # ************************************************************* |
| 3411 | 3995 |
| 3412 return($result); | 3996 return($result); |
| 3413 } | 3997 } |
| 3414 | 3998 |
| 3415 | 3999 |
| 3416 # | 4000 # |
| 3417 # write_source_epilog(filehandle) | 4001 # write_source_epilog(filehandle) |
| 3418 # | 4002 # |
| 3419 # Write end of source code table. | 4003 # Write end of source code table. |
| (...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 3453 | 4037 |
| 3454 if (defined($_[2])) | 4038 if (defined($_[2])) |
| 3455 { | 4039 { |
| 3456 $break_code = " target=\"_parent\""; | 4040 $break_code = " target=\"_parent\""; |
| 3457 } | 4041 } |
| 3458 | 4042 |
| 3459 # ************************************************************* | 4043 # ************************************************************* |
| 3460 | 4044 |
| 3461 write_html($_[0], <<END_OF_HTML) | 4045 write_html($_[0], <<END_OF_HTML) |
| 3462 <table width="100%" border=0 cellspacing=0 cellpadding=0> | 4046 <table width="100%" border=0 cellspacing=0 cellpadding=0> |
| 3463 » <tr><td class="ruler"><img src="$_[1]glass.png" width=3 height=3 alt="
"></td></tr> | 4047 » <tr><td class="ruler"><img src="$_[1]glass.png" width=3 height=3 alt
=""></td></tr> |
| 3464 » <tr><td class="versionInfo">Generated by: <a href="$lcov_url"$break_co
de>$lcov_version</a></td></tr> | 4048 » <tr><td class="versionInfo">Generated by: <a href="$lcov_url"$break_
code>$lcov_version</a></td></tr> |
| 3465 </table> | 4049 </table> |
| 3466 <br> | 4050 <br> |
| 3467 END_OF_HTML | 4051 END_OF_HTML |
| 3468 ; | 4052 ; |
| 3469 | 4053 |
| 3470 $epilog = $html_epilog; | 4054 $epilog = $html_epilog; |
| 3471 $epilog =~ s/\@basedir\@/$basedir/g; | 4055 $epilog =~ s/\@basedir\@/$basedir/g; |
| 3472 | 4056 |
| 3473 write_html($_[0], $epilog); | 4057 write_html($_[0], $epilog); |
| 3474 } | 4058 } |
| 3475 | 4059 |
| 3476 | 4060 |
| 3477 # | 4061 # |
| 3478 # write_frameset(filehandle, basedir, basename, pagetitle) | 4062 # write_frameset(filehandle, basedir, basename, pagetitle) |
| 3479 # | 4063 # |
| 3480 # | 4064 # |
| 3481 | 4065 |
| 3482 sub write_frameset(*$$$) | 4066 sub write_frameset(*$$$) |
| 3483 { | 4067 { |
| 3484 my $frame_width = $overview_width + 40; | 4068 my $frame_width = $overview_width + 40; |
| 3485 | 4069 |
| 3486 # ************************************************************* | 4070 # ************************************************************* |
| 3487 | 4071 |
| 3488 write_html($_[0], <<END_OF_HTML) | 4072 write_html($_[0], <<END_OF_HTML) |
| 3489 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Frameset//EN"> | 4073 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Frameset//EN"> |
| 3490 | 4074 |
| 3491 <html lang="en"> | 4075 <html lang="en"> |
| 3492 | 4076 |
| 3493 <head> | 4077 <head> |
| 3494 » <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1
"> | 4078 » <meta http-equiv="Content-Type" content="text/html; charset=$charset"> |
| 3495 <title>$_[3]</title> | 4079 <title>$_[3]</title> |
| 3496 <link rel="stylesheet" type="text/css" href="$_[1]gcov.css"> | 4080 <link rel="stylesheet" type="text/css" href="$_[1]gcov.css"> |
| 3497 </head> | 4081 </head> |
| 3498 | 4082 |
| 3499 <frameset cols="$frame_width,*"> | 4083 <frameset cols="$frame_width,*"> |
| 3500 <frame src="$_[2].gcov.overview.$html_ext" name="overview"> | 4084 <frame src="$_[2].gcov.overview.$html_ext" name="overview"> |
| 3501 <frame src="$_[2].gcov.$html_ext" name="source"> | 4085 <frame src="$_[2].gcov.$html_ext" name="source"> |
| 3502 <noframes> | 4086 <noframes> |
| 3503 <center>Frames not supported by your browser!<br></center> | 4087 <center>Frames not supported by your browser!<br></center> |
| 3504 </noframes> | 4088 </noframes> |
| (...skipping 42 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 3547 | 4131 |
| 3548 # ************************************************************* | 4132 # ************************************************************* |
| 3549 | 4133 |
| 3550 write_html($_[0], <<END_OF_HTML) | 4134 write_html($_[0], <<END_OF_HTML) |
| 3551 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> | 4135 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> |
| 3552 | 4136 |
| 3553 <html lang="en"> | 4137 <html lang="en"> |
| 3554 | 4138 |
| 3555 <head> | 4139 <head> |
| 3556 <title>$_[3]</title> | 4140 <title>$_[3]</title> |
| 3557 » <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1
"> | 4141 » <meta http-equiv="Content-Type" content="text/html; charset=$charset"> |
| 3558 <link rel="stylesheet" type="text/css" href="$_[1]gcov.css"> | 4142 <link rel="stylesheet" type="text/css" href="$_[1]gcov.css"> |
| 3559 </head> | 4143 </head> |
| 3560 | 4144 |
| 3561 <body> | 4145 <body> |
| 3562 <map name="overview"> | 4146 <map name="overview"> |
| 3563 END_OF_HTML | 4147 END_OF_HTML |
| 3564 ; | 4148 ; |
| 3565 | 4149 |
| 3566 # ************************************************************* | 4150 # ************************************************************* |
| 3567 | 4151 |
| (...skipping 26 matching lines...) Expand all Loading... |
| 3594 </center> | 4178 </center> |
| 3595 </body> | 4179 </body> |
| 3596 </html> | 4180 </html> |
| 3597 END_OF_HTML | 4181 END_OF_HTML |
| 3598 ; | 4182 ; |
| 3599 | 4183 |
| 3600 # ************************************************************* | 4184 # ************************************************************* |
| 3601 } | 4185 } |
| 3602 | 4186 |
| 3603 | 4187 |
| 3604 # rate_to_col(found, hit) | 4188 sub max($$) |
| 3605 # | 4189 { |
| 3606 # Return Lo, Med or Hi, depending on the coverage rate. | 4190 » my ($a, $b) = @_; |
| 3607 # | |
| 3608 | 4191 |
| 3609 sub rate_to_col($$) | 4192 » return $a if ($a > $b); |
| 3610 { | 4193 » return $b; |
| 3611 » my ($found, $hit) = @_; | |
| 3612 » my $rate; | |
| 3613 | |
| 3614 » if ($found == 0) { | |
| 3615 » » return "Hi"; | |
| 3616 » } | |
| 3617 » $rate = 100 * $hit / $found; | |
| 3618 » if ($rate < $med_limit) { | |
| 3619 » » return "Lo"; | |
| 3620 » } elsif ($rate < $hi_limit) { | |
| 3621 » » return "Med"; | |
| 3622 » } | |
| 3623 » return "Hi"; | |
| 3624 } | 4194 } |
| 3625 | 4195 |
| 3626 # format_rate(found, hit) | |
| 3627 # | |
| 3628 # Return formatted percent string for coverage rate. | |
| 3629 # | |
| 3630 | |
| 3631 sub format_rate($$) | |
| 3632 { | |
| 3633 return $_[0] == 0 ? "-" : sprintf("%.1f", $_[1] * 100 / $_[0])." %"; | |
| 3634 } | |
| 3635 | |
| 3636 sub get_legend_code($$$) | |
| 3637 { | |
| 3638 my ($text, $med, $hi) = @_; | |
| 3639 my $result; | |
| 3640 | |
| 3641 $result = <<EOF; | |
| 3642 $text<br> | |
| 3643 <span class="coverLegendLo">0% to $med%</span> | |
| 3644 <span class="coverLegendMed">$med% to $hi%</span> | |
| 3645 <span class="coverLegendHi">$hi% to 100%</span> | |
| 3646 EOF | |
| 3647 return $result; | |
| 3648 } | |
| 3649 | 4196 |
| 3650 # | 4197 # |
| 3651 # write_header(filehandle, type, trunc_file_name, rel_file_name, lines_found, | 4198 # write_header(filehandle, type, trunc_file_name, rel_file_name, lines_found, |
| 3652 # lines_hit, funcs_found, funcs_hit, sort_type) | 4199 # lines_hit, funcs_found, funcs_hit, sort_type) |
| 3653 # | 4200 # |
| 3654 # Write a complete standard page header. TYPE may be (0, 1, 2, 3, 4) | 4201 # Write a complete standard page header. TYPE may be (0, 1, 2, 3, 4) |
| 3655 # corresponding to (directory view header, file view header, source view | 4202 # corresponding to (directory view header, file view header, source view |
| 3656 # header, test case description header, function view header) | 4203 # header, test case description header, function view header) |
| 3657 # | 4204 # |
| 3658 | 4205 |
| 3659 sub write_header(*$$$$$$$$) | 4206 sub write_header(*$$$$$$$$$$) |
| 3660 { | 4207 { |
| 3661 local *HTML_HANDLE = $_[0]; | 4208 local *HTML_HANDLE = $_[0]; |
| 3662 my $type = $_[1]; | 4209 my $type = $_[1]; |
| 3663 my $trunc_name = $_[2]; | 4210 my $trunc_name = $_[2]; |
| 3664 my $rel_filename = $_[3]; | 4211 my $rel_filename = $_[3]; |
| 3665 my $lines_found = $_[4]; | 4212 my $lines_found = $_[4]; |
| 3666 my $lines_hit = $_[5]; | 4213 my $lines_hit = $_[5]; |
| 3667 my $fn_found = $_[6]; | 4214 my $fn_found = $_[6]; |
| 3668 my $fn_hit = $_[7]; | 4215 my $fn_hit = $_[7]; |
| 3669 » my $sort_type = $_[8]; | 4216 » my $br_found = $_[8]; |
| 4217 » my $br_hit = $_[9]; |
| 4218 » my $sort_type = $_[10]; |
| 3670 my $base_dir; | 4219 my $base_dir; |
| 3671 my $view; | 4220 my $view; |
| 3672 my $test; | 4221 my $test; |
| 3673 my $base_name; | 4222 my $base_name; |
| 4223 my $style; |
| 4224 my $rate; |
| 4225 my @row_left; |
| 4226 my @row_right; |
| 4227 my $num_rows; |
| 4228 my $i; |
| 4229 my $esc_trunc_name = escape_html($trunc_name); |
| 3674 | 4230 |
| 3675 $base_name = basename($rel_filename); | 4231 $base_name = basename($rel_filename); |
| 3676 | 4232 |
| 3677 # Prepare text for "current view" field | 4233 # Prepare text for "current view" field |
| 3678 » if ($type == 0) | 4234 » if ($type == $HDR_DIR) |
| 3679 { | 4235 { |
| 3680 # Main overview | 4236 # Main overview |
| 3681 $base_dir = ""; | 4237 $base_dir = ""; |
| 3682 $view = $overview_title; | 4238 $view = $overview_title; |
| 3683 } | 4239 } |
| 3684 » elsif ($type == 1) | 4240 » elsif ($type == $HDR_FILE) |
| 3685 { | 4241 { |
| 3686 # Directory overview | 4242 # Directory overview |
| 3687 $base_dir = get_relative_base_path($rel_filename); | 4243 $base_dir = get_relative_base_path($rel_filename); |
| 3688 $view = "<a href=\"$base_dir"."index.$html_ext\">". | 4244 $view = "<a href=\"$base_dir"."index.$html_ext\">". |
| 3689 » » » "$overview_title</a> - $trunc_name"; | 4245 » » » "$overview_title</a> - $esc_trunc_name"; |
| 3690 } | 4246 } |
| 3691 » elsif ($type == 2 || $type == 4) | 4247 » elsif ($type == $HDR_SOURCE || $type == $HDR_FUNC) |
| 3692 { | 4248 { |
| 3693 # File view | 4249 # File view |
| 3694 my $dir_name = dirname($rel_filename); | 4250 my $dir_name = dirname($rel_filename); |
| 4251 my $esc_base_name = escape_html($base_name); |
| 4252 my $esc_dir_name = escape_html($dir_name); |
| 3695 | 4253 |
| 3696 $base_dir = get_relative_base_path($dir_name); | 4254 $base_dir = get_relative_base_path($dir_name); |
| 3697 if ($frames) | 4255 if ($frames) |
| 3698 { | 4256 { |
| 3699 # Need to break frameset when clicking any of these | 4257 # Need to break frameset when clicking any of these |
| 3700 # links | 4258 # links |
| 3701 $view = "<a href=\"$base_dir"."index.$html_ext\" ". | 4259 $view = "<a href=\"$base_dir"."index.$html_ext\" ". |
| 3702 "target=\"_parent\">$overview_title</a> - ". | 4260 "target=\"_parent\">$overview_title</a> - ". |
| 3703 "<a href=\"index.$html_ext\" target=\"_parent\">
". | 4261 "<a href=\"index.$html_ext\" target=\"_parent\">
". |
| 3704 » » » » "$dir_name</a> - $base_name"; | 4262 » » » » "$esc_dir_name</a> - $esc_base_name"; |
| 3705 } | 4263 } |
| 3706 else | 4264 else |
| 3707 { | 4265 { |
| 3708 $view = "<a href=\"$base_dir"."index.$html_ext\">". | 4266 $view = "<a href=\"$base_dir"."index.$html_ext\">". |
| 3709 "$overview_title</a> - ". | 4267 "$overview_title</a> - ". |
| 3710 "<a href=\"index.$html_ext\">". | 4268 "<a href=\"index.$html_ext\">". |
| 3711 » » » » "$dir_name</a> - $base_name"; | 4269 » » » » "$esc_dir_name</a> - $esc_base_name"; |
| 3712 } | 4270 } |
| 3713 | 4271 |
| 3714 # Add function suffix | 4272 # Add function suffix |
| 3715 if ($func_coverage) { | 4273 if ($func_coverage) { |
| 3716 » » » if ($type == 2) { | 4274 » » » $view .= "<span style=\"font-size: 80%;\">"; |
| 4275 » » » if ($type == $HDR_SOURCE) { |
| 3717 $view .= " (source / <a href=\"$base_name.func.$
html_ext\">functions</a>)"; | 4276 $view .= " (source / <a href=\"$base_name.func.$
html_ext\">functions</a>)"; |
| 3718 » » » } elsif ($type == 4) { | 4277 » » » } elsif ($type == $HDR_FUNC) { |
| 3719 $view .= " (<a href=\"$base_name.gcov.$html_ext\
">source</a> / functions)"; | 4278 $view .= " (<a href=\"$base_name.gcov.$html_ext\
">source</a> / functions)"; |
| 3720 } | 4279 } |
| 4280 $view .= "</span>"; |
| 3721 } | 4281 } |
| 3722 } | 4282 } |
| 3723 » elsif ($type == 3) | 4283 » elsif ($type == $HDR_TESTDESC) |
| 3724 { | 4284 { |
| 3725 # Test description header | 4285 # Test description header |
| 3726 $base_dir = ""; | 4286 $base_dir = ""; |
| 3727 $view = "<a href=\"$base_dir"."index.$html_ext\">". | 4287 $view = "<a href=\"$base_dir"."index.$html_ext\">". |
| 3728 "$overview_title</a> - test case descriptions"; | 4288 "$overview_title</a> - test case descriptions"; |
| 3729 } | 4289 } |
| 3730 | 4290 |
| 3731 # Prepare text for "test" field | 4291 # Prepare text for "test" field |
| 3732 $test = escape_html($test_title); | 4292 $test = escape_html($test_title); |
| 3733 | 4293 |
| 3734 # Append link to test description page if available | 4294 # Append link to test description page if available |
| 3735 » if (%test_description && ($type != 3)) | 4295 » if (%test_description && ($type != $HDR_TESTDESC)) |
| 3736 { | 4296 { |
| 3737 » » if ($frames && ($type == 2 || $type == 4)) | 4297 » » if ($frames && ($type == $HDR_SOURCE || $type == $HDR_FUNC)) |
| 3738 { | 4298 { |
| 3739 # Need to break frameset when clicking this link | 4299 # Need to break frameset when clicking this link |
| 3740 » » » $test .= " ( <a href=\"$base_dir". | 4300 » » » $test .= " ( <span style=\"font-size:80%;\">". |
| 4301 » » » » "<a href=\"$base_dir". |
| 3741 "descriptions.$html_ext\" target=\"_parent\">". | 4302 "descriptions.$html_ext\" target=\"_parent\">". |
| 3742 » » » » "view descriptions</a> )"; | 4303 » » » » "view descriptions</a></span> )"; |
| 3743 } | 4304 } |
| 3744 else | 4305 else |
| 3745 { | 4306 { |
| 3746 » » » $test .= " ( <a href=\"$base_dir". | 4307 » » » $test .= " ( <span style=\"font-size:80%;\">". |
| 4308 » » » » "<a href=\"$base_dir". |
| 3747 "descriptions.$html_ext\">". | 4309 "descriptions.$html_ext\">". |
| 3748 » » » » "view descriptions</a> )"; | 4310 » » » » "view descriptions</a></span> )"; |
| 3749 } | 4311 } |
| 3750 } | 4312 } |
| 3751 | 4313 |
| 3752 # Write header | 4314 # Write header |
| 3753 write_header_prolog(*HTML_HANDLE, $base_dir); | 4315 write_header_prolog(*HTML_HANDLE, $base_dir); |
| 3754 » write_header_line(*HTML_HANDLE, 0, "Current view:", $view, | 4316 |
| 3755 » » » "Found", "Hit", "Coverage"); | 4317 » # Left row |
| 3756 » write_header_line(*HTML_HANDLE, 1, "Test:", $test, "Lines:", | 4318 » push(@row_left, [[ "10%", "headerItem", "Current view:" ], |
| 3757 » » » $lines_found, $lines_hit, | 4319 » » » [ "35%", "headerValue", $view ]]); |
| 3758 » » » $rate_name[classify_rate($lines_found, $lines_hit, | 4320 » push(@row_left, [[undef, "headerItem", "Test:"], |
| 3759 » » » » » » $med_limit, $hi_limit)], | 4321 » » » [undef, "headerValue", $test]]); |
| 3760 » » » format_rate($lines_found, $lines_hit)); | 4322 » push(@row_left, [[undef, "headerItem", "Date:"], |
| 4323 » » » [undef, "headerValue", $date]]); |
| 4324 |
| 4325 » # Right row |
| 4326 » if ($legend && ($type == $HDR_SOURCE || $type == $HDR_FUNC)) { |
| 4327 » » my $text = <<END_OF_HTML; |
| 4328 Lines: |
| 4329 <span class="coverLegendCov">hit</span> |
| 4330 <span class="coverLegendNoCov">not hit</span> |
| 4331 END_OF_HTML |
| 4332 » » if ($br_coverage) { |
| 4333 » » » $text .= <<END_OF_HTML; |
| 4334 | Branches: |
| 4335 <span class="coverLegendCov">+</span> taken |
| 4336 <span class="coverLegendNoCov">-</span> not taken |
| 4337 <span class="coverLegendNoCov">#</span> not executed |
| 4338 END_OF_HTML |
| 4339 » » } |
| 4340 » » push(@row_left, [[undef, "headerItem", "Legend:"], |
| 4341 » » » » [undef, "headerValueLeg", $text]]); |
| 4342 » } elsif ($legend && ($type != $HDR_TESTDESC)) { |
| 4343 » » my $text = <<END_OF_HTML; |
| 4344 » Rating: |
| 4345 <span class="coverLegendCovLo" title="Coverage rates below $med_limi
t % are classified as low">low: < $med_limit %</span> |
| 4346 <span class="coverLegendCovMed" title="Coverage rates between $med_l
imit % and $hi_limit % are classified as medium">medium: >= $med_limit %</spa
n> |
| 4347 <span class="coverLegendCovHi" title="Coverage rates of $hi_limit %
and more are classified as high">high: >= $hi_limit %</span> |
| 4348 END_OF_HTML |
| 4349 » » push(@row_left, [[undef, "headerItem", "Legend:"], |
| 4350 » » » » [undef, "headerValueLeg", $text]]); |
| 4351 » } |
| 4352 » if ($type == $HDR_TESTDESC) { |
| 4353 » » push(@row_right, [[ "55%" ]]); |
| 4354 » } else { |
| 4355 » » push(@row_right, [["15%", undef, undef ], |
| 4356 » » » » ["10%", "headerCovTableHead", "Hit" ], |
| 4357 » » » » ["10%", "headerCovTableHead", "Total" ], |
| 4358 » » » » ["15%", "headerCovTableHead", "Coverage"]]); |
| 4359 » } |
| 4360 » # Line coverage |
| 4361 » $style = $rate_name[classify_rate($lines_found, $lines_hit, |
| 4362 » » » » » $med_limit, $hi_limit)]; |
| 4363 » $rate = rate($lines_hit, $lines_found, " %"); |
| 4364 » push(@row_right, [[undef, "headerItem", "Lines:"], |
| 4365 » » » [undef, "headerCovTableEntry", $lines_hit], |
| 4366 » » » [undef, "headerCovTableEntry", $lines_found], |
| 4367 » » » [undef, "headerCovTableEntry$style", $rate]]) |
| 4368 » » » if ($type != $HDR_TESTDESC); |
| 4369 » # Function coverage |
| 3761 if ($func_coverage) { | 4370 if ($func_coverage) { |
| 3762 » » write_header_line(*HTML_HANDLE, 1, "Date:", $date, "Functions:", | 4371 » » $style = $rate_name[classify_rate($fn_found, $fn_hit, |
| 3763 » » » » $fn_found, $fn_hit, | 4372 » » » » » » $fn_med_limit, $fn_hi_limit)]; |
| 3764 » » » » $rate_name[classify_rate($fn_found, | 4373 » » $rate = rate($fn_hit, $fn_found, " %"); |
| 3765 » » » » » » » $fn_hit, | 4374 » » push(@row_right, [[undef, "headerItem", "Functions:"], |
| 3766 » » » » » » » $fn_med_limit, | 4375 » » » » [undef, "headerCovTableEntry", $fn_hit], |
| 3767 » » » » » » » $fn_hi_limit)], | 4376 » » » » [undef, "headerCovTableEntry", $fn_found], |
| 3768 » » » » format_rate($fn_found, $fn_hit)); | 4377 » » » » [undef, "headerCovTableEntry$style", $rate]]) |
| 3769 » } else { | 4378 » » » if ($type != $HDR_TESTDESC); |
| 3770 » » write_header_line(*HTML_HANDLE, 4, "Date:", $date); | |
| 3771 } | 4379 } |
| 3772 » if ($legend) { | 4380 » # Branch coverage |
| 3773 » » if ($type == 0 || $type == 1) { | 4381 » if ($br_coverage) { |
| 3774 » » » my $line_code = get_legend_code("Line coverage:", | 4382 » » $style = $rate_name[classify_rate($br_found, $br_hit, |
| 3775 » » » » » » $med_limit, $hi_limit); | 4383 » » » » » » $br_med_limit, $br_hi_limit)]; |
| 3776 » » » my $func_code = ""; | 4384 » » $rate = rate($br_hit, $br_found, " %"); |
| 4385 » » push(@row_right, [[undef, "headerItem", "Branches:"], |
| 4386 » » » » [undef, "headerCovTableEntry", $br_hit], |
| 4387 » » » » [undef, "headerCovTableEntry", $br_found], |
| 4388 » » » » [undef, "headerCovTableEntry$style", $rate]]) |
| 4389 » » » if ($type != $HDR_TESTDESC); |
| 4390 » } |
| 3777 | 4391 |
| 3778 » » » if ($func_coverage) { | 4392 » # Print rows |
| 3779 » » » » $func_code = get_legend_code( | 4393 » $num_rows = max(scalar(@row_left), scalar(@row_right)); |
| 3780 » » » » » » "Function coverage:", | 4394 » for ($i = 0; $i < $num_rows; $i++) { |
| 3781 » » » » » » $fn_med_limit, | 4395 » » my $left = $row_left[$i]; |
| 3782 » » » » » » $fn_hi_limit); | 4396 » » my $right = $row_right[$i]; |
| 3783 » » » } | 4397 |
| 3784 » » » write_header_line(*HTML_HANDLE, 2, "Colors:", | 4398 » » if (!defined($left)) { |
| 3785 » » » » $line_code, $func_code); | 4399 » » » $left = [[undef, undef, undef], [undef, undef, undef]]; |
| 3786 » » } elsif ($type == 2 || $type == 4) { | |
| 3787 » » » write_header_line(*HTML_HANDLE, 3, "Colors:", | |
| 3788 » » » » » "not hit", "hit"); | |
| 3789 } | 4400 } |
| 4401 if (!defined($right)) { |
| 4402 $right = []; |
| 4403 } |
| 4404 write_header_line(*HTML_HANDLE, @{$left}, |
| 4405 [ $i == 0 ? "5%" : undef, undef, undef], |
| 4406 @{$right}); |
| 3790 } | 4407 } |
| 4408 |
| 4409 # Fourth line |
| 3791 write_header_epilog(*HTML_HANDLE, $base_dir); | 4410 write_header_epilog(*HTML_HANDLE, $base_dir); |
| 3792 } | 4411 } |
| 3793 | 4412 |
| 3794 | 4413 |
| 3795 # | 4414 # |
| 3796 # split_filename(filename) | |
| 3797 # | |
| 3798 # Return (path, filename, extension) for a given FILENAME. | |
| 3799 # | |
| 3800 | |
| 3801 sub split_filename($) | |
| 3802 { | |
| 3803 if (!$_[0]) { return(); } | |
| 3804 my @path_components = split('/', $_[0]); | |
| 3805 my @file_components = split('\.', pop(@path_components)); | |
| 3806 my $extension = pop(@file_components); | |
| 3807 | |
| 3808 return (join("/",@path_components), join(".",@file_components), | |
| 3809 $extension); | |
| 3810 } | |
| 3811 | |
| 3812 # | |
| 3813 # get_sorted_keys(hash_ref, sort_type) | 4415 # get_sorted_keys(hash_ref, sort_type) |
| 3814 # | 4416 # |
| 3815 | 4417 |
| 3816 sub get_sorted_keys($$) | 4418 sub get_sorted_keys($$) |
| 3817 { | 4419 { |
| 3818 my ($hash, $type) = @_; | 4420 my ($hash, $type) = @_; |
| 3819 | 4421 |
| 3820 » if ($type == 0) { | 4422 » if ($type == $SORT_FILE) { |
| 3821 # Sort by name | 4423 # Sort by name |
| 3822 return sort(keys(%{$hash})); | 4424 return sort(keys(%{$hash})); |
| 3823 » } elsif ($type == 1) { | 4425 » } elsif ($type == $SORT_LINE) { |
| 3824 # Sort by line coverage | 4426 # Sort by line coverage |
| 3825 » » return sort({$hash->{$a}[5] <=> $hash->{$b}[5]} keys(%{$hash})); | 4427 » » return sort({$hash->{$a}[7] <=> $hash->{$b}[7]} keys(%{$hash})); |
| 3826 » } elsif ($type == 2) { | 4428 » } elsif ($type == $SORT_FUNC) { |
| 3827 # Sort by function coverage; | 4429 # Sort by function coverage; |
| 3828 » » return sort({$hash->{$a}[6] <=> $hash->{$b}[6]}»keys(%{$hash})); | 4430 » » return sort({$hash->{$a}[8] <=> $hash->{$b}[8]}»keys(%{$hash})); |
| 4431 » } elsif ($type == $SORT_BRANCH) { |
| 4432 » » # Sort by br coverage; |
| 4433 » » return sort({$hash->{$a}[9] <=> $hash->{$b}[9]}»keys(%{$hash})); |
| 3829 } | 4434 } |
| 3830 } | 4435 } |
| 3831 | 4436 |
| 3832 sub get_sort_code($$$) | 4437 sub get_sort_code($$$) |
| 3833 { | 4438 { |
| 3834 my ($link, $alt, $base) = @_; | 4439 my ($link, $alt, $base) = @_; |
| 3835 my $png; | 4440 my $png; |
| 3836 my $link_start; | 4441 my $link_start; |
| 3837 my $link_end; | 4442 my $link_end; |
| 3838 | 4443 |
| (...skipping 12 matching lines...) Expand all Loading... |
| 3851 'alt="'.$alt.'" title="'.$alt.'" border=0>'.$link_end.'</span>'; | 4456 'alt="'.$alt.'" title="'.$alt.'" border=0>'.$link_end.'</span>'; |
| 3852 } | 4457 } |
| 3853 | 4458 |
| 3854 sub get_file_code($$$$) | 4459 sub get_file_code($$$$) |
| 3855 { | 4460 { |
| 3856 my ($type, $text, $sort_button, $base) = @_; | 4461 my ($type, $text, $sort_button, $base) = @_; |
| 3857 my $result = $text; | 4462 my $result = $text; |
| 3858 my $link; | 4463 my $link; |
| 3859 | 4464 |
| 3860 if ($sort_button) { | 4465 if ($sort_button) { |
| 3861 » » if ($type == 1) { | 4466 » » if ($type == $HEAD_NO_DETAIL) { |
| 3862 $link = "index.$html_ext"; | 4467 $link = "index.$html_ext"; |
| 3863 } else { | 4468 } else { |
| 3864 $link = "index-detail.$html_ext"; | 4469 $link = "index-detail.$html_ext"; |
| 3865 } | 4470 } |
| 3866 } | 4471 } |
| 3867 $result .= get_sort_code($link, "Sort by name", $base); | 4472 $result .= get_sort_code($link, "Sort by name", $base); |
| 3868 | 4473 |
| 3869 return $result; | 4474 return $result; |
| 3870 } | 4475 } |
| 3871 | 4476 |
| 3872 sub get_line_code($$$$$) | 4477 sub get_line_code($$$$$) |
| 3873 { | 4478 { |
| 3874 my ($type, $sort_type, $text, $sort_button, $base) = @_; | 4479 my ($type, $sort_type, $text, $sort_button, $base) = @_; |
| 3875 my $result = $text; | 4480 my $result = $text; |
| 3876 my $sort_link; | 4481 my $sort_link; |
| 3877 | 4482 |
| 3878 » if ($type == 1) { | 4483 » if ($type == $HEAD_NO_DETAIL) { |
| 3879 # Just text | 4484 # Just text |
| 3880 if ($sort_button) { | 4485 if ($sort_button) { |
| 3881 $sort_link = "index-sort-l.$html_ext"; | 4486 $sort_link = "index-sort-l.$html_ext"; |
| 3882 } | 4487 } |
| 3883 » } elsif ($type == 2) { | 4488 » } elsif ($type == $HEAD_DETAIL_HIDDEN) { |
| 3884 # Text + link to detail view | 4489 # Text + link to detail view |
| 3885 $result .= ' ( <a class="detail" href="index-detail'. | 4490 $result .= ' ( <a class="detail" href="index-detail'. |
| 3886 $fileview_sortname[$sort_type].'.'.$html_ext. | 4491 $fileview_sortname[$sort_type].'.'.$html_ext. |
| 3887 '">show details</a> )'; | 4492 '">show details</a> )'; |
| 3888 if ($sort_button) { | 4493 if ($sort_button) { |
| 3889 $sort_link = "index-sort-l.$html_ext"; | 4494 $sort_link = "index-sort-l.$html_ext"; |
| 3890 } | 4495 } |
| 3891 } else { | 4496 } else { |
| 3892 # Text + link to standard view | 4497 # Text + link to standard view |
| 3893 $result .= ' ( <a class="detail" href="index'. | 4498 $result .= ' ( <a class="detail" href="index'. |
| 3894 $fileview_sortname[$sort_type].'.'.$html_ext. | 4499 $fileview_sortname[$sort_type].'.'.$html_ext. |
| 3895 '">hide details</a> )'; | 4500 '">hide details</a> )'; |
| 3896 if ($sort_button) { | 4501 if ($sort_button) { |
| 3897 $sort_link = "index-detail-sort-l.$html_ext"; | 4502 $sort_link = "index-detail-sort-l.$html_ext"; |
| 3898 } | 4503 } |
| 3899 } | 4504 } |
| 3900 # Add sort button | 4505 # Add sort button |
| 3901 $result .= get_sort_code($sort_link, "Sort by line coverage", $base); | 4506 $result .= get_sort_code($sort_link, "Sort by line coverage", $base); |
| 3902 | 4507 |
| 3903 return $result; | 4508 return $result; |
| 3904 } | 4509 } |
| 3905 | 4510 |
| 3906 sub get_func_code($$$$) | 4511 sub get_func_code($$$$) |
| 3907 { | 4512 { |
| 3908 my ($type, $text, $sort_button, $base) = @_; | 4513 my ($type, $text, $sort_button, $base) = @_; |
| 3909 my $result = $text; | 4514 my $result = $text; |
| 3910 my $link; | 4515 my $link; |
| 3911 | 4516 |
| 3912 if ($sort_button) { | 4517 if ($sort_button) { |
| 3913 » » if ($type == 1) { | 4518 » » if ($type == $HEAD_NO_DETAIL) { |
| 3914 $link = "index-sort-f.$html_ext"; | 4519 $link = "index-sort-f.$html_ext"; |
| 3915 } else { | 4520 } else { |
| 3916 $link = "index-detail-sort-f.$html_ext"; | 4521 $link = "index-detail-sort-f.$html_ext"; |
| 3917 } | 4522 } |
| 3918 } | 4523 } |
| 3919 $result .= get_sort_code($link, "Sort by function coverage", $base); | 4524 $result .= get_sort_code($link, "Sort by function coverage", $base); |
| 3920 return $result; | 4525 return $result; |
| 3921 } | 4526 } |
| 3922 | 4527 |
| 4528 sub get_br_code($$$$) |
| 4529 { |
| 4530 my ($type, $text, $sort_button, $base) = @_; |
| 4531 my $result = $text; |
| 4532 my $link; |
| 4533 |
| 4534 if ($sort_button) { |
| 4535 if ($type == $HEAD_NO_DETAIL) { |
| 4536 $link = "index-sort-b.$html_ext"; |
| 4537 } else { |
| 4538 $link = "index-detail-sort-b.$html_ext"; |
| 4539 } |
| 4540 } |
| 4541 $result .= get_sort_code($link, "Sort by branch coverage", $base); |
| 4542 return $result; |
| 4543 } |
| 4544 |
| 3923 # | 4545 # |
| 3924 # write_file_table(filehandle, base_dir, overview, testhash, testfnchash, | 4546 # write_file_table(filehandle, base_dir, overview, testhash, testfnchash, |
| 3925 # fileview, sort_type) | 4547 # testbrhash, fileview, sort_type) |
| 3926 # | 4548 # |
| 3927 # Write a complete file table. OVERVIEW is a reference to a hash containing | 4549 # Write a complete file table. OVERVIEW is a reference to a hash containing |
| 3928 # the following mapping: | 4550 # the following mapping: |
| 3929 # | 4551 # |
| 3930 # filename -> "lines_found,lines_hit,funcs_found,funcs_hit,page_link, | 4552 # filename -> "lines_found,lines_hit,funcs_found,funcs_hit,page_link, |
| 3931 # func_link" | 4553 # func_link" |
| 3932 # | 4554 # |
| 3933 # TESTHASH is a reference to the following hash: | 4555 # TESTHASH is a reference to the following hash: |
| 3934 # | 4556 # |
| 3935 # filename -> \%testdata | 4557 # filename -> \%testdata |
| 3936 # %testdata: name of test affecting this file -> \%testcount | 4558 # %testdata: name of test affecting this file -> \%testcount |
| 3937 # %testcount: line number -> execution count for a single test | 4559 # %testcount: line number -> execution count for a single test |
| 3938 # | 4560 # |
| 3939 # Heading of first column is "Filename" if FILEVIEW is true, "Directory name" | 4561 # Heading of first column is "Filename" if FILEVIEW is true, "Directory name" |
| 3940 # otherwise. | 4562 # otherwise. |
| 3941 # | 4563 # |
| 3942 | 4564 |
| 3943 sub write_file_table(*$$$$$$) | 4565 sub write_file_table(*$$$$$$$) |
| 3944 { | 4566 { |
| 3945 local *HTML_HANDLE = $_[0]; | 4567 local *HTML_HANDLE = $_[0]; |
| 3946 my $base_dir = $_[1]; | 4568 my $base_dir = $_[1]; |
| 3947 my $overview = $_[2]; | 4569 my $overview = $_[2]; |
| 3948 my $testhash = $_[3]; | 4570 my $testhash = $_[3]; |
| 3949 my $testfnchash = $_[4]; | 4571 my $testfnchash = $_[4]; |
| 3950 » my $fileview = $_[5]; | 4572 » my $testbrhash = $_[5]; |
| 3951 » my $sort_type = $_[6]; | 4573 » my $fileview = $_[6]; |
| 4574 » my $sort_type = $_[7]; |
| 3952 my $filename; | 4575 my $filename; |
| 3953 my $bar_graph; | 4576 my $bar_graph; |
| 3954 my $hit; | 4577 my $hit; |
| 3955 my $found; | 4578 my $found; |
| 3956 my $fn_found; | 4579 my $fn_found; |
| 3957 my $fn_hit; | 4580 my $fn_hit; |
| 4581 my $br_found; |
| 4582 my $br_hit; |
| 3958 my $page_link; | 4583 my $page_link; |
| 3959 my $testname; | 4584 my $testname; |
| 3960 my $testdata; | 4585 my $testdata; |
| 3961 my $testfncdata; | 4586 my $testfncdata; |
| 3962 » my $testcount; | 4587 » my $testbrdata; |
| 3963 » my $testfnccount; | |
| 3964 my %affecting_tests; | 4588 my %affecting_tests; |
| 3965 my $line_code = ""; | 4589 my $line_code = ""; |
| 3966 my $func_code; | 4590 my $func_code; |
| 4591 my $br_code; |
| 3967 my $file_code; | 4592 my $file_code; |
| 4593 my @head_columns; |
| 3968 | 4594 |
| 3969 # Determine HTML code for column headings | 4595 # Determine HTML code for column headings |
| 3970 if (($base_dir ne "") && $show_details) | 4596 if (($base_dir ne "") && $show_details) |
| 3971 { | 4597 { |
| 3972 my $detailed = keys(%{$testhash}); | 4598 my $detailed = keys(%{$testhash}); |
| 3973 | 4599 |
| 3974 » » $file_code = get_file_code($detailed ? 2 : 1, | 4600 » » $file_code = get_file_code($detailed ? $HEAD_DETAIL_HIDDEN : |
| 4601 » » » » » $HEAD_NO_DETAIL, |
| 3975 $fileview ? "Filename" : "Directory", | 4602 $fileview ? "Filename" : "Directory", |
| 3976 » » » » » $sort && $sort_type != 0, $base_dir); | 4603 » » » » » $sort && $sort_type != $SORT_FILE, |
| 3977 » » $line_code = get_line_code($detailed ? 3 : 2, $sort_type, | 4604 » » » » » $base_dir); |
| 4605 » » $line_code = get_line_code($detailed ? $HEAD_DETAIL_SHOWN : |
| 4606 » » » » » $HEAD_DETAIL_HIDDEN, |
| 4607 » » » » » $sort_type, |
| 3978 "Line Coverage", | 4608 "Line Coverage", |
| 3979 » » » » » $sort && $sort_type != 1, $base_dir); | 4609 » » » » » $sort && $sort_type != $SORT_LINE, |
| 3980 » » $func_code = get_func_code($detailed ? 2 : 1, "Functions", | 4610 » » » » » $base_dir); |
| 3981 » » » » » $sort && $sort_type != 2, $base_dir); | 4611 » » $func_code = get_func_code($detailed ? $HEAD_DETAIL_HIDDEN : |
| 4612 » » » » » $HEAD_NO_DETAIL, |
| 4613 » » » » » "Functions", |
| 4614 » » » » » $sort && $sort_type != $SORT_FUNC, |
| 4615 » » » » » $base_dir); |
| 4616 » » $br_code = get_br_code($detailed ? $HEAD_DETAIL_HIDDEN : |
| 4617 » » » » » $HEAD_NO_DETAIL, |
| 4618 » » » » » "Branches", |
| 4619 » » » » » $sort && $sort_type != $SORT_BRANCH, |
| 4620 » » » » » $base_dir); |
| 3982 } else { | 4621 } else { |
| 3983 » » $file_code = get_file_code(1, | 4622 » » $file_code = get_file_code($HEAD_NO_DETAIL, |
| 3984 $fileview ? "Filename" : "Directory", | 4623 $fileview ? "Filename" : "Directory", |
| 3985 » » » » » $sort && $sort_type != 0, $base_dir); | 4624 » » » » » $sort && $sort_type != $SORT_FILE, |
| 3986 » » $line_code = get_line_code(1, $sort_type, "Line Coverage", | 4625 » » » » » $base_dir); |
| 3987 » » » » » $sort && $sort_type != 1, $base_dir); | 4626 » » $line_code = get_line_code($HEAD_NO_DETAIL, $sort_type, "Line Co
verage", |
| 3988 » » $func_code = get_func_code(1, "Functions", | 4627 » » » » » $sort && $sort_type != $SORT_LINE, |
| 3989 » » » » » $sort && $sort_type != 2, $base_dir); | 4628 » » » » » $base_dir); |
| 4629 » » $func_code = get_func_code($HEAD_NO_DETAIL, "Functions", |
| 4630 » » » » » $sort && $sort_type != $SORT_FUNC, |
| 4631 » » » » » $base_dir); |
| 4632 » » $br_code = get_br_code($HEAD_NO_DETAIL, "Branches", |
| 4633 » » » » » $sort && $sort_type != $SORT_BRANCH, |
| 4634 » » » » » $base_dir); |
| 3990 } | 4635 } |
| 4636 push(@head_columns, [ $line_code, 3 ]); |
| 4637 push(@head_columns, [ $func_code, 2]) if ($func_coverage); |
| 4638 push(@head_columns, [ $br_code, 2]) if ($br_coverage); |
| 3991 | 4639 |
| 3992 » write_file_table_prolog(*HTML_HANDLE, $file_code, $line_code, | 4640 » write_file_table_prolog(*HTML_HANDLE, $file_code, @head_columns); |
| 3993 » » » » $func_code); | |
| 3994 | 4641 |
| 3995 foreach $filename (get_sorted_keys($overview, $sort_type)) | 4642 foreach $filename (get_sorted_keys($overview, $sort_type)) |
| 3996 { | 4643 { |
| 3997 » » ($found, $hit, $fn_found, $fn_hit, $page_link) | 4644 » » my @columns; |
| 3998 » » » = @{$overview->{$filename}}; | 4645 » » ($found, $hit, $fn_found, $fn_hit, $br_found, $br_hit, |
| 3999 » » $bar_graph = get_bar_graph_code($base_dir, $found, $hit); | 4646 » » $page_link) = @{$overview->{$filename}}; |
| 4647 |
| 4648 » » # Line coverage |
| 4649 » » push(@columns, [$found, $hit, $med_limit, $hi_limit, 1]); |
| 4650 » » # Function coverage |
| 4651 » » if ($func_coverage) { |
| 4652 » » » push(@columns, [$fn_found, $fn_hit, $fn_med_limit, |
| 4653 » » » » » $fn_hi_limit, 0]); |
| 4654 » » } |
| 4655 » » # Branch coverage |
| 4656 » » if ($br_coverage) { |
| 4657 » » » push(@columns, [$br_found, $br_hit, $br_med_limit, |
| 4658 » » » » » $br_hi_limit, 0]); |
| 4659 » » } |
| 4660 » » write_file_table_entry(*HTML_HANDLE, $base_dir, $filename, |
| 4661 » » » » $page_link, @columns); |
| 4000 | 4662 |
| 4001 $testdata = $testhash->{$filename}; | 4663 $testdata = $testhash->{$filename}; |
| 4002 $testfncdata = $testfnchash->{$filename}; | 4664 $testfncdata = $testfnchash->{$filename}; |
| 4003 | 4665 » » $testbrdata = $testbrhash->{$filename}; |
| 4004 » » write_file_table_entry(*HTML_HANDLE, $filename, $bar_graph, | |
| 4005 » » » » $found, $hit, $fn_found, $fn_hit, | |
| 4006 » » » » $page_link); | |
| 4007 | 4666 |
| 4008 # Check whether we should write test specific coverage | 4667 # Check whether we should write test specific coverage |
| 4009 # as well | 4668 # as well |
| 4010 if (!($show_details && $testdata)) { next; } | 4669 if (!($show_details && $testdata)) { next; } |
| 4011 | 4670 |
| 4012 # Filter out those tests that actually affect this file | 4671 # Filter out those tests that actually affect this file |
| 4013 %affecting_tests = %{ get_affecting_tests($testdata, | 4672 %affecting_tests = %{ get_affecting_tests($testdata, |
| 4014 » » » » » $testfncdata) }; | 4673 » » » » » $testfncdata, $testbrdata) }; |
| 4015 | 4674 |
| 4016 # Does any of the tests affect this file at all? | 4675 # Does any of the tests affect this file at all? |
| 4017 if (!%affecting_tests) { next; } | 4676 if (!%affecting_tests) { next; } |
| 4018 | 4677 |
| 4019 # Write test details for this entry | |
| 4020 write_file_table_detail_heading(*HTML_HANDLE, "Test name", | |
| 4021 "Lines hit", "Functions hit"); | |
| 4022 | |
| 4023 foreach $testname (keys(%affecting_tests)) | 4678 foreach $testname (keys(%affecting_tests)) |
| 4024 { | 4679 { |
| 4025 » » » ($found, $hit, $fn_found, $fn_hit) = | 4680 » » » my @results; |
| 4681 » » » ($found, $hit, $fn_found, $fn_hit, $br_found, $br_hit) = |
| 4026 split(",", $affecting_tests{$testname}); | 4682 split(",", $affecting_tests{$testname}); |
| 4027 | 4683 |
| 4028 # Insert link to description of available | 4684 # Insert link to description of available |
| 4029 if ($test_description{$testname}) | 4685 if ($test_description{$testname}) |
| 4030 { | 4686 { |
| 4031 $testname = "<a href=\"$base_dir". | 4687 $testname = "<a href=\"$base_dir". |
| 4032 "descriptions.$html_ext#$testname\">
". | 4688 "descriptions.$html_ext#$testname\">
". |
| 4033 "$testname</a>"; | 4689 "$testname</a>"; |
| 4034 } | 4690 } |
| 4035 | 4691 |
| 4692 push(@results, [$found, $hit]); |
| 4693 push(@results, [$fn_found, $fn_hit]) if ($func_coverage)
; |
| 4694 push(@results, [$br_found, $br_hit]) if ($br_coverage); |
| 4036 write_file_table_detail_entry(*HTML_HANDLE, $testname, | 4695 write_file_table_detail_entry(*HTML_HANDLE, $testname, |
| 4037 » » » » $found, $hit, $fn_found, $fn_hit); | 4696 » » » » @results); |
| 4038 } | 4697 } |
| 4039 } | 4698 } |
| 4040 | 4699 |
| 4041 write_file_table_epilog(*HTML_HANDLE); | 4700 write_file_table_epilog(*HTML_HANDLE); |
| 4042 } | 4701 } |
| 4043 | 4702 |
| 4044 | 4703 |
| 4045 # | 4704 # |
| 4046 # get_found_and_hit(hash) | 4705 # get_found_and_hit(hash) |
| 4047 # | 4706 # |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4088 foreach $function (keys(%{$sumfnccount})) { | 4747 foreach $function (keys(%{$sumfnccount})) { |
| 4089 if ($sumfnccount->{$function} > 0) { | 4748 if ($sumfnccount->{$function} > 0) { |
| 4090 $fn_hit++; | 4749 $fn_hit++; |
| 4091 } | 4750 } |
| 4092 } | 4751 } |
| 4093 return ($fn_found, $fn_hit); | 4752 return ($fn_found, $fn_hit); |
| 4094 } | 4753 } |
| 4095 | 4754 |
| 4096 | 4755 |
| 4097 # | 4756 # |
| 4098 # get_affecting_tests(testdata, testfncdata) | 4757 # br_taken_to_num(taken) |
| 4758 # |
| 4759 # Convert a branch taken value .info format to number format. |
| 4760 # |
| 4761 |
| 4762 sub br_taken_to_num($) |
| 4763 { |
| 4764 » my ($taken) = @_; |
| 4765 |
| 4766 » return 0 if ($taken eq '-'); |
| 4767 » return $taken + 1; |
| 4768 } |
| 4769 |
| 4770 |
| 4771 # |
| 4772 # br_num_to_taken(taken) |
| 4773 # |
| 4774 # Convert a branch taken value in number format to .info format. |
| 4775 # |
| 4776 |
| 4777 sub br_num_to_taken($) |
| 4778 { |
| 4779 » my ($taken) = @_; |
| 4780 |
| 4781 » return '-' if ($taken == 0); |
| 4782 » return $taken - 1; |
| 4783 } |
| 4784 |
| 4785 |
| 4786 # |
| 4787 # br_taken_add(taken1, taken2) |
| 4788 # |
| 4789 # Return the result of taken1 + taken2 for 'branch taken' values. |
| 4790 # |
| 4791 |
| 4792 sub br_taken_add($$) |
| 4793 { |
| 4794 » my ($t1, $t2) = @_; |
| 4795 |
| 4796 » return $t1 if (!defined($t2)); |
| 4797 » return $t2 if (!defined($t1)); |
| 4798 » return $t1 if ($t2 eq '-'); |
| 4799 » return $t2 if ($t1 eq '-'); |
| 4800 » return $t1 + $t2; |
| 4801 } |
| 4802 |
| 4803 |
| 4804 # |
| 4805 # br_taken_sub(taken1, taken2) |
| 4806 # |
| 4807 # Return the result of taken1 - taken2 for 'branch taken' values. Return 0 |
| 4808 # if the result would become negative. |
| 4809 # |
| 4810 |
| 4811 sub br_taken_sub($$) |
| 4812 { |
| 4813 » my ($t1, $t2) = @_; |
| 4814 |
| 4815 » return $t1 if (!defined($t2)); |
| 4816 » return undef if (!defined($t1)); |
| 4817 » return $t1 if ($t1 eq '-'); |
| 4818 » return $t1 if ($t2 eq '-'); |
| 4819 » return 0 if $t2 > $t1; |
| 4820 » return $t1 - $t2; |
| 4821 } |
| 4822 |
| 4823 |
| 4824 # |
| 4825 # br_ivec_len(vector) |
| 4826 # |
| 4827 # Return the number of entries in the branch coverage vector. |
| 4828 # |
| 4829 |
| 4830 sub br_ivec_len($) |
| 4831 { |
| 4832 » my ($vec) = @_; |
| 4833 |
| 4834 » return 0 if (!defined($vec)); |
| 4835 » return (length($vec) * 8 / $BR_VEC_WIDTH) / $BR_VEC_ENTRIES; |
| 4836 } |
| 4837 |
| 4838 |
| 4839 # |
| 4840 # br_ivec_get(vector, number) |
| 4841 # |
| 4842 # Return an entry from the branch coverage vector. |
| 4843 # |
| 4844 |
| 4845 sub br_ivec_get($$) |
| 4846 { |
| 4847 » my ($vec, $num) = @_; |
| 4848 » my $block; |
| 4849 » my $branch; |
| 4850 » my $taken; |
| 4851 » my $offset = $num * $BR_VEC_ENTRIES; |
| 4852 |
| 4853 » # Retrieve data from vector |
| 4854 » $block» = vec($vec, $offset + $BR_BLOCK, $BR_VEC_WIDTH); |
| 4855 » $branch = vec($vec, $offset + $BR_BRANCH, $BR_VEC_WIDTH); |
| 4856 » $taken» = vec($vec, $offset + $BR_TAKEN, $BR_VEC_WIDTH); |
| 4857 |
| 4858 » # Decode taken value from an integer |
| 4859 » $taken = br_num_to_taken($taken); |
| 4860 |
| 4861 » return ($block, $branch, $taken); |
| 4862 } |
| 4863 |
| 4864 |
| 4865 # |
| 4866 # br_ivec_push(vector, block, branch, taken) |
| 4867 # |
| 4868 # Add an entry to the branch coverage vector. If an entry with the same |
| 4869 # branch ID already exists, add the corresponding taken values. |
| 4870 # |
| 4871 |
| 4872 sub br_ivec_push($$$$) |
| 4873 { |
| 4874 » my ($vec, $block, $branch, $taken) = @_; |
| 4875 » my $offset; |
| 4876 » my $num = br_ivec_len($vec); |
| 4877 » my $i; |
| 4878 |
| 4879 » $vec = "" if (!defined($vec)); |
| 4880 |
| 4881 » # Check if branch already exists in vector |
| 4882 » for ($i = 0; $i < $num; $i++) { |
| 4883 » » my ($v_block, $v_branch, $v_taken) = br_ivec_get($vec, $i); |
| 4884 |
| 4885 » » next if ($v_block != $block || $v_branch != $branch); |
| 4886 |
| 4887 » » # Add taken counts |
| 4888 » » $taken = br_taken_add($taken, $v_taken); |
| 4889 » » last; |
| 4890 » } |
| 4891 |
| 4892 » $offset = $i * $BR_VEC_ENTRIES; |
| 4893 » $taken = br_taken_to_num($taken); |
| 4894 |
| 4895 » # Add to vector |
| 4896 » vec($vec, $offset + $BR_BLOCK, $BR_VEC_WIDTH) = $block; |
| 4897 » vec($vec, $offset + $BR_BRANCH, $BR_VEC_WIDTH) = $branch; |
| 4898 » vec($vec, $offset + $BR_TAKEN, $BR_VEC_WIDTH) = $taken; |
| 4899 |
| 4900 » return $vec; |
| 4901 } |
| 4902 |
| 4903 |
| 4904 # |
| 4905 # get_br_found_and_hit(sumbrcount) |
| 4906 # |
| 4907 # Return (br_found, br_hit) for sumbrcount |
| 4908 # |
| 4909 |
| 4910 sub get_br_found_and_hit($) |
| 4911 { |
| 4912 » my ($sumbrcount) = @_; |
| 4913 » my $line; |
| 4914 » my $br_found = 0; |
| 4915 » my $br_hit = 0; |
| 4916 |
| 4917 » foreach $line (keys(%{$sumbrcount})) { |
| 4918 » » my $brdata = $sumbrcount->{$line}; |
| 4919 » » my $i; |
| 4920 » » my $num = br_ivec_len($brdata); |
| 4921 |
| 4922 » » for ($i = 0; $i < $num; $i++) { |
| 4923 » » » my $taken; |
| 4924 |
| 4925 » » » (undef, undef, $taken) = br_ivec_get($brdata, $i); |
| 4926 |
| 4927 » » » $br_found++; |
| 4928 » » » $br_hit++ if ($taken ne "-" && $taken > 0); |
| 4929 » » } |
| 4930 » } |
| 4931 |
| 4932 » return ($br_found, $br_hit); |
| 4933 } |
| 4934 |
| 4935 |
| 4936 # |
| 4937 # get_affecting_tests(testdata, testfncdata, testbrdata) |
| 4099 # | 4938 # |
| 4100 # HASHREF contains a mapping filename -> (linenumber -> exec count). Return | 4939 # HASHREF contains a mapping filename -> (linenumber -> exec count). Return |
| 4101 # a hash containing mapping filename -> "lines found, lines hit" for each | 4940 # a hash containing mapping filename -> "lines found, lines hit" for each |
| 4102 # filename which has a nonzero hit count. | 4941 # filename which has a nonzero hit count. |
| 4103 # | 4942 # |
| 4104 | 4943 |
| 4105 sub get_affecting_tests($$) | 4944 sub get_affecting_tests($$$) |
| 4106 { | 4945 { |
| 4107 » my $testdata = $_[0]; | 4946 » my ($testdata, $testfncdata, $testbrdata) = @_; |
| 4108 » my $testfncdata = $_[1]; | |
| 4109 my $testname; | 4947 my $testname; |
| 4110 my $testcount; | 4948 my $testcount; |
| 4111 my $testfnccount; | 4949 my $testfnccount; |
| 4950 my $testbrcount; |
| 4112 my %result; | 4951 my %result; |
| 4113 my $found; | 4952 my $found; |
| 4114 my $hit; | 4953 my $hit; |
| 4115 my $fn_found; | 4954 my $fn_found; |
| 4116 my $fn_hit; | 4955 my $fn_hit; |
| 4956 my $br_found; |
| 4957 my $br_hit; |
| 4117 | 4958 |
| 4118 foreach $testname (keys(%{$testdata})) | 4959 foreach $testname (keys(%{$testdata})) |
| 4119 { | 4960 { |
| 4120 # Get (line number -> count) hash for this test case | 4961 # Get (line number -> count) hash for this test case |
| 4121 $testcount = $testdata->{$testname}; | 4962 $testcount = $testdata->{$testname}; |
| 4122 $testfnccount = $testfncdata->{$testname}; | 4963 $testfnccount = $testfncdata->{$testname}; |
| 4964 $testbrcount = $testbrdata->{$testname}; |
| 4123 | 4965 |
| 4124 # Calculate sum | 4966 # Calculate sum |
| 4125 ($found, $hit) = get_found_and_hit($testcount); | 4967 ($found, $hit) = get_found_and_hit($testcount); |
| 4126 ($fn_found, $fn_hit) = get_func_found_and_hit($testfnccount); | 4968 ($fn_found, $fn_hit) = get_func_found_and_hit($testfnccount); |
| 4969 ($br_found, $br_hit) = get_br_found_and_hit($testbrcount); |
| 4127 | 4970 |
| 4128 if ($hit>0) | 4971 if ($hit>0) |
| 4129 { | 4972 { |
| 4130 » » » $result{$testname} = "$found,$hit,$fn_found,$fn_hit"; | 4973 » » » $result{$testname} = "$found,$hit,$fn_found,$fn_hit,". |
| 4974 » » » » » "$br_found,$br_hit"; |
| 4131 } | 4975 } |
| 4132 } | 4976 } |
| 4133 | 4977 |
| 4134 return(\%result); | 4978 return(\%result); |
| 4135 } | 4979 } |
| 4136 | 4980 |
| 4137 | 4981 |
| 4138 sub get_hash_reverse($) | 4982 sub get_hash_reverse($) |
| 4139 { | 4983 { |
| 4140 my ($hash) = @_; | 4984 my ($hash) = @_; |
| 4141 my %result; | 4985 my %result; |
| 4142 | 4986 |
| 4143 foreach (keys(%{$hash})) { | 4987 foreach (keys(%{$hash})) { |
| 4144 $result{$hash->{$_}} = $_; | 4988 $result{$hash->{$_}} = $_; |
| 4145 } | 4989 } |
| 4146 | 4990 |
| 4147 return \%result; | 4991 return \%result; |
| 4148 } | 4992 } |
| 4149 | 4993 |
| 4150 # | 4994 # |
| 4151 # write_source(filehandle, source_filename, count_data, checksum_data, | 4995 # write_source(filehandle, source_filename, count_data, checksum_data, |
| 4152 # converted_data, func_data) | 4996 # converted_data, func_data, sumbrcount) |
| 4153 # | 4997 # |
| 4154 # Write an HTML view of a source code file. Returns a list containing | 4998 # Write an HTML view of a source code file. Returns a list containing |
| 4155 # data as needed by gen_png(). | 4999 # data as needed by gen_png(). |
| 4156 # | 5000 # |
| 4157 # Die on error. | 5001 # Die on error. |
| 4158 # | 5002 # |
| 4159 | 5003 |
| 4160 sub write_source($$$$$$) | 5004 sub write_source($$$$$$$) |
| 4161 { | 5005 { |
| 4162 local *HTML_HANDLE = $_[0]; | 5006 local *HTML_HANDLE = $_[0]; |
| 4163 local *SOURCE_HANDLE; | 5007 local *SOURCE_HANDLE; |
| 4164 my $source_filename = $_[1]; | 5008 my $source_filename = $_[1]; |
| 4165 my %count_data; | 5009 my %count_data; |
| 4166 my $line_number; | 5010 my $line_number; |
| 4167 my @result; | 5011 my @result; |
| 4168 my $checkdata = $_[3]; | 5012 my $checkdata = $_[3]; |
| 4169 my $converted = $_[4]; | 5013 my $converted = $_[4]; |
| 4170 my $funcdata = $_[5]; | 5014 my $funcdata = $_[5]; |
| 5015 my $sumbrcount = $_[6]; |
| 4171 my $datafunc = get_hash_reverse($funcdata); | 5016 my $datafunc = get_hash_reverse($funcdata); |
| 4172 my $add_anchor; | 5017 my $add_anchor; |
| 5018 my @file; |
| 4173 | 5019 |
| 4174 if ($_[2]) | 5020 if ($_[2]) |
| 4175 { | 5021 { |
| 4176 %count_data = %{$_[2]}; | 5022 %count_data = %{$_[2]}; |
| 4177 } | 5023 } |
| 4178 | 5024 |
| 4179 » open(SOURCE_HANDLE, "<".$source_filename) | 5025 » if (!open(SOURCE_HANDLE, "<", $source_filename)) { |
| 4180 # or die("ERROR: cannot open $source_filename for reading!\n"); | 5026 » » my @lines; |
| 4181 or open(SOURCE_HANDLE, "</dev/null"); | 5027 » » my $last_line = 0; |
| 5028 |
| 5029 » » if (!$ignore[$ERROR_SOURCE]) { |
| 5030 » » » die("ERROR: cannot read $source_filename\n"); |
| 5031 » » } |
| 5032 |
| 5033 » » # Continue without source file |
| 5034 » » warn("WARNING: cannot read $source_filename!\n"); |
| 5035 |
| 5036 » » @lines = sort( { $a <=> $b } keys(%count_data)); |
| 5037 » » if (@lines) { |
| 5038 » » » $last_line = $lines[scalar(@lines) - 1]; |
| 5039 » » } |
| 5040 » » return ( ":" ) if ($last_line < 1); |
| 5041 |
| 5042 » » # Simulate gcov behavior |
| 5043 » » for ($line_number = 1; $line_number <= $last_line; |
| 5044 » » $line_number++) { |
| 5045 » » » push(@file, "/* EOF */"); |
| 5046 » » } |
| 5047 » } else { |
| 5048 » » @file = <SOURCE_HANDLE>; |
| 5049 » } |
| 4182 | 5050 |
| 4183 write_source_prolog(*HTML_HANDLE); | 5051 write_source_prolog(*HTML_HANDLE); |
| 5052 $line_number = 0; |
| 5053 foreach (@file) { |
| 5054 $line_number++; |
| 5055 chomp($_); |
| 4184 | 5056 |
| 4185 » for ($line_number = 1; <SOURCE_HANDLE> ; $line_number++) | 5057 » » # Also remove CR from line-end |
| 4186 » { | 5058 » » s/\015$//; |
| 4187 » » chomp($_); | |
| 4188 | 5059 |
| 4189 # Source code matches coverage data? | 5060 # Source code matches coverage data? |
| 4190 if (defined($checkdata->{$line_number}) && | 5061 if (defined($checkdata->{$line_number}) && |
| 4191 ($checkdata->{$line_number} ne md5_base64($_))) | 5062 ($checkdata->{$line_number} ne md5_base64($_))) |
| 4192 { | 5063 { |
| 4193 die("ERROR: checksum mismatch at $source_filename:". | 5064 die("ERROR: checksum mismatch at $source_filename:". |
| 4194 "$line_number\n"); | 5065 "$line_number\n"); |
| 4195 } | 5066 } |
| 4196 | 5067 |
| 4197 $add_anchor = 0; | 5068 $add_anchor = 0; |
| 4198 if ($frames) { | 5069 if ($frames) { |
| 4199 if (($line_number - 1) % $nav_resolution == 0) { | 5070 if (($line_number - 1) % $nav_resolution == 0) { |
| 4200 $add_anchor = 1; | 5071 $add_anchor = 1; |
| 4201 } | 5072 } |
| 4202 } | 5073 } |
| 4203 if ($func_coverage) { | 5074 if ($func_coverage) { |
| 4204 if ($line_number == 1) { | 5075 if ($line_number == 1) { |
| 4205 $add_anchor = 1; | 5076 $add_anchor = 1; |
| 4206 } elsif (defined($datafunc->{$line_number + | 5077 } elsif (defined($datafunc->{$line_number + |
| 4207 $func_offset})) { | 5078 $func_offset})) { |
| 4208 $add_anchor = 1; | 5079 $add_anchor = 1; |
| 4209 } | 5080 } |
| 4210 } | 5081 } |
| 4211 push (@result, | 5082 push (@result, |
| 4212 write_source_line(HTML_HANDLE, $line_number, | 5083 write_source_line(HTML_HANDLE, $line_number, |
| 4213 $_, $count_data{$line_number}, | 5084 $_, $count_data{$line_number}, |
| 4214 $converted->{$line_number}, | 5085 $converted->{$line_number}, |
| 4215 » » » » » $add_anchor)); | 5086 » » » » » $sumbrcount->{$line_number}, $add_anchor
)); |
| 4216 } | 5087 } |
| 4217 | 5088 |
| 4218 close(SOURCE_HANDLE); | 5089 close(SOURCE_HANDLE); |
| 4219 write_source_epilog(*HTML_HANDLE); | 5090 write_source_epilog(*HTML_HANDLE); |
| 4220 return(@result); | 5091 return(@result); |
| 4221 } | 5092 } |
| 4222 | 5093 |
| 4223 | 5094 |
| 4224 sub funcview_get_func_code($$$) | 5095 sub funcview_get_func_code($$$) |
| 4225 { | 5096 { |
| (...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4264 | 5135 |
| 4265 if ($type == 0) { | 5136 if ($type == 0) { |
| 4266 return sort(keys(%{$funcdata})); | 5137 return sort(keys(%{$funcdata})); |
| 4267 } | 5138 } |
| 4268 return sort({$sumfncdata->{$b} <=> $sumfncdata->{$a}} | 5139 return sort({$sumfncdata->{$b} <=> $sumfncdata->{$a}} |
| 4269 keys(%{$sumfncdata})); | 5140 keys(%{$sumfncdata})); |
| 4270 } | 5141 } |
| 4271 | 5142 |
| 4272 # | 5143 # |
| 4273 # write_function_table(filehandle, source_file, sumcount, funcdata, | 5144 # write_function_table(filehandle, source_file, sumcount, funcdata, |
| 4274 #» » sumfnccount, testfncdata) | 5145 #» » sumfnccount, testfncdata, sumbrcount, testbrdata, |
| 5146 #» » base_name, base_dir, sort_type) |
| 4275 # | 5147 # |
| 4276 # Write an HTML table listing all functions in a source file, including | 5148 # Write an HTML table listing all functions in a source file, including |
| 4277 # also function call counts and line coverages inside of each function. | 5149 # also function call counts and line coverages inside of each function. |
| 4278 # | 5150 # |
| 4279 # Die on error. | 5151 # Die on error. |
| 4280 # | 5152 # |
| 4281 | 5153 |
| 4282 sub write_function_table(*$$$$$$$$) | 5154 sub write_function_table(*$$$$$$$$$$) |
| 4283 { | 5155 { |
| 4284 local *HTML_HANDLE = $_[0]; | 5156 local *HTML_HANDLE = $_[0]; |
| 4285 my $source = $_[1]; | 5157 my $source = $_[1]; |
| 4286 my $sumcount = $_[2]; | 5158 my $sumcount = $_[2]; |
| 4287 my $funcdata = $_[3]; | 5159 my $funcdata = $_[3]; |
| 4288 my $sumfncdata = $_[4]; | 5160 my $sumfncdata = $_[4]; |
| 4289 my $testfncdata = $_[5]; | 5161 my $testfncdata = $_[5]; |
| 4290 » my $name = $_[6]; | 5162 » my $sumbrcount = $_[6]; |
| 4291 » my $base = $_[7]; | 5163 » my $testbrdata = $_[7]; |
| 4292 » my $type = $_[8]; | 5164 » my $name = $_[8]; |
| 5165 » my $base = $_[9]; |
| 5166 » my $type = $_[10]; |
| 4293 my $func; | 5167 my $func; |
| 4294 my $func_code; | 5168 my $func_code; |
| 4295 my $count_code; | 5169 my $count_code; |
| 4296 | 5170 |
| 4297 # Get HTML code for headings | 5171 # Get HTML code for headings |
| 4298 $func_code = funcview_get_func_code($name, $base, $type); | 5172 $func_code = funcview_get_func_code($name, $base, $type); |
| 4299 $count_code = funcview_get_count_code($name, $base, $type); | 5173 $count_code = funcview_get_count_code($name, $base, $type); |
| 4300 write_html(*HTML_HANDLE, <<END_OF_HTML) | 5174 write_html(*HTML_HANDLE, <<END_OF_HTML) |
| 4301 <center> | 5175 <center> |
| 4302 <table width="60%" cellpadding=1 cellspacing=1 border=0> | 5176 <table width="60%" cellpadding=1 cellspacing=1 border=0> |
| 4303 <tr><td><br></td></tr> | 5177 <tr><td><br></td></tr> |
| 4304 <tr> | 5178 <tr> |
| 4305 <td width="80%" class="tableHead">$func_code</td> | 5179 <td width="80%" class="tableHead">$func_code</td> |
| 4306 <td width="20%" class="tableHead">$count_code</td> | 5180 <td width="20%" class="tableHead">$count_code</td> |
| 4307 </tr> | 5181 </tr> |
| 4308 END_OF_HTML | 5182 END_OF_HTML |
| 4309 ; | 5183 ; |
| 4310 | 5184 |
| 4311 # Get a sorted table | 5185 # Get a sorted table |
| 4312 foreach $func (funcview_get_sorted($funcdata, $sumfncdata, $type)) { | 5186 foreach $func (funcview_get_sorted($funcdata, $sumfncdata, $type)) { |
| 5187 if (!defined($funcdata->{$func})) |
| 5188 { |
| 5189 next; |
| 5190 } |
| 5191 |
| 4313 my $startline = $funcdata->{$func} - $func_offset; | 5192 my $startline = $funcdata->{$func} - $func_offset; |
| 4314 » » my $name = escape_html($func); | 5193 » » my $name = $func; |
| 4315 my $count = $sumfncdata->{$name}; | 5194 my $count = $sumfncdata->{$name}; |
| 4316 my $countstyle; | 5195 my $countstyle; |
| 4317 | 5196 |
| 5197 # Demangle C++ function names if requested |
| 5198 if ($demangle_cpp) { |
| 5199 $name = `c++filt "$name"`; |
| 5200 chomp($name); |
| 5201 } |
| 5202 # Escape any remaining special characters |
| 5203 $name = escape_html($name); |
| 4318 if ($startline < 1) { | 5204 if ($startline < 1) { |
| 4319 $startline = 1; | 5205 $startline = 1; |
| 4320 } | 5206 } |
| 4321 if ($count == 0) { | 5207 if ($count == 0) { |
| 4322 $countstyle = "coverFnLo"; | 5208 $countstyle = "coverFnLo"; |
| 4323 } else { | 5209 } else { |
| 4324 $countstyle = "coverFnHi"; | 5210 $countstyle = "coverFnHi"; |
| 4325 } | 5211 } |
| 4326 | 5212 |
| 4327 write_html(*HTML_HANDLE, <<END_OF_HTML) | 5213 write_html(*HTML_HANDLE, <<END_OF_HTML) |
| (...skipping 68 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4396 | 5282 |
| 4397 # | 5283 # |
| 4398 # subtract_fnccounts(data, base) | 5284 # subtract_fnccounts(data, base) |
| 4399 # | 5285 # |
| 4400 # Subtract function call counts found in base from those in data. | 5286 # Subtract function call counts found in base from those in data. |
| 4401 # Return (data, f_found, f_hit). | 5287 # Return (data, f_found, f_hit). |
| 4402 # | 5288 # |
| 4403 | 5289 |
| 4404 sub subtract_fnccounts($$) | 5290 sub subtract_fnccounts($$) |
| 4405 { | 5291 { |
| 4406 » my %data = %{$_[0]}; | 5292 » my %data; |
| 4407 » my %base = %{$_[1]}; | 5293 » my %base; |
| 4408 my $func; | 5294 my $func; |
| 4409 my $data_count; | 5295 my $data_count; |
| 4410 my $base_count; | 5296 my $base_count; |
| 4411 my $fn_hit = 0; | 5297 my $fn_hit = 0; |
| 4412 my $fn_found = 0; | 5298 my $fn_found = 0; |
| 4413 | 5299 |
| 5300 %data = %{$_[0]} if (defined($_[0])); |
| 5301 %base = %{$_[1]} if (defined($_[1])); |
| 4414 foreach $func (keys(%data)) { | 5302 foreach $func (keys(%data)) { |
| 4415 $fn_found++; | 5303 $fn_found++; |
| 4416 $data_count = $data{$func}; | 5304 $data_count = $data{$func}; |
| 4417 $base_count = $base{$func}; | 5305 $base_count = $base{$func}; |
| 4418 | 5306 |
| 4419 if (defined($base_count)) { | 5307 if (defined($base_count)) { |
| 4420 $data_count -= $base_count; | 5308 $data_count -= $base_count; |
| 4421 | 5309 |
| 4422 # Make sure we don't get negative numbers | 5310 # Make sure we don't get negative numbers |
| 4423 if ($data_count < 0) { | 5311 if ($data_count < 0) { |
| (...skipping 22 matching lines...) Expand all Loading... |
| 4446 { | 5334 { |
| 4447 my %data_hash = %{$_[0]}; | 5335 my %data_hash = %{$_[0]}; |
| 4448 my %base_hash = %{$_[1]}; | 5336 my %base_hash = %{$_[1]}; |
| 4449 my $filename; | 5337 my $filename; |
| 4450 my $testname; | 5338 my $testname; |
| 4451 my $data; | 5339 my $data; |
| 4452 my $data_testdata; | 5340 my $data_testdata; |
| 4453 my $data_funcdata; | 5341 my $data_funcdata; |
| 4454 my $data_checkdata; | 5342 my $data_checkdata; |
| 4455 my $data_testfncdata; | 5343 my $data_testfncdata; |
| 5344 my $data_testbrdata; |
| 4456 my $data_count; | 5345 my $data_count; |
| 4457 my $data_testfnccount; | 5346 my $data_testfnccount; |
| 5347 my $data_testbrcount; |
| 4458 my $base; | 5348 my $base; |
| 4459 my $base_checkdata; | 5349 my $base_checkdata; |
| 4460 my $base_sumfnccount; | 5350 my $base_sumfnccount; |
| 5351 my $base_sumbrcount; |
| 4461 my $base_count; | 5352 my $base_count; |
| 4462 my $sumcount; | 5353 my $sumcount; |
| 4463 my $sumfnccount; | 5354 my $sumfnccount; |
| 5355 my $sumbrcount; |
| 4464 my $found; | 5356 my $found; |
| 4465 my $hit; | 5357 my $hit; |
| 4466 my $fn_found; | 5358 my $fn_found; |
| 4467 my $fn_hit; | 5359 my $fn_hit; |
| 5360 my $br_found; |
| 5361 my $br_hit; |
| 4468 | 5362 |
| 4469 foreach $filename (keys(%data_hash)) | 5363 foreach $filename (keys(%data_hash)) |
| 4470 { | 5364 { |
| 4471 # Get data set for data and baseline | 5365 # Get data set for data and baseline |
| 4472 $data = $data_hash{$filename}; | 5366 $data = $data_hash{$filename}; |
| 4473 $base = $base_hash{$filename}; | 5367 $base = $base_hash{$filename}; |
| 4474 | 5368 |
| 4475 # Skip data entries for which no base entry exists | 5369 # Skip data entries for which no base entry exists |
| 4476 if (!defined($base)) | 5370 if (!defined($base)) |
| 4477 { | 5371 { |
| 4478 next; | 5372 next; |
| 4479 } | 5373 } |
| 4480 | 5374 |
| 4481 # Get set entries for data and baseline | 5375 # Get set entries for data and baseline |
| 4482 ($data_testdata, undef, $data_funcdata, $data_checkdata, | 5376 ($data_testdata, undef, $data_funcdata, $data_checkdata, |
| 4483 » » $data_testfncdata) = get_info_entry($data); | 5377 » » $data_testfncdata, undef, $data_testbrdata) = |
| 5378 » » » get_info_entry($data); |
| 4484 (undef, $base_count, undef, $base_checkdata, undef, | 5379 (undef, $base_count, undef, $base_checkdata, undef, |
| 4485 » » $base_sumfnccount) = get_info_entry($base); | 5380 » » $base_sumfnccount, undef, $base_sumbrcount) = |
| 5381 » » » get_info_entry($base); |
| 4486 | 5382 |
| 4487 # Check for compatible checksums | 5383 # Check for compatible checksums |
| 4488 merge_checksums($data_checkdata, $base_checkdata, $filename); | 5384 merge_checksums($data_checkdata, $base_checkdata, $filename); |
| 4489 | 5385 |
| 4490 # sumcount has to be calculated anew | 5386 # sumcount has to be calculated anew |
| 4491 $sumcount = {}; | 5387 $sumcount = {}; |
| 4492 $sumfnccount = {}; | 5388 $sumfnccount = {}; |
| 5389 $sumbrcount = {}; |
| 4493 | 5390 |
| 4494 # For each test case, subtract test specific counts | 5391 # For each test case, subtract test specific counts |
| 4495 foreach $testname (keys(%{$data_testdata})) | 5392 foreach $testname (keys(%{$data_testdata})) |
| 4496 { | 5393 { |
| 4497 # Get counts of both data and baseline | 5394 # Get counts of both data and baseline |
| 4498 $data_count = $data_testdata->{$testname}; | 5395 $data_count = $data_testdata->{$testname}; |
| 4499 $data_testfnccount = $data_testfncdata->{$testname}; | 5396 $data_testfnccount = $data_testfncdata->{$testname}; |
| 5397 $data_testbrcount = $data_testbrdata->{$testname}; |
| 4500 | 5398 |
| 4501 ($data_count, undef, $hit) = | 5399 ($data_count, undef, $hit) = |
| 4502 subtract_counts($data_count, $base_count); | 5400 subtract_counts($data_count, $base_count); |
| 4503 ($data_testfnccount) = | 5401 ($data_testfnccount) = |
| 4504 subtract_fnccounts($data_testfnccount, | 5402 subtract_fnccounts($data_testfnccount, |
| 4505 $base_sumfnccount); | 5403 $base_sumfnccount); |
| 5404 ($data_testbrcount) = |
| 5405 combine_brcount($data_testbrcount, |
| 5406 $base_sumbrcount, $BR_SUB); |
| 5407 |
| 4506 | 5408 |
| 4507 # Check whether this test case did hit any line at all | 5409 # Check whether this test case did hit any line at all |
| 4508 if ($hit > 0) | 5410 if ($hit > 0) |
| 4509 { | 5411 { |
| 4510 # Write back resulting hash | 5412 # Write back resulting hash |
| 4511 $data_testdata->{$testname} = $data_count; | 5413 $data_testdata->{$testname} = $data_count; |
| 4512 $data_testfncdata->{$testname} = | 5414 $data_testfncdata->{$testname} = |
| 4513 $data_testfnccount; | 5415 $data_testfnccount; |
| 5416 $data_testbrdata->{$testname} = |
| 5417 $data_testbrcount; |
| 4514 } | 5418 } |
| 4515 else | 5419 else |
| 4516 { | 5420 { |
| 4517 # Delete test case which did not impact this | 5421 # Delete test case which did not impact this |
| 4518 # file | 5422 # file |
| 4519 delete($data_testdata->{$testname}); | 5423 delete($data_testdata->{$testname}); |
| 4520 delete($data_testfncdata->{$testname}); | 5424 delete($data_testfncdata->{$testname}); |
| 5425 delete($data_testbrdata->{$testname}); |
| 4521 } | 5426 } |
| 4522 | 5427 |
| 4523 # Add counts to sum of counts | 5428 # Add counts to sum of counts |
| 4524 ($sumcount, $found, $hit) = | 5429 ($sumcount, $found, $hit) = |
| 4525 add_counts($sumcount, $data_count); | 5430 add_counts($sumcount, $data_count); |
| 4526 ($sumfnccount, $fn_found, $fn_hit) = | 5431 ($sumfnccount, $fn_found, $fn_hit) = |
| 4527 » » » » add_fnccounts($sumfnccount, $data_testfnccount); | 5432 » » » » add_fnccount($sumfnccount, $data_testfnccount); |
| 5433 » » » ($sumbrcount, $br_found, $br_hit) = |
| 5434 » » » » combine_brcount($sumbrcount, $data_testbrcount, |
| 5435 » » » » » » $BR_ADD); |
| 4528 } | 5436 } |
| 4529 | 5437 |
| 4530 # Write back resulting entry | 5438 # Write back resulting entry |
| 4531 set_info_entry($data, $data_testdata, $sumcount, $data_funcdata, | 5439 set_info_entry($data, $data_testdata, $sumcount, $data_funcdata, |
| 4532 $data_checkdata, $data_testfncdata, $sumfnccount, | 5440 $data_checkdata, $data_testfncdata, $sumfnccount, |
| 4533 » » » $found, $hit, $fn_found, $fn_hit); | 5441 » » » $data_testbrdata, $sumbrcount, $found, $hit, |
| 5442 » » » $fn_found, $fn_hit, $br_found, $br_hit); |
| 4534 | 5443 |
| 4535 $data_hash{$filename} = $data; | 5444 $data_hash{$filename} = $data; |
| 4536 } | 5445 } |
| 4537 | 5446 |
| 4538 return (\%data_hash); | 5447 return (\%data_hash); |
| 4539 } | 5448 } |
| 4540 | 5449 |
| 4541 | 5450 |
| 4542 # | 5451 # |
| 4543 # remove_unused_descriptions() | 5452 # remove_unused_descriptions() |
| (...skipping 76 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4620 # | 5529 # |
| 4621 | 5530 |
| 4622 sub system_no_output($@) | 5531 sub system_no_output($@) |
| 4623 { | 5532 { |
| 4624 my $mode = shift; | 5533 my $mode = shift; |
| 4625 my $result; | 5534 my $result; |
| 4626 local *OLD_STDERR; | 5535 local *OLD_STDERR; |
| 4627 local *OLD_STDOUT; | 5536 local *OLD_STDOUT; |
| 4628 | 5537 |
| 4629 # Save old stdout and stderr handles | 5538 # Save old stdout and stderr handles |
| 4630 » ($mode & 1) && open(OLD_STDOUT, ">>&STDOUT"); | 5539 » ($mode & 1) && open(OLD_STDOUT, ">>&", "STDOUT"); |
| 4631 » ($mode & 2) && open(OLD_STDERR, ">>&STDERR"); | 5540 » ($mode & 2) && open(OLD_STDERR, ">>&", "STDERR"); |
| 4632 | 5541 |
| 4633 # Redirect to /dev/null | 5542 # Redirect to /dev/null |
| 4634 » ($mode & 1) && open(STDOUT, ">/dev/null"); | 5543 » ($mode & 1) && open(STDOUT, ">", "/dev/null"); |
| 4635 » ($mode & 2) && open(STDERR, ">/dev/null"); | 5544 » ($mode & 2) && open(STDERR, ">", "/dev/null"); |
| 4636 | 5545 |
| 4637 system(@_); | 5546 system(@_); |
| 4638 $result = $?; | 5547 $result = $?; |
| 4639 | 5548 |
| 4640 # Close redirected handles | 5549 # Close redirected handles |
| 4641 ($mode & 1) && close(STDOUT); | 5550 ($mode & 1) && close(STDOUT); |
| 4642 ($mode & 2) && close(STDERR); | 5551 ($mode & 2) && close(STDERR); |
| 4643 | 5552 |
| 4644 # Restore old handles | 5553 # Restore old handles |
| 4645 » ($mode & 1) && open(STDOUT, ">>&OLD_STDOUT"); | 5554 » ($mode & 1) && open(STDOUT, ">>&", "OLD_STDOUT"); |
| 4646 » ($mode & 2) && open(STDERR, ">>&OLD_STDERR"); | 5555 » ($mode & 2) && open(STDERR, ">>&", "OLD_STDERR"); |
| 4647 | 5556 |
| 4648 return $result; | 5557 return $result; |
| 4649 } | 5558 } |
| 4650 | 5559 |
| 4651 | 5560 |
| 4652 # | 5561 # |
| 4653 # read_config(filename) | 5562 # read_config(filename) |
| 4654 # | 5563 # |
| 4655 # Read configuration file FILENAME and return a reference to a hash containing | 5564 # Read configuration file FILENAME and return a reference to a hash containing |
| 4656 # all valid key=value pairs found. | 5565 # all valid key=value pairs found. |
| 4657 # | 5566 # |
| 4658 | 5567 |
| 4659 sub read_config($) | 5568 sub read_config($) |
| 4660 { | 5569 { |
| 4661 my $filename = $_[0]; | 5570 my $filename = $_[0]; |
| 4662 my %result; | 5571 my %result; |
| 4663 my $key; | 5572 my $key; |
| 4664 my $value; | 5573 my $value; |
| 4665 local *HANDLE; | 5574 local *HANDLE; |
| 4666 | 5575 |
| 4667 » if (!open(HANDLE, "<$filename")) | 5576 » if (!open(HANDLE, "<", $filename)) |
| 4668 { | 5577 { |
| 4669 warn("WARNING: cannot read configuration file $filename\n"); | 5578 warn("WARNING: cannot read configuration file $filename\n"); |
| 4670 return undef; | 5579 return undef; |
| 4671 } | 5580 } |
| 4672 while (<HANDLE>) | 5581 while (<HANDLE>) |
| 4673 { | 5582 { |
| 4674 chomp; | 5583 chomp; |
| 4675 # Skip comments | 5584 # Skip comments |
| 4676 s/#.*//; | 5585 s/#.*//; |
| 4677 # Remove leading blanks | 5586 # Remove leading blanks |
| (...skipping 18 matching lines...) Expand all Loading... |
| 4696 | 5605 |
| 4697 | 5606 |
| 4698 # | 5607 # |
| 4699 # apply_config(REF) | 5608 # apply_config(REF) |
| 4700 # | 5609 # |
| 4701 # REF is a reference to a hash containing the following mapping: | 5610 # REF is a reference to a hash containing the following mapping: |
| 4702 # | 5611 # |
| 4703 # key_string => var_ref | 5612 # key_string => var_ref |
| 4704 # | 5613 # |
| 4705 # where KEY_STRING is a keyword and VAR_REF is a reference to an associated | 5614 # where KEY_STRING is a keyword and VAR_REF is a reference to an associated |
| 4706 # variable. If the global configuration hash CONFIG contains a value for | 5615 # variable. If the global configuration hashes CONFIG or OPT_RC contain a value |
| 4707 # keyword KEY_STRING, VAR_REF will be assigned the value for that keyword. | 5616 # for keyword KEY_STRING, VAR_REF will be assigned the value for that keyword. |
| 4708 # | 5617 # |
| 4709 | 5618 |
| 4710 sub apply_config($) | 5619 sub apply_config($) |
| 4711 { | 5620 { |
| 4712 my $ref = $_[0]; | 5621 my $ref = $_[0]; |
| 4713 | 5622 |
| 4714 foreach (keys(%{$ref})) | 5623 foreach (keys(%{$ref})) |
| 4715 { | 5624 { |
| 4716 » » if (defined($config->{$_})) | 5625 » » if (defined($opt_rc{$_})) { |
| 4717 » » { | 5626 » » » ${$ref->{$_}} = $opt_rc{$_}; |
| 5627 » » } elsif (defined($config->{$_})) { |
| 4718 ${$ref->{$_}} = $config->{$_}; | 5628 ${$ref->{$_}} = $config->{$_}; |
| 4719 } | 5629 } |
| 4720 } | 5630 } |
| 4721 } | 5631 } |
| 4722 | 5632 |
| 4723 | 5633 |
| 4724 # | 5634 # |
| 4725 # get_html_prolog(FILENAME) | 5635 # get_html_prolog(FILENAME) |
| 4726 # | 5636 # |
| 4727 # If FILENAME is defined, return contents of file. Otherwise return default | 5637 # If FILENAME is defined, return contents of file. Otherwise return default |
| 4728 # HTML prolog. Die on error. | 5638 # HTML prolog. Die on error. |
| 4729 # | 5639 # |
| 4730 | 5640 |
| 4731 sub get_html_prolog($) | 5641 sub get_html_prolog($) |
| 4732 { | 5642 { |
| 4733 my $filename = $_[0]; | 5643 my $filename = $_[0]; |
| 4734 my $result = ""; | 5644 my $result = ""; |
| 4735 | 5645 |
| 4736 if (defined($filename)) | 5646 if (defined($filename)) |
| 4737 { | 5647 { |
| 4738 local *HANDLE; | 5648 local *HANDLE; |
| 4739 | 5649 |
| 4740 » » open(HANDLE, "<".$filename) | 5650 » » open(HANDLE, "<", $filename) |
| 4741 or die("ERROR: cannot open html prolog $filename!\n"); | 5651 or die("ERROR: cannot open html prolog $filename!\n"); |
| 4742 while (<HANDLE>) | 5652 while (<HANDLE>) |
| 4743 { | 5653 { |
| 4744 $result .= $_; | 5654 $result .= $_; |
| 4745 } | 5655 } |
| 4746 close(HANDLE); | 5656 close(HANDLE); |
| 4747 } | 5657 } |
| 4748 else | 5658 else |
| 4749 { | 5659 { |
| 4750 $result = <<END_OF_HTML | 5660 $result = <<END_OF_HTML |
| 4751 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> | 5661 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> |
| 4752 | 5662 |
| 4753 <html lang="en"> | 5663 <html lang="en"> |
| 4754 | 5664 |
| 4755 <head> | 5665 <head> |
| 4756 <meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1"> | 5666 <meta http-equiv="Content-Type" content="text/html; charset=$charset"> |
| 4757 <title>\@pagetitle\@</title> | 5667 <title>\@pagetitle\@</title> |
| 4758 <link rel="stylesheet" type="text/css" href="\@basedir\@gcov.css"> | 5668 <link rel="stylesheet" type="text/css" href="\@basedir\@gcov.css"> |
| 4759 </head> | 5669 </head> |
| 4760 | 5670 |
| 4761 <body> | 5671 <body> |
| 4762 | 5672 |
| 4763 END_OF_HTML | 5673 END_OF_HTML |
| 4764 ; | 5674 ; |
| 4765 } | 5675 } |
| 4766 | 5676 |
| 4767 return $result; | 5677 return $result; |
| 4768 } | 5678 } |
| 4769 | 5679 |
| 4770 | 5680 |
| 4771 # | 5681 # |
| 4772 # get_html_epilog(FILENAME) | 5682 # get_html_epilog(FILENAME) |
| 4773 # | 5683 # |
| 4774 # If FILENAME is defined, return contents of file. Otherwise return default | 5684 # If FILENAME is defined, return contents of file. Otherwise return default |
| 4775 # HTML epilog. Die on error. | 5685 # HTML epilog. Die on error. |
| 4776 # | 5686 # |
| 4777 sub get_html_epilog($) | 5687 sub get_html_epilog($) |
| 4778 { | 5688 { |
| 4779 my $filename = $_[0]; | 5689 my $filename = $_[0]; |
| 4780 my $result = ""; | 5690 my $result = ""; |
| 4781 | 5691 |
| 4782 if (defined($filename)) | 5692 if (defined($filename)) |
| 4783 { | 5693 { |
| 4784 local *HANDLE; | 5694 local *HANDLE; |
| 4785 | 5695 |
| 4786 » » open(HANDLE, "<".$filename) | 5696 » » open(HANDLE, "<", $filename) |
| 4787 or die("ERROR: cannot open html epilog $filename!\n"); | 5697 or die("ERROR: cannot open html epilog $filename!\n"); |
| 4788 while (<HANDLE>) | 5698 while (<HANDLE>) |
| 4789 { | 5699 { |
| 4790 $result .= $_; | 5700 $result .= $_; |
| 4791 } | 5701 } |
| 4792 close(HANDLE); | 5702 close(HANDLE); |
| 4793 } | 5703 } |
| 4794 else | 5704 else |
| 4795 { | 5705 { |
| 4796 $result = <<END_OF_HTML | 5706 $result = <<END_OF_HTML |
| (...skipping 14 matching lines...) Expand all Loading... |
| 4811 | 5721 |
| 4812 warn("$tool_name: $msg"); | 5722 warn("$tool_name: $msg"); |
| 4813 } | 5723 } |
| 4814 | 5724 |
| 4815 sub die_handler($) | 5725 sub die_handler($) |
| 4816 { | 5726 { |
| 4817 my ($msg) = @_; | 5727 my ($msg) = @_; |
| 4818 | 5728 |
| 4819 die("$tool_name: $msg"); | 5729 die("$tool_name: $msg"); |
| 4820 } | 5730 } |
| 5731 |
| 5732 # |
| 5733 # parse_ignore_errors(@ignore_errors) |
| 5734 # |
| 5735 # Parse user input about which errors to ignore. |
| 5736 # |
| 5737 |
| 5738 sub parse_ignore_errors(@) |
| 5739 { |
| 5740 my (@ignore_errors) = @_; |
| 5741 my @items; |
| 5742 my $item; |
| 5743 |
| 5744 return if (!@ignore_errors); |
| 5745 |
| 5746 foreach $item (@ignore_errors) { |
| 5747 $item =~ s/\s//g; |
| 5748 if ($item =~ /,/) { |
| 5749 # Split and add comma-separated parameters |
| 5750 push(@items, split(/,/, $item)); |
| 5751 } else { |
| 5752 # Add single parameter |
| 5753 push(@items, $item); |
| 5754 } |
| 5755 } |
| 5756 foreach $item (@items) { |
| 5757 my $item_id = $ERROR_ID{lc($item)}; |
| 5758 |
| 5759 if (!defined($item_id)) { |
| 5760 die("ERROR: unknown argument for --ignore-errors: ". |
| 5761 "$item\n"); |
| 5762 } |
| 5763 $ignore[$item_id] = 1; |
| 5764 } |
| 5765 } |
| 5766 |
| 5767 # |
| 5768 # rate(hit, found[, suffix, precision, width]) |
| 5769 # |
| 5770 # Return the coverage rate [0..100] for HIT and FOUND values. 0 is only |
| 5771 # returned when HIT is 0. 100 is only returned when HIT equals FOUND. |
| 5772 # PRECISION specifies the precision of the result. SUFFIX defines a |
| 5773 # string that is appended to the result if FOUND is non-zero. Spaces |
| 5774 # are added to the start of the resulting string until it is at least WIDTH |
| 5775 # characters wide. |
| 5776 # |
| 5777 |
| 5778 sub rate($$;$$$) |
| 5779 { |
| 5780 my ($hit, $found, $suffix, $precision, $width) = @_; |
| 5781 my $rate; |
| 5782 |
| 5783 # Assign defaults if necessary |
| 5784 $precision = 1 if (!defined($precision)); |
| 5785 $suffix = "" if (!defined($suffix)); |
| 5786 $width = 0 if (!defined($width)); |
| 5787 |
| 5788 return sprintf("%*s", $width, "-") if (!defined($found) || $found == 0); |
| 5789 $rate = sprintf("%.*f", $precision, $hit * 100 / $found); |
| 5790 |
| 5791 # Adjust rates if necessary |
| 5792 if ($rate == 0 && $hit > 0) { |
| 5793 $rate = sprintf("%.*f", $precision, 1 / 10 ** $precision); |
| 5794 } elsif ($rate == 100 && $hit != $found) { |
| 5795 $rate = sprintf("%.*f", $precision, 100 - 1 / 10 ** $precision); |
| 5796 } |
| 5797 |
| 5798 return sprintf("%*s", $width, $rate.$suffix); |
| 5799 } |
| OLD | NEW |