blob: c6e9d048f053b7b6dc3c2ddde800ad18d06f5af7 [file] [log] [blame]
Kuang-che Wu3eb6b502018-06-06 16:15:18 +08001#!/usr/bin/env python2
Kuang-che Wu6e4beca2018-06-27 17:45:02 +08002# -*- coding: utf-8 -*-
Kuang-che Wu3eb6b502018-06-06 16:15:18 +08003# 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
8This bisector bisects commits between branched chrome and releases.
9"""
10from __future__ import print_function
11import logging
12import os
13
14from bisect_kit import cli
15from bisect_kit import codechange
16from bisect_kit import configure
17from bisect_kit import core
18from bisect_kit import cros_util
19from bisect_kit import cr_util
20from bisect_kit import gclient_util
21from bisect_kit import util
22
23logger = logging.getLogger(__name__)
24
25revtype = 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
31def 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
40class 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
183if __name__ == '__main__':
184 cli.BisectorCommandLine(ChromeSrcDomain).main()