Kuang-che Wu | 3eb6b50 | 2018-06-06 16:15:18 +0800 | [diff] [blame] | 1 | #!/usr/bin/env python2 |
Kuang-che Wu | 6e4beca | 2018-06-27 17:45:02 +0800 | [diff] [blame] | 2 | # -*- coding: utf-8 -*- |
Kuang-che Wu | 3eb6b50 | 2018-06-06 16:15:18 +0800 | [diff] [blame] | 3 | # Copyright 2018 The Chromium OS Authors. All rights reserved. |
| 4 | # Use of this source code is governed by a BSD-style license that can be |
| 5 | # found in the LICENSE file. |
| 6 | """Chrome bisector to bisect a range of chrome commits. |
| 7 | |
| 8 | This bisector bisects commits between branched chrome and releases. |
| 9 | """ |
| 10 | from __future__ import print_function |
| 11 | import logging |
| 12 | import os |
| 13 | |
| 14 | from bisect_kit import cli |
| 15 | from bisect_kit import codechange |
| 16 | from bisect_kit import configure |
| 17 | from bisect_kit import core |
| 18 | from bisect_kit import cros_util |
| 19 | from bisect_kit import cr_util |
| 20 | from bisect_kit import gclient_util |
| 21 | from bisect_kit import util |
| 22 | |
| 23 | logger = logging.getLogger(__name__) |
| 24 | |
| 25 | revtype = cli.argtype_multiplexer(cr_util.argtype_chrome_version, |
| 26 | cros_util.argtype_cros_version, |
| 27 | codechange.argtype_intra_rev( |
| 28 | cr_util.argtype_chrome_version)) |
| 29 | |
| 30 | |
| 31 | def guess_chrome_version(opts, rev): |
| 32 | if cros_util.is_cros_version(rev): |
| 33 | assert opts.board, 'need to specify BOARD for cros version' |
| 34 | rev = cros_util.query_chrome_version(opts.board, rev) |
| 35 | assert cr_util.is_chrome_version(rev) |
| 36 | |
| 37 | return rev |
| 38 | |
| 39 | |
| 40 | class ChromeSrcDomain(core.BisectDomain): |
| 41 | """BisectDomain for Chrome branched tree""" |
| 42 | revtype = staticmethod(revtype) |
| 43 | help = globals()['__doc__'] |
| 44 | |
| 45 | @staticmethod |
| 46 | def add_init_arguments(parser): |
| 47 | parser.add_argument( |
| 48 | '--chrome_root', |
| 49 | required=True, |
| 50 | metavar='CHROME_ROOT', |
| 51 | type=cli.argtype_dir_path, |
| 52 | default=configure.get('CHROME_ROOT'), |
| 53 | help='Root of chrome source tree, like ~/chromium') |
| 54 | parser.add_argument( |
| 55 | '--gclient_cache_dir', |
| 56 | required=True, |
| 57 | metavar='GCLIENT_CACHE_DIR', |
| 58 | type=cli.argtype_dir_path, |
| 59 | default=configure.get('GCLIENT_CACHE_DIR'), |
| 60 | help='gclient cache dir') |
| 61 | |
| 62 | # Only used for Chrome on ChromeOS. |
| 63 | parser.add_argument( |
| 64 | '--dut', |
| 65 | type=cli.argtype_notempty, |
| 66 | metavar='DUT', |
| 67 | default=configure.get('DUT'), |
| 68 | help='For ChromeOS, address of DUT (Device Under Test)') |
| 69 | parser.add_argument( |
| 70 | '--board', |
| 71 | metavar='BOARD', |
| 72 | default=configure.get('BOARD'), |
| 73 | help='For ChromeOS, board name') |
| 74 | |
| 75 | @staticmethod |
| 76 | def init(opts): |
| 77 | chrome_src = os.path.join(opts.chrome_root, 'src') |
| 78 | if not os.path.exists(chrome_src): |
| 79 | raise core.ExecutionFatalError("chrome src directory doesn't exist") |
| 80 | |
| 81 | if opts.dut: |
| 82 | assert cros_util.is_dut(opts.dut) |
| 83 | if not opts.board: |
| 84 | opts.board = cros_util.query_dut_board(opts.dut) |
| 85 | |
| 86 | old = guess_chrome_version(opts, opts.old) |
| 87 | new = guess_chrome_version(opts, opts.new) |
| 88 | assert old != new |
| 89 | |
| 90 | if not util.is_version_lesseq(old, new): |
| 91 | logger.error('old=%s is newer than new=%s !?', old, new) |
| 92 | raise Exception |
| 93 | |
| 94 | config = dict( |
| 95 | chrome_root=opts.chrome_root, |
| 96 | old=old, |
| 97 | new=new, |
| 98 | board=opts.board, |
| 99 | dut=opts.dut, |
| 100 | gclient_cache_dir=opts.gclient_cache_dir) |
| 101 | |
| 102 | spec_manager = cr_util.ChromeSpecManager(config) |
| 103 | cache = gclient_util.GclientCache(opts.gclient_cache_dir) |
| 104 | |
| 105 | # Initial sync to get all buildspecs. |
| 106 | release_deps_file = cr_util.ChromeSpecManager.get_release_deps(new) |
| 107 | path = os.path.join(opts.chrome_root, 'buildspec', release_deps_file) |
| 108 | if not os.path.exists(path): |
| 109 | spec_manager.sync_to_release(new) |
| 110 | |
| 111 | for rev in (old, new): |
| 112 | release_deps_file = cr_util.ChromeSpecManager.get_release_deps(rev) |
| 113 | path = os.path.join(opts.chrome_root, 'buildspec', release_deps_file) |
| 114 | if not os.path.exists(path): |
| 115 | raise Exception("%s doesn't exist" % path) |
| 116 | |
| 117 | # Make sure all repos in between are cached |
| 118 | float_specs = spec_manager.collect_float_spec(old, new) |
| 119 | for spec in reversed(float_specs): |
| 120 | spec_manager.parse_spec(spec) |
| 121 | if cache.are_spec_commits_available(spec): |
| 122 | continue |
| 123 | branch = spec.path.split(os.sep)[-2] |
| 124 | spec_manager.sync_to_branch_rev(branch, spec.name) |
| 125 | |
| 126 | if not util.is_direct_relative_version(old, new): |
| 127 | logger.warning('old=%s is not parent of new=%s', old, new) |
| 128 | old_ancestor = spec_manager.enumerate_ancestor(old) |
| 129 | new_ancestor = spec_manager.enumerate_ancestor(new) |
| 130 | for rev in reversed(old_ancestor): |
| 131 | if rev in new_ancestor: |
| 132 | lowest_common_ancestor = rev |
| 133 | break |
| 134 | else: |
| 135 | raise Exception('Unable to find their common ancestor') |
| 136 | logger.warning('Assume their lowest common ancestor, %s,' |
| 137 | 'still have expected old behavior as %s', |
| 138 | lowest_common_ancestor, old) |
| 139 | config['old'] = old = lowest_common_ancestor |
| 140 | |
| 141 | code_manager = codechange.CodeManager(opts.chrome_root, spec_manager, cache) |
| 142 | revlist = code_manager.build_revlist(old, new) |
| 143 | |
| 144 | return config, revlist |
| 145 | |
| 146 | def __init__(self, config): |
| 147 | self.config = config |
| 148 | |
| 149 | def setenv(self, env, rev): |
| 150 | env['CHROME_ROOT'] = self.config['chrome_root'] |
| 151 | env['GCLIENT_CACHE_DIR'] = self.config['gclient_cache_dir'] |
| 152 | env['REV'] = rev |
| 153 | |
| 154 | if self.config['board']: |
| 155 | env['BOARD'] = self.config['board'] |
| 156 | if self.config['dut']: |
| 157 | env['DUT'] = self.config['dut'] |
| 158 | |
| 159 | def view(self, old, new): |
| 160 | print('old', old) |
| 161 | print('new', new) |
| 162 | |
| 163 | old_base, old_next, _ = codechange.parse_intra_rev(old) |
| 164 | new_base, new_next, _ = codechange.parse_intra_rev(new) |
| 165 | # Only print log url if the range is within two releases. |
| 166 | if old_next in (new_base, new_next): |
| 167 | log_url = ('https://chromium.googlesource.com/chromium/src/+log/%s..%s' |
| 168 | '?pretty=fuller&n=10000') |
| 169 | print('Be careful the following url only lists chrome src/ commits. For ' |
| 170 | 'example, commits inside v8 and third party repos are not listed.') |
| 171 | if old_base != old_next: |
| 172 | print(log_url % (old_base, old_next)) |
| 173 | if new_base != new_next and old_next != new_next: |
| 174 | print(log_url % (new_base, new_next)) |
| 175 | |
| 176 | cache = gclient_util.GclientCache(self.config['gclient_cache_dir']) |
| 177 | spec_manager = cr_util.ChromeSpecManager(self.config) |
| 178 | code_manager = codechange.CodeManager(self.config['chrome_root'], |
| 179 | spec_manager, cache) |
| 180 | code_manager.view_rev_diff(old, new) |
| 181 | |
| 182 | |
| 183 | if __name__ == '__main__': |
| 184 | cli.BisectorCommandLine(ChromeSrcDomain).main() |