blob: eb1379378dd53bb02de9ae590a114e0fba849165 [file] [log] [blame]
Yunlian Jiange84ea3d2016-12-12 11:07:40 -08001#!/usr/bin/env python2
Ting-Yuan Huange5819872016-12-15 14:22:26 -08002#
3# Copyright 2016 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.
Yunlian Jiang14cf5962015-12-11 15:50:14 -08006"""Script for running nightly compiler tests on ChromeOS.
cmtice46093e52014-12-09 14:59:16 -08007
8This script launches a buildbot to build ChromeOS with the latest compiler on
9a particular board; then it finds and downloads the trybot image and the
10corresponding official image, and runs crosperf performance tests comparing
11the two. It then generates a report, emails it to the c-compiler-chrome, as
12well as copying the images into the seven-day reports directory.
13"""
14
15# Script to test different toolchains against ChromeOS benchmarks.
Yunlian Jiang14cf5962015-12-11 15:50:14 -080016
17from __future__ import print_function
18
Caroline Ticeeddb0632016-04-14 09:19:02 -070019import argparse
cmticece5ffa42015-02-12 15:18:43 -080020import datetime
cmtice46093e52014-12-09 14:59:16 -080021import os
Luis Lozanoc75fd052016-02-19 17:37:01 -080022import re
cmtice46093e52014-12-09 14:59:16 -080023import sys
24import time
cmtice46093e52014-12-09 14:59:16 -080025
Caroline Ticea8af9a72016-07-20 12:52:59 -070026from cros_utils import command_executer
27from cros_utils import logger
cmtice46093e52014-12-09 14:59:16 -080028
Caroline Ticea8af9a72016-07-20 12:52:59 -070029from cros_utils import buildbot_utils
cmtice46093e52014-12-09 14:59:16 -080030
Manoj Guptac4110352016-12-28 13:47:12 -080031# CL that uses LLVM-Next to build the images (includes chrome).
Manoj Gupta66682c72017-05-24 12:19:57 -070032USE_LLVM_NEXT_PATCH = '513590'
Manoj Guptac4110352016-12-28 13:47:12 -080033
Luis Lozanof2a3ef42015-12-15 13:49:30 -080034CROSTC_ROOT = '/usr/local/google/crostc'
35ROLE_ACCOUNT = 'mobiletc-prebuild'
cmtice46093e52014-12-09 14:59:16 -080036TOOLCHAIN_DIR = os.path.dirname(os.path.realpath(__file__))
Luis Lozanof2a3ef42015-12-15 13:49:30 -080037MAIL_PROGRAM = '~/var/bin/mail-sheriff'
Luis Lozanof2a3ef42015-12-15 13:49:30 -080038PENDING_ARCHIVES_DIR = os.path.join(CROSTC_ROOT, 'pending_archives')
39NIGHTLY_TESTS_DIR = os.path.join(CROSTC_ROOT, 'nightly_test_reports')
40
Ting-Yuan Huange5819872016-12-15 14:22:26 -080041IMAGE_DIR = '{board}-{image_type}'
42IMAGE_VERSION_STR = r'{chrome_version}-{tip}\.{branch}\.{branch_branch}'
43IMAGE_FS = IMAGE_DIR + '/' + IMAGE_VERSION_STR
Ting-Yuan Huang112d5622018-03-26 13:49:42 -070044TRYBOT_IMAGE_FS = IMAGE_FS + '-{build_id}'
Luis Lozanoc75fd052016-02-19 17:37:01 -080045PFQ_IMAGE_FS = IMAGE_FS + '-rc1'
Manoj Guptaaee96b72016-10-24 13:43:28 -070046IMAGE_RE_GROUPS = {
47 'board': r'(?P<board>\S+)',
48 'image_type': r'(?P<image_type>\S+)',
49 'chrome_version': r'(?P<chrome_version>R\d+)',
50 'tip': r'(?P<tip>\d+)',
51 'branch': r'(?P<branch>\d+)',
52 'branch_branch': r'(?P<branch_branch>\d+)',
53 'build_id': r'(?P<build_id>b\d+)'
54}
Luis Lozanoc75fd052016-02-19 17:37:01 -080055TRYBOT_IMAGE_RE = TRYBOT_IMAGE_FS.format(**IMAGE_RE_GROUPS)
56
cmtice46093e52014-12-09 14:59:16 -080057
Yunlian Jiang14cf5962015-12-11 15:50:14 -080058class ToolchainComparator(object):
59 """Class for doing the nightly tests work."""
cmtice46093e52014-12-09 14:59:16 -080060
Luis Lozanof2a3ef42015-12-15 13:49:30 -080061 def __init__(self,
62 board,
63 remotes,
64 chromeos_root,
65 weekday,
66 patches,
67 noschedv2=False):
cmtice46093e52014-12-09 14:59:16 -080068 self._board = board
69 self._remotes = remotes
70 self._chromeos_root = chromeos_root
71 self._base_dir = os.getcwd()
72 self._ce = command_executer.GetCommandExecuter()
73 self._l = logger.GetLogger()
Yunlian Jiang73580dd2017-11-21 04:48:40 -080074 self._build = '%s-release-tryjob' % board
Manoj Guptad575b8a2017-03-08 10:51:28 -080075 self._patches = patches.split(',') if patches else []
Yunlian Jiang3c6e4672015-08-24 15:58:22 -070076 self._patches_string = '_'.join(str(p) for p in self._patches)
Han Shen43494292015-09-14 10:26:40 -070077 self._noschedv2 = noschedv2
Yunlian Jiang3c6e4672015-08-24 15:58:22 -070078
cmtice46093e52014-12-09 14:59:16 -080079 if not weekday:
Luis Lozanof2a3ef42015-12-15 13:49:30 -080080 self._weekday = time.strftime('%a')
cmtice46093e52014-12-09 14:59:16 -080081 else:
82 self._weekday = weekday
cmtice7f3190b2015-05-22 14:14:51 -070083 timestamp = datetime.datetime.strftime(datetime.datetime.now(),
Luis Lozanof2a3ef42015-12-15 13:49:30 -080084 '%Y-%m-%d_%H:%M:%S')
Manoj Guptaaee96b72016-10-24 13:43:28 -070085 self._reports_dir = os.path.join(
86 NIGHTLY_TESTS_DIR,
Caroline Tice9c4003a2017-11-07 16:37:33 -080087 '%s.%s' % (timestamp, board),
88 )
cmtice46093e52014-12-09 14:59:16 -080089
Luis Lozanoc75fd052016-02-19 17:37:01 -080090 def _GetVanillaImageName(self, trybot_image):
Ting-Yuan Huange5819872016-12-15 14:22:26 -080091 """Given a trybot artifact name, get latest vanilla image name.
cmtice46093e52014-12-09 14:59:16 -080092
Luis Lozano783954f2015-12-21 18:06:29 -080093 Args:
94 trybot_image: artifact name such as
Ting-Yuan Huang112d5622018-03-26 13:49:42 -070095 'daisy-release-tryjob/R40-6394.0.0-b1389'
Luis Lozano783954f2015-12-21 18:06:29 -080096
97 Returns:
Ting-Yuan Huange5819872016-12-15 14:22:26 -080098 Latest official image name, e.g. 'daisy-release/R57-9089.0.0'.
cmtice46093e52014-12-09 14:59:16 -080099 """
Yunlian Jiang2e0ad052017-11-22 14:06:45 -0800100 # We need to filter out -tryjob in the trybot_image.
101 trybot = re.sub('-tryjob', '', trybot_image)
102 mo = re.search(TRYBOT_IMAGE_RE, trybot)
Luis Lozanoc75fd052016-02-19 17:37:01 -0800103 assert mo
Ting-Yuan Huange5819872016-12-15 14:22:26 -0800104 dirname = IMAGE_DIR.replace('\\', '').format(**mo.groupdict())
Ting-Yuan Huang99d32c42017-04-24 20:34:43 -0700105 return buildbot_utils.GetLatestImage(self._chromeos_root, dirname)
cmtice46093e52014-12-09 14:59:16 -0800106
Luis Lozanoc75fd052016-02-19 17:37:01 -0800107 def _GetNonAFDOImageName(self, trybot_image):
108 """Given a trybot artifact name, get corresponding non-AFDO image name.
109
110 We get the non-AFDO image from the PFQ builders. This image
111 is not generated for all the boards and, the closest PFQ image
112 was the one build for the previous ChromeOS version (the chrome
113 used in the current version is the one validated in the previous
114 version).
115 The previous ChromeOS does not always exist either. So, we try
116 a couple of versions before.
Luis Lozano783954f2015-12-21 18:06:29 -0800117
118 Args:
119 trybot_image: artifact name such as
Ting-Yuan Huang112d5622018-03-26 13:49:42 -0700120 'daisy-release-tryjob/R40-6394.0.0-b1389'
Luis Lozano783954f2015-12-21 18:06:29 -0800121
122 Returns:
123 Corresponding chrome PFQ image name, e.g.
Luis Lozanoc75fd052016-02-19 17:37:01 -0800124 'daisy-chrome-pfq/R40-6393.0.0-rc1'.
Luis Lozano783954f2015-12-21 18:06:29 -0800125 """
Yunlian Jiang2e0ad052017-11-22 14:06:45 -0800126 trybot = re.sub('-tryjob', '', trybot_image)
127 mo = re.search(TRYBOT_IMAGE_RE, trybot)
Luis Lozanoc75fd052016-02-19 17:37:01 -0800128 assert mo
129 image_dict = mo.groupdict()
130 image_dict['image_type'] = 'chrome-pfq'
131 for _ in xrange(2):
132 image_dict['tip'] = str(int(image_dict['tip']) - 1)
133 nonafdo_image = PFQ_IMAGE_FS.replace('\\', '').format(**image_dict)
134 if buildbot_utils.DoesImageExist(self._chromeos_root, nonafdo_image):
135 return nonafdo_image
136 return ''
Luis Lozano783954f2015-12-21 18:06:29 -0800137
Luis Lozano783954f2015-12-21 18:06:29 -0800138 def _TestImages(self, trybot_image, vanilla_image, nonafdo_image):
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800139 """Create crosperf experiment file.
cmtice46093e52014-12-09 14:59:16 -0800140
Luis Lozano783954f2015-12-21 18:06:29 -0800141 Given the names of the trybot, vanilla and non-AFDO images, create the
cmtice46093e52014-12-09 14:59:16 -0800142 appropriate crosperf experiment file and launch crosperf on it.
143 """
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800144 experiment_file_dir = os.path.join(self._chromeos_root, '..', self._weekday)
145 experiment_file_name = '%s_toolchain_experiment.txt' % self._board
Yunlian Jiang2f563562015-08-28 13:54:04 -0700146
Manoj Guptad575b8a2017-03-08 10:51:28 -0800147 compiler_string = 'llvm'
Manoj Guptac4110352016-12-28 13:47:12 -0800148 if USE_LLVM_NEXT_PATCH in self._patches_string:
149 experiment_file_name = '%s_llvm_next_experiment.txt' % self._board
150 compiler_string = 'llvm_next'
Yunlian Jiang2f563562015-08-28 13:54:04 -0700151
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800152 experiment_file = os.path.join(experiment_file_dir, experiment_file_name)
cmtice46093e52014-12-09 14:59:16 -0800153 experiment_header = """
154 board: %s
155 remote: %s
Luis Lozanoe1efeb82015-06-16 16:35:44 -0700156 retries: 1
cmtice46093e52014-12-09 14:59:16 -0800157 """ % (self._board, self._remotes)
158 experiment_tests = """
Luis Lozano1489d642015-12-08 10:08:19 -0800159 benchmark: all_toolchain_perf {
cmtice46093e52014-12-09 14:59:16 -0800160 suite: telemetry_Crosperf
Ting-Yuan Huang83323642017-02-17 12:20:02 -0800161 iterations: 0
Manoj Gupta5a516382017-08-23 12:27:54 -0700162 run_local: False
cmtice46093e52014-12-09 14:59:16 -0800163 }
Caroline Ticee82513b2016-10-27 12:45:15 -0700164
165 benchmark: page_cycler_v2.typical_25 {
166 suite: telemetry_Crosperf
Ting-Yuan Huang83323642017-02-17 12:20:02 -0800167 iterations: 0
Caroline Ticee82513b2016-10-27 12:45:15 -0700168 run_local: False
169 retries: 0
170 }
cmtice46093e52014-12-09 14:59:16 -0800171 """
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800172
173 with open(experiment_file, 'w') as f:
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800174 f.write(experiment_header)
175 f.write(experiment_tests)
cmtice46093e52014-12-09 14:59:16 -0800176
177 # Now add vanilla to test file.
Caroline Tice372e1642018-05-24 15:44:12 -0700178 if os.getlogin() == ROLE_ACCOUNT:
179 official_image = """
180 vanilla_image {
181 chromeos_root: %s
182 build: %s
183 compiler: llvm
184 chrome_src: /usr/local/google/crostc/chrome-src-internal
185 }
186 """ % (self._chromeos_root, vanilla_image)
187 else:
188 official_image = """
cmtice46093e52014-12-09 14:59:16 -0800189 vanilla_image {
190 chromeos_root: %s
191 build: %s
Yunlian Jiang8c18be12017-03-20 16:52:33 -0700192 compiler: llvm
cmtice46093e52014-12-09 14:59:16 -0800193 }
194 """ % (self._chromeos_root, vanilla_image)
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800195 f.write(official_image)
cmtice46093e52014-12-09 14:59:16 -0800196
Luis Lozano783954f2015-12-21 18:06:29 -0800197 # Now add non-AFDO image to test file.
Luis Lozano439f2b72016-01-08 11:56:02 -0800198 if nonafdo_image:
Caroline Tice372e1642018-05-24 15:44:12 -0700199 if os.getlogin() == ROLE_ACCOUNT:
200 official_nonafdo_image = """
201 nonafdo_image {
202 chromeos_root: %s
203 build: %s
204 compiler: llvm
205 chrome_src: /usr/local/google/crostc/chrome-src-internal
206 }
207 """ % (self._chromeos_root, nonafdo_image)
208 else:
209 official_nonafdo_image = """
Luis Lozano783954f2015-12-21 18:06:29 -0800210 nonafdo_image {
211 chromeos_root: %s
212 build: %s
Yunlian Jiang8c18be12017-03-20 16:52:33 -0700213 compiler: llvm
Luis Lozano783954f2015-12-21 18:06:29 -0800214 }
215 """ % (self._chromeos_root, nonafdo_image)
Luis Lozano439f2b72016-01-08 11:56:02 -0800216 f.write(official_nonafdo_image)
Luis Lozano783954f2015-12-21 18:06:29 -0800217
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800218 label_string = '%s_trybot_image' % compiler_string
Caroline Tice80eab982015-11-04 14:03:14 -0800219
Manoj Gupta3594db82017-01-31 11:48:57 -0800220 # Reuse autotest files from vanilla image for trybot images
221 autotest_files = os.path.join('/tmp', vanilla_image, 'autotest_files')
Caroline Tice372e1642018-05-24 15:44:12 -0700222 if os.getlogin() == ROLE_ACCOUNT:
223 experiment_image = """
224 %s {
225 chromeos_root: %s
226 build: %s
227 autotest_path: %s
228 compiler: %s
229 chrome_src: /usr/local/google/crostc/chrome-src-internal
230 }
231 """ % (label_string, self._chromeos_root, trybot_image,
232 autotest_files, compiler_string)
233 else:
234 experiment_image = """
Caroline Tice80eab982015-11-04 14:03:14 -0800235 %s {
cmtice46093e52014-12-09 14:59:16 -0800236 chromeos_root: %s
237 build: %s
Manoj Gupta3594db82017-01-31 11:48:57 -0800238 autotest_path: %s
Caroline Ticeddde5052015-09-23 09:43:35 -0700239 compiler: %s
cmtice46093e52014-12-09 14:59:16 -0800240 }
Caroline Tice80eab982015-11-04 14:03:14 -0800241 """ % (label_string, self._chromeos_root, trybot_image,
Manoj Gupta3594db82017-01-31 11:48:57 -0800242 autotest_files, compiler_string)
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800243 f.write(experiment_image)
cmtice46093e52014-12-09 14:59:16 -0800244
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800245 crosperf = os.path.join(TOOLCHAIN_DIR, 'crosperf', 'crosperf')
Han Shen43494292015-09-14 10:26:40 -0700246 noschedv2_opts = '--noschedv2' if self._noschedv2 else ''
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800247 command = ('{crosperf} --no_email=True --results_dir={r_dir} '
Yunlian Jiang4260aae2017-09-06 11:49:15 -0700248 '--json_report=True {noschedv2_opts} {exp_file} '
249 '--locks_dir=/usr/local/google/home/mobiletc-prebuild/locks '
250 '--use_file_locks=True').format(
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800251 crosperf=crosperf,
252 r_dir=self._reports_dir,
253 noschedv2_opts=noschedv2_opts,
254 exp_file=experiment_file)
cmticeaa700b02015-06-12 13:26:47 -0700255
cmtice46093e52014-12-09 14:59:16 -0800256 ret = self._ce.RunCommand(command)
cmtice7f3190b2015-05-22 14:14:51 -0700257 if ret != 0:
Manoj Guptaaee96b72016-10-24 13:43:28 -0700258 raise RuntimeError('Crosperf execution error!')
Caroline Ticeebbc3da2015-09-03 10:27:20 -0700259 else:
260 # Copy json report to pending archives directory.
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800261 command = 'cp %s/*.json %s/.' % (self._reports_dir, PENDING_ARCHIVES_DIR)
Caroline Ticeebbc3da2015-09-03 10:27:20 -0700262 ret = self._ce.RunCommand(command)
cmtice7f3190b2015-05-22 14:14:51 -0700263 return
cmtice46093e52014-12-09 14:59:16 -0800264
cmtice7f3190b2015-05-22 14:14:51 -0700265 def _SendEmail(self):
266 """Find email message generated by crosperf and send it."""
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800267 filename = os.path.join(self._reports_dir, 'msg_body.html')
cmtice7f3190b2015-05-22 14:14:51 -0700268 if (os.path.exists(filename) and
269 os.path.exists(os.path.expanduser(MAIL_PROGRAM))):
Manoj Guptad575b8a2017-03-08 10:51:28 -0800270 email_title = 'buildbot llvm test results'
Manoj Guptac4110352016-12-28 13:47:12 -0800271 if USE_LLVM_NEXT_PATCH in self._patches_string:
272 email_title = 'buildbot llvm_next test results'
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800273 command = ('cat %s | %s -s "%s, %s" -team -html' %
274 (filename, MAIL_PROGRAM, email_title, self._board))
cmtice7f3190b2015-05-22 14:14:51 -0700275 self._ce.RunCommand(command)
cmtice46093e52014-12-09 14:59:16 -0800276
Ting-Yuan Huang6a9a98a2018-03-07 17:35:13 -0800277 def DoAll(self):
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800278 """Main function inside ToolchainComparator class.
cmtice46093e52014-12-09 14:59:16 -0800279
280 Launch trybot, get image names, create crosperf experiment file, run
281 crosperf, and copy images into seven-day report directories.
282 """
Ting-Yuan Huang6a9a98a2018-03-07 17:35:13 -0800283 buildbucket_id, trybot_image = buildbot_utils.GetTrybotImage(
284 self._chromeos_root,
285 self._build,
286 self._patches,
Ting-Yuan Huang6a9a98a2018-03-07 17:35:13 -0800287 tryjob_flags=['--notests'],
288 build_toolchain=True)
cmtice46093e52014-12-09 14:59:16 -0800289
Yunlian Jiange84ea3d2016-12-12 11:07:40 -0800290 print('trybot_url: \
Ting-Yuan Huang6a9a98a2018-03-07 17:35:13 -0800291 http://cros-goldeneye/chromeos/healthmonitoring/buildDetails?buildbucketId=%s'
292 % buildbucket_id)
cmticed54f9802015-02-05 11:04:11 -0800293 if len(trybot_image) == 0:
Caroline Ticeefb79902018-04-18 11:23:01 -0700294 self._l.LogError('Unable to find trybot_image!')
Yunlian Jiangcdd19072017-09-29 10:15:56 -0700295 return 2
Luis Lozano783954f2015-12-21 18:06:29 -0800296
Luis Lozanoc75fd052016-02-19 17:37:01 -0800297 vanilla_image = self._GetVanillaImageName(trybot_image)
298 nonafdo_image = self._GetNonAFDOImageName(trybot_image)
Luis Lozano783954f2015-12-21 18:06:29 -0800299
300 print('trybot_image: %s' % trybot_image)
301 print('vanilla_image: %s' % vanilla_image)
302 print('nonafdo_image: %s' % nonafdo_image)
Luis Lozanoc75fd052016-02-19 17:37:01 -0800303
Luis Lozano783954f2015-12-21 18:06:29 -0800304 self._TestImages(trybot_image, vanilla_image, nonafdo_image)
cmtice7f3190b2015-05-22 14:14:51 -0700305 self._SendEmail()
cmtice46093e52014-12-09 14:59:16 -0800306 return 0
307
308
309def Main(argv):
310 """The main function."""
311
312 # Common initializations
313 command_executer.InitCommandExecuter()
Caroline Ticeeddb0632016-04-14 09:19:02 -0700314 parser = argparse.ArgumentParser()
Manoj Guptaaee96b72016-10-24 13:43:28 -0700315 parser.add_argument(
316 '--remote', dest='remote', help='Remote machines to run tests on.')
317 parser.add_argument(
318 '--board', dest='board', default='x86-zgb', help='The target board.')
319 parser.add_argument(
320 '--chromeos_root',
321 dest='chromeos_root',
322 help='The chromeos root from which to run tests.')
323 parser.add_argument(
324 '--weekday',
325 default='',
326 dest='weekday',
327 help='The day of the week for which to run tests.')
328 parser.add_argument(
329 '--patch',
330 dest='patches',
331 help='The patches to use for the testing, '
332 "seprate the patch numbers with ',' "
333 'for more than one patches.')
334 parser.add_argument(
335 '--noschedv2',
336 dest='noschedv2',
337 action='store_true',
338 default=False,
339 help='Pass --noschedv2 to crosperf.')
Han Shen36413122015-08-28 11:05:40 -0700340
Caroline Ticeeddb0632016-04-14 09:19:02 -0700341 options = parser.parse_args(argv[1:])
cmtice46093e52014-12-09 14:59:16 -0800342 if not options.board:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800343 print('Please give a board.')
cmtice46093e52014-12-09 14:59:16 -0800344 return 1
345 if not options.remote:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800346 print('Please give at least one remote machine.')
cmtice46093e52014-12-09 14:59:16 -0800347 return 1
348 if not options.chromeos_root:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800349 print('Please specify the ChromeOS root directory.')
cmtice46093e52014-12-09 14:59:16 -0800350 return 1
Yunlian Jiange52838c2015-08-20 14:32:37 -0700351
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800352 fc = ToolchainComparator(options.board, options.remote, options.chromeos_root,
Manoj Guptad575b8a2017-03-08 10:51:28 -0800353 options.weekday, options.patches, options.noschedv2)
Ting-Yuan Huang6a9a98a2018-03-07 17:35:13 -0800354 return fc.DoAll()
cmtice46093e52014-12-09 14:59:16 -0800355
356
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800357if __name__ == '__main__':
cmtice46093e52014-12-09 14:59:16 -0800358 retval = Main(sys.argv)
359 sys.exit(retval)