diff options
author | jrg@chromium.org <jrg@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-07-07 18:20:53 +0000 |
---|---|---|
committer | jrg@chromium.org <jrg@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-07-07 18:20:53 +0000 |
commit | e8f6ff442e9c2eb59332990093ec1af65b2c1991 (patch) | |
tree | 78f5be88b569c8043311627f98280e602a2c614a | |
parent | ade4397a3af8a2d4dd8dbc60af01b19ace85d5c9 (diff) | |
download | chromium_src-e8f6ff442e9c2eb59332990093ec1af65b2c1991.zip chromium_src-e8f6ff442e9c2eb59332990093ec1af65b2c1991.tar.gz chromium_src-e8f6ff442e9c2eb59332990093ec1af65b2c1991.tar.bz2 |
Beginning of code coverage on Windows.
Review URL: http://codereview.chromium.org/155123
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@20045 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | build/common.gypi | 20 | ||||
-rw-r--r-- | chrome/chrome.gyp | 9 | ||||
-rwxr-xr-x | tools/code_coverage/coverage_posix.py | 176 |
3 files changed, 175 insertions, 30 deletions
diff --git a/build/common.gypi b/build/common.gypi index ed5992f..302a531 100644 --- a/build/common.gypi +++ b/build/common.gypi @@ -128,10 +128,22 @@ '-fprofile-arcs' ], 'link_settings': { 'libraries': [ '-lgcov' ] }, }], - ]}, - # TODO(jrg): options for code coverage on Windows - ], - ], + # Finally, for Windows, we simply turn on profiling. + ['OS=="win"', { + 'msvs_settings': { + 'VCLinkerTool': { + 'Profile': 'true', + }, + 'VCCLCompilerTool': { + # /Z7, not /Zi, so coverage is happyb + 'DebugInformationFormat': '1', + 'AdditionalOptions': '/Yd', + } + } + }], # OS==win + ], # conditions for coverage + }], # coverage!=0 + ], # conditions for 'target_defaults' 'default_configuration': 'Debug', 'configurations': { # VCLinkerTool LinkIncremental values below: diff --git a/chrome/chrome.gyp b/chrome/chrome.gyp index 8f9cebe..b5eceb7 100644 --- a/chrome/chrome.gyp +++ b/chrome/chrome.gyp @@ -4864,7 +4864,7 @@ ]}, # 'targets' ], # OS=="win" # TODO(jrg): add in Windows code coverage targets. - ['coverage!=0 and OS!="win"', + ['coverage!=0', { 'targets': [ { 'target_name': 'coverage', @@ -4887,13 +4887,16 @@ # requires the 'coverage' target be run from within # src/chrome. 'message': 'Running coverage_posix.py to generate coverage numbers', - 'inputs': [], - 'outputs': [], + # MSVS must have an input file and an output file. + 'inputs': [ '../tools/code_coverage/coverage_posix.py' ], + 'outputs': [ '<(PRODUCT_DIR)/coverage.info' ], 'action_name': 'coverage', 'action': [ 'python', '../tools/code_coverage/coverage_posix.py', '--directory', '<(PRODUCT_DIR)', + '--src_root', + '..', '--', '<@(_dependencies)'], # Use outputs of this action as inputs for the main target build. diff --git a/tools/code_coverage/coverage_posix.py b/tools/code_coverage/coverage_posix.py index 3a122fa..47207eb 100755 --- a/tools/code_coverage/coverage_posix.py +++ b/tools/code_coverage/coverage_posix.py @@ -3,10 +3,12 @@ # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. -"""Generate and process code coverage on POSIX systems. +"""Generate and process code coverage. -Written for and tested on Mac and Linux. To use this script to -generate coverage numbers, please run from within a gyp-generated +TODO(jrg): rename this from coverage_posix.py to coverage_all.py! + +Written for and tested on Mac, Linux, and Windows. To use this script +to generate coverage numbers, please run from within a gyp-generated project. All platforms, to set up coverage: @@ -53,6 +55,7 @@ class Coverage(object): def __init__(self, directory, options, args): super(Coverage, self).__init__() + logging.basicConfig(level=logging.DEBUG) self.directory = directory self.options = options self.args = args @@ -60,15 +63,48 @@ class Coverage(object): self.output_directory = os.path.join(self.directory, 'coverage') if not os.path.exists(self.output_directory): os.mkdir(self.output_directory) - self.lcov_directory = os.path.join(sys.path[0], - '../../third_party/lcov/bin') - self.lcov = os.path.join(self.lcov_directory, 'lcov') - self.mcov = os.path.join(self.lcov_directory, 'mcov') - self.genhtml = os.path.join(self.lcov_directory, 'genhtml') + # The "final" lcov-format file self.coverage_info_file = os.path.join(self.directory, 'coverage.info') + # If needed, an intermediate VSTS-format file + self.vsts_output = os.path.join(self.directory, 'coverage.vsts') + # Needed for Windows. + self.src_root = options.src_root + self.FindPrograms() self.ConfirmPlatformAndPaths() self.tests = [] + def FindInPath(self, program): + """Find program in our path. Return abs path to it, or None.""" + if not 'PATH' in os.environ: + logging.fatal('No PATH environment variable?') + sys.exit(1) + paths = os.environ['PATH'].split(os.pathsep) + for path in paths: + fullpath = os.path.join(path, program) + if os.path.exists(fullpath): + return fullpath + return None + + def FindPrograms(self): + """Find programs we may want to run.""" + if self.IsPosix(): + self.lcov_directory = os.path.join(sys.path[0], + '../../third_party/lcov/bin') + self.lcov = os.path.join(self.lcov_directory, 'lcov') + self.mcov = os.path.join(self.lcov_directory, 'mcov') + self.genhtml = os.path.join(self.lcov_directory, 'genhtml') + self.programs = [self.lcov, self.mcov, self.genhtml] + else: + commands = ['vsperfcmd.exe', 'vsinstr.exe', 'coverage_analyzer.exe'] + self.perf = self.FindInPath('vsperfcmd.exe') + self.instrument = self.FindInPath('vsinstr.exe') + self.analyzer = self.FindInPath('coverage_analyzer.exe') + if not self.perf or not self.instrument or not self.analyzer: + logging.fatal('Could not find Win performance commands.') + logging.fatal('Commands needed in PATH: ' + str(commands)) + sys.exit(1) + self.programs = [self.perf, self.instrument, self.analyzer] + def FindTests(self): """Find unit tests to run; set self.tests to this list. @@ -86,28 +122,53 @@ class Coverage(object): self.tests += [os.path.join(self.directory, testname.split(':')[1])] else: self.tests += [os.path.join(self.directory, testname)] - - # Needs to be run in the "chrome" directory? - # ut = os.path.join(self.directory, 'unit_tests') - # if os.path.exists(ut): - # self.tests.append(ut) # Medium tests? # Not sure all of these work yet (e.g. page_cycler_tests) # self.tests += glob.glob(os.path.join(self.directory, '*_tests')) + # If needed, append .exe to tests since vsinstr.exe likes it that + # way. + if self.IsWindows(): + for ind in range(len(self.tests)): + test = self.tests[ind] + test_exe = test + '.exe' + if not test.endswith('.exe') and os.path.exists(test_exe): + self.tests[ind] = test_exe + + # Temporarily make Windows quick for bringup by filtering + # out all except base_unittests. Easier than a chrome.cyp change. + # TODO(jrg): remove this + if self.IsWindows(): + t2 = [] + for test in self.tests: + if 'base_unittests' in test: + t2.append(test) + self.tests = t2 + + + def ConfirmPlatformAndPaths(self): """Confirm OS and paths (e.g. lcov).""" - if not self.IsPosix(): - logging.fatal('Not posix.') - sys.exit(1) - programs = [self.lcov, self.genhtml] - if self.IsMac(): - programs.append(self.mcov) - for program in programs: + for program in self.programs: if not os.path.exists(program): - logging.fatal('lcov program missing: ' + program) + logging.fatal('Program missing: ' + program) sys.exit(1) + def Run(self, cmdlist, ignore_error=False, ignore_retcode=None, + explanation=None): + """Run the command list; exit fatally on error.""" + logging.info('Running ' + str(cmdlist)) + retcode = subprocess.call(cmdlist) + if retcode: + if ignore_error or retcode == ignore_retcode: + logging.warning('COVERAGE: %s unhappy but errors ignored %s' % + (str(cmdlist), explanation or '')) + else: + logging.fatal('COVERAGE: %s failed; return code: %d' % + (str(cmdlist), retcode)) + sys.exit(retcode) + + def IsPosix(self): """Return True if we are POSIX.""" return self.IsMac() or self.IsLinux() @@ -118,13 +179,36 @@ class Coverage(object): def IsLinux(self): return sys.platform == 'linux2' + def IsWindows(self): + """Return True if we are Windows.""" + return sys.platform in ('win32', 'cygwin') + def ClearData(self): """Clear old gcda files""" + if not self.IsPosix(): + return subprocess.call([self.lcov, '--directory', self.directory_parent, '--zerocounters']) shutil.rmtree(os.path.join(self.directory, 'coverage')) + def BeforeRunTests(self): + """Do things before running tests.""" + if not self.IsWindows(): + return + # Stop old counters if needed + cmdlist = [self.perf, '-shutdown'] + self.Run(cmdlist, ignore_error=True) + # Instrument binaries + for fulltest in self.tests: + if os.path.exists(fulltest): + cmdlist = [self.instrument, '/COVERAGE', fulltest] + self.Run(cmdlist, ignore_retcode=4, + explanation='OK with a multiple-instrument') + # Start new counters + cmdlist = [self.perf, '-start:coverage', '-output:' + self.vsts_output] + self.Run(cmdlist) + def RunTests(self): """Run all unit tests.""" for fulltest in self.tests: @@ -138,7 +222,8 @@ class Coverage(object): # If asked, make this REAL fast for testing. if self.options.fast_test: - cmdlist.append('--gtest_filter=RenderWidgetHost*') + # cmdlist.append('--gtest_filter=RenderWidgetHost*') + cmdlist.append('--gtest_filter=CommandLine*') retcode = subprocess.call(cmdlist) if retcode: @@ -147,7 +232,17 @@ class Coverage(object): if self.options.strict: sys.exit(retcode) - def GenerateLcov(self): + def AfterRunTests(self): + """Do things right after running tests.""" + if not self.IsWindows(): + return + # Stop counters + cmdlist = [self.perf, '-shutdown'] + self.Run(cmdlist) + full_output = self.vsts_output + '.coverage' + shutil.move(full_output, self.vsts_output) + + def GenerateLcovPosix(self): """Convert profile data to lcov.""" command = [self.mcov, '--directory', self.directory_parent, @@ -160,6 +255,34 @@ class Coverage(object): if self.options.strict: sys.exit(retcode) + def GenerateLcovWindows(self): + """Convert VSTS format to lcov.""" + lcov_file = self.vsts_output + '.lcov' + if os.path.exists(lcov_file): + os.remove(lcov_file) + # generates the file (self.vsts_output + ".lcov") + + cmdlist = [self.analyzer, + '-sym_path=' + self.directory, + '-src_root=' + self.src_root, + self.vsts_output] + self.Run(cmdlist) + if not os.path.exists(lcov_file): + logging.fatal('Output file %s not created' % lcov_file) + sys.exit(1) + # So we name it appropriately + if os.path.exists(self.coverage_info_file): + os.remove(self.coverage_info_file) + logging.info('Renaming LCOV file to %s to be consistent' % + self.coverage_info_file) + shutil.move(self.vsts_output + '.lcov', self.coverage_info_file) + + def GenerateLcov(self): + if self.IsPosix(): + self.GenerateLcovPosix() + else: + self.GenerateLcovWindows() + def GenerateHtml(self): """Convert lcov to html.""" # TODO(jrg): This isn't happy when run with unit_tests since V8 has a @@ -206,13 +329,20 @@ def main(): dest='strict', default=False, help='Be strict and die on test failure.') + parser.add_option('-S', + '--src_root', + dest='src_root', + default='.', + help='Source root (only used on Windows)') (options, args) = parser.parse_args() if not options.directory: parser.error('Directory not specified') coverage = Coverage(options.directory, options, args) coverage.ClearData() coverage.FindTests() + coverage.BeforeRunTests() coverage.RunTests() + coverage.AfterRunTests() coverage.GenerateLcov() if options.genhtml: coverage.GenerateHtml() |