1#!/usr/bin/env python 2# 3# Copyright 2018, The Android Open Source Project 4# 5# Licensed under the Apache License, Version 2.0 (the "License"); 6# you may not use this file except in compliance with the License. 7# You may obtain a copy of the License at 8# 9# http://www.apache.org/licenses/LICENSE-2.0 10# 11# Unless required by applicable law or agreed to in writing, software 12# distributed under the License is distributed on an "AS IS" BASIS, 13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14# See the License for the specific language governing permissions and 15# limitations under the License. 16 17import argparse 18import datetime 19import logging 20import json 21import os 22import shutil 23import subprocess 24import sys 25import webbrowser 26 27from run_host_unit_tests import * 28""" 29This script is used to generate code coverage results host supported libraries. 30The script by default will generate an html report that summarizes the coverage 31results of the specified tests. The results can also be browsed to provide a 32report of which lines have been traveled upon execution of the binary. 33 34NOTE: Code that is compiled out or hidden by a #DEFINE will be listed as 35having been executed 0 times, thus reducing overall coverage. 36 37The steps in order to add coverage support to a new library and its 38corrisponding host test are as follows. 39 401. Add "clang_file_coverage" (defined in //build/Android.bp) as a default to the 41 source library(s) you want statistics for. 42 NOTE: Forgoing this default will cause no coverage data to be generated for 43 the source files in the library. 44 452. Add "clang_coverage_bin" as a default to the host supported test binary that 46 excercises the libraries that you covered in step 1. 47 NOTE: Forgetting to add this will cause there to be *NO* coverage data 48 generated when the binary is run. 49 503. Add the host test binary name and the files/directories you want coverage 51 statistics for to the COVERAGE_TESTS variable defined below. You may add 52 individual filenames or a directory to be tested. 53 NOTE: Avoid using a / at the beginning of a covered_files entry as this 54 breaks how the coverage generator resolves filenames. 55 56TODO: Support generating XML data and printing results to standard out. 57""" 58 59COVERAGE_TESTS = [ 60 { 61 "test_name": "net_test_avrcp", 62 "covered_files": [ 63 "system/bt/profile/avrcp", 64 ], 65 }, 66 { 67 "test_name": "bluetooth_test_sdp", 68 "covered_files": [ 69 "system/bt/profile/sdp", 70 ], 71 }, 72 { 73 "test_name": "test-vendor_test_host", 74 "covered_files": [ 75 "system/bt/vendor_libs/test_vendor_lib/include", 76 "system/bt/vendor_libs/test_vendor_lib/src", 77 ], 78 }, 79 { 80 "test_name": "rootcanal-packets_test_host", 81 "covered_files": [ 82 "system/bt/vendor_libs/test_vendor_lib/packets", 83 ], 84 }, 85 { 86 "test_name": "bluetooth_test_common", 87 "covered_files": [ 88 "system/bt/common", 89 ], 90 }, 91] 92 93WORKING_DIR = '/tmp/coverage' 94SOONG_UI_BASH = 'build/soong/soong_ui.bash' 95LLVM_DIR = 'prebuilts/clang/host/linux-x86/clang-r353983b/bin' 96LLVM_MERGE = LLVM_DIR + '/llvm-profdata' 97LLVM_COV = LLVM_DIR + '/llvm-cov' 98 99 100def write_root_html_head(f): 101 # Write the header part of the root html file. This was pulled from the 102 # page source of one of the generated html files. 103 f.write("<!doctype html><html><head>" \ 104 "<meta name='viewport' content='width=device-width,initial-scale=1'><met" \ 105 "a charset='UTF-8'><link rel='stylesheet' type='text/css' href='style.cs" \ 106 "s'></head><body><h2>Coverage Report</h2><h4>Created: " + 107 str(datetime.datetime.now().strftime('%Y-%m-%d %H:%M')) + 108 "</h4><p>Click <a href='http://clang.llvm.org/docs/SourceBasedCodeCovera" \ 109 "ge.html#interpreting-reports'>here</a> for information about interpreti" \ 110 "ng this report.</p><div class='centered'><table><tr><td class='column-e" \ 111 "ntry-bold'>Filename</td><td class='column-entry-bold'>Function Coverage" \ 112 "</td><td class='column-entry-bold'>Instantiation Coverage</td><td class" \ 113 "='column-entry-bold'>Line Coverage</td><td class='column-entry-bold'>Re" \ 114 "gion Coverage</td></tr>" 115 ) 116 117 118def write_root_html_column(f, covered, count): 119 percent = covered * 100.0 / count 120 value = "%.2f%% (%d/%d) " % (percent, covered, count) 121 color = 'column-entry-yellow' 122 if percent == 100: 123 color = 'column-entry-green' 124 if percent < 80.0: 125 color = 'column-entry-red' 126 f.write("<td class=\'" + color + "\'><pre>" + value + "</pre></td>") 127 128 129def write_root_html_rows(f, tests): 130 totals = { 131 "functions": { 132 "covered": 0, 133 "count": 0 134 }, 135 "instantiations": { 136 "covered": 0, 137 "count": 0 138 }, 139 "lines": { 140 "covered": 0, 141 "count": 0 142 }, 143 "regions": { 144 "covered": 0, 145 "count": 0 146 } 147 } 148 149 # Write the tests with their coverage summaries. 150 for test in tests: 151 test_name = test['test_name'] 152 covered_files = test['covered_files'] 153 json_results = generate_coverage_json(test) 154 test_totals = json_results['data'][0]['totals'] 155 156 f.write("<tr class='light-row'><td><pre><a href=\'" + os.path.join(test_name, "index.html") + "\'>" + 157 test_name + "</a></pre></td>") 158 for field_name in ['functions', 'instantiations', 'lines', 'regions']: 159 field = test_totals[field_name] 160 totals[field_name]['covered'] += field['covered'] 161 totals[field_name]['count'] += field['count'] 162 write_root_html_column(f, field['covered'], field['count']) 163 f.write("</tr>") 164 165 #Write the totals row. 166 f.write("<tr class='light-row-bold'><td><pre>Totals</a></pre></td>") 167 for field_name in ['functions', 'instantiations', 'lines', 'regions']: 168 field = totals[field_name] 169 write_root_html_column(f, field['covered'], field['count']) 170 f.write("</tr>") 171 172 173def write_root_html_tail(f): 174 # Pulled from the generated html coverage report. 175 f.write("</table></div><h5>Generated by llvm-cov -- llvm version 7.0.2svn<" \ 176 "/h5></body></html>") 177 178 179def generate_root_html(tests): 180 # Copy the css file from one of the coverage reports. 181 source_file = os.path.join(os.path.join(WORKING_DIR, tests[0]['test_name']), "style.css") 182 dest_file = os.path.join(WORKING_DIR, "style.css") 183 shutil.copy2(source_file, dest_file) 184 185 # Write the root index.html file that sumarizes all the tests. 186 f = open(os.path.join(WORKING_DIR, "index.html"), "w") 187 write_root_html_head(f) 188 write_root_html_rows(f, tests) 189 write_root_html_tail(f) 190 191 192def get_profraw_for_test(test_name): 193 test_root = get_native_test_root_or_die() 194 test_cmd = os.path.join(os.path.join(test_root, test_name), test_name) 195 if not os.path.isfile(test_cmd): 196 logging.error('The test ' + test_name + ' does not exist, please compile first') 197 sys.exit(1) 198 199 profraw_file_name = test_name + ".profraw" 200 profraw_path = os.path.join(WORKING_DIR, os.path.join(test_name, profraw_file_name)) 201 llvm_env_var = "LLVM_PROFILE_FILE=\"" + profraw_path + "\"" 202 203 test_cmd = llvm_env_var + " " + test_cmd 204 logging.info('Generating profraw data for ' + test_name) 205 logging.debug('cmd: ' + test_cmd) 206 if subprocess.call(test_cmd, shell=True) != 0: 207 logging.error('Test ' + test_name + ' failed. Please fix the test before generating coverage.') 208 sys.exit(1) 209 210 if not os.path.isfile(profraw_path): 211 logging.error( 212 'Generating the profraw file failed. Did you remember to add the proper compiler flags to your build?') 213 sys.exit(1) 214 215 return profraw_file_name 216 217 218def merge_profraw_data(test_name): 219 cmd = [] 220 cmd.append(os.path.join(get_android_root_or_die(), LLVM_MERGE + " merge ")) 221 222 test_working_dir = os.path.join(WORKING_DIR, test_name) 223 cmd.append(os.path.join(test_working_dir, test_name + ".profraw")) 224 profdata_file = os.path.join(test_working_dir, test_name + ".profdata") 225 226 cmd.append('-o ' + profdata_file) 227 logging.info('Combining profraw files into profdata for ' + test_name) 228 logging.debug('cmd: ' + " ".join(cmd)) 229 if subprocess.call(" ".join(cmd), shell=True) != 0: 230 logging.error('Failed to merge profraw files for ' + test_name) 231 sys.exit(1) 232 233 234def generate_coverage_html(test): 235 COVERAGE_ROOT = '/proc/self/cwd' 236 237 test_name = test['test_name'] 238 file_list = test['covered_files'] 239 240 test_working_dir = os.path.join(WORKING_DIR, test_name) 241 test_profdata_file = os.path.join(test_working_dir, test_name + ".profdata") 242 243 cmd = [ 244 os.path.join(get_android_root_or_die(), LLVM_COV), "show", "-format=html", "-summary-only", 245 "-show-line-counts-or-regions", "-show-instantiation-summary", "-instr-profile=" + test_profdata_file, 246 "-path-equivalence=\"" + COVERAGE_ROOT + "\",\"" + get_android_root_or_die() + "\"", 247 "-output-dir=" + test_working_dir 248 ] 249 250 # We have to have one object file not as an argument otherwise we can't specify source files. 251 test_cmd = os.path.join(os.path.join(get_native_test_root_or_die(), test_name), test_name) 252 cmd.append(test_cmd) 253 254 # Filter out the specific files we want coverage for 255 for filename in file_list: 256 cmd.append(os.path.join(get_android_root_or_die(), filename)) 257 258 logging.info('Generating coverage report for ' + test['test_name']) 259 logging.debug('cmd: ' + " ".join(cmd)) 260 if subprocess.call(" ".join(cmd), shell=True) != 0: 261 logging.error('Failed to generate coverage for ' + test['test_name']) 262 sys.exit(1) 263 264 265def generate_coverage_json(test): 266 COVERAGE_ROOT = '/proc/self/cwd' 267 test_name = test['test_name'] 268 file_list = test['covered_files'] 269 270 test_working_dir = os.path.join(WORKING_DIR, test_name) 271 test_profdata_file = os.path.join(test_working_dir, test_name + ".profdata") 272 273 cmd = [ 274 os.path.join(get_android_root_or_die(), LLVM_COV), 275 "export", 276 "-summary-only", 277 "-show-region-summary", 278 "-instr-profile=" + test_profdata_file, 279 "-path-equivalence=\"" + COVERAGE_ROOT + "\",\"" + get_android_root_or_die() + "\"", 280 ] 281 282 test_cmd = os.path.join(os.path.join(get_native_test_root_or_die(), test_name), test_name) 283 cmd.append(test_cmd) 284 285 # Filter out the specific files we want coverage for 286 for filename in file_list: 287 cmd.append(os.path.join(get_android_root_or_die(), filename)) 288 289 logging.info('Generating coverage json for ' + test['test_name']) 290 logging.debug('cmd: ' + " ".join(cmd)) 291 292 json_str = subprocess.check_output(" ".join(cmd), shell=True) 293 return json.loads(json_str) 294 295 296def write_json_summary(test): 297 test_name = test['test_name'] 298 test_working_dir = os.path.join(WORKING_DIR, test_name) 299 test_json_summary_file = os.path.join(test_working_dir, test_name + '.json') 300 logging.debug('Writing json summary file: ' + test_json_summary_file) 301 json_file = open(test_json_summary_file, 'w') 302 json.dump(generate_coverage_json(test), json_file) 303 json_file.close() 304 305 306def list_tests(): 307 for test in COVERAGE_TESTS: 308 print "Test Name: " + test['test_name'] 309 print "Covered Files: " 310 for covered_file in test['covered_files']: 311 print " " + covered_file 312 print 313 314 315def main(): 316 parser = argparse.ArgumentParser(description='Generate code coverage for enabled tests.') 317 parser.add_argument( 318 '-l', 319 '--list-tests', 320 action='store_true', 321 dest='list_tests', 322 help='List all the available tests to be run as well as covered files.') 323 parser.add_argument( 324 '-a', '--all', 325 action='store_true', 326 help='Runs all available tests and prints their outputs. If no tests ' \ 327 'are specified via the -t option all tests will be run.') 328 parser.add_argument( 329 '-t', '--test', 330 dest='tests', 331 action='append', 332 type=str, 333 metavar='TESTNAME', 334 default=[], 335 help='Specifies a test to be run. Multiple tests can be specified by ' \ 336 'using this option multiple times. ' \ 337 'Example: \"gen_coverage.py -t test1 -t test2\"') 338 parser.add_argument( 339 '-o', '--output', 340 type=str, 341 metavar='DIRECTORY', 342 default='/tmp/coverage', 343 help='Specifies the directory to store all files. The directory will be ' \ 344 'created if it does not exist. Default is \"/tmp/coverage\"') 345 parser.add_argument( 346 '-s', 347 '--skip-html', 348 dest='skip_html', 349 action='store_true', 350 help='Skip opening up the results of the coverage report in a browser.') 351 parser.add_argument( 352 '-j', 353 '--json-file', 354 dest='json_file', 355 action='store_true', 356 help='Write out summary results to json file in test directory.') 357 358 logging.basicConfig(stream=sys.stderr, level=logging.DEBUG, format='%(levelname)s %(message)s') 359 logging.addLevelName(logging.DEBUG, "[\033[1;34m%s\033[0m]" % logging.getLevelName(logging.DEBUG)) 360 logging.addLevelName(logging.INFO, "[\033[1;34m%s\033[0m]" % logging.getLevelName(logging.INFO)) 361 logging.addLevelName(logging.WARNING, "[\033[1;31m%s\033[0m]" % logging.getLevelName(logging.WARNING)) 362 logging.addLevelName(logging.ERROR, "[\033[1;31m%s\033[0m]" % logging.getLevelName(logging.ERROR)) 363 364 args = parser.parse_args() 365 logging.debug("Args: " + str(args)) 366 367 # Set the working directory 368 global WORKING_DIR 369 WORKING_DIR = os.path.abspath(args.output) 370 logging.debug("Working Dir: " + WORKING_DIR) 371 372 # Print out the list of tests then exit 373 if args.list_tests: 374 list_tests() 375 sys.exit(0) 376 377 # Check to see if a test was specified and if so only generate coverage for 378 # that test. 379 if len(args.tests) == 0: 380 args.all = True 381 382 tests_to_run = [] 383 for test in COVERAGE_TESTS: 384 if args.all or test['test_name'] in args.tests: 385 tests_to_run.append(test) 386 if test['test_name'] in args.tests: 387 args.tests.remove(test['test_name']) 388 389 # Error if a test was specified but doesn't exist. 390 if len(args.tests) != 0: 391 for test_name in args.tests: 392 logging.error('\"' + test_name + '\" was not found in the list of available tests.') 393 sys.exit(1) 394 395 # Generate the info for the tests 396 for test in tests_to_run: 397 logging.info('Getting coverage for ' + test['test_name']) 398 get_profraw_for_test(test['test_name']) 399 merge_profraw_data(test['test_name']) 400 if args.json_file: 401 write_json_summary(test) 402 generate_coverage_html(test) 403 404 # Generate the root index.html page that sumarizes all of the coverage reports. 405 generate_root_html(tests_to_run) 406 407 # Open the results in a browser. 408 if not args.skip_html: 409 webbrowser.open('file://' + os.path.join(WORKING_DIR, 'index.html')) 410 411 412if __name__ == '__main__': 413 main() 414