blob: afb704c6f5cdc6b753006d836874ec0b794e1527 [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 Guptad575b8a2017-03-08 10:51:28 -080032USE_LLVM_NEXT_PATCH = '419149'
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
Luis Lozanoc75fd052016-02-19 17:37:01 -080044TRYBOT_IMAGE_FS = 'trybot-' + IMAGE_FS + '-{build_id}'
45PFQ_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()
Luis Lozanof2a3ef42015-12-15 13:49:30 -080074 self._build = '%s-release' % 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,
87 '%s.%s' % (timestamp, board),)
cmtice46093e52014-12-09 14:59:16 -080088
Luis Lozanoc75fd052016-02-19 17:37:01 -080089 def _GetVanillaImageName(self, trybot_image):
Ting-Yuan Huange5819872016-12-15 14:22:26 -080090 """Given a trybot artifact name, get latest vanilla image name.
cmtice46093e52014-12-09 14:59:16 -080091
Luis Lozano783954f2015-12-21 18:06:29 -080092 Args:
93 trybot_image: artifact name such as
94 'trybot-daisy-release/R40-6394.0.0-b1389'
95
96 Returns:
Ting-Yuan Huange5819872016-12-15 14:22:26 -080097 Latest official image name, e.g. 'daisy-release/R57-9089.0.0'.
cmtice46093e52014-12-09 14:59:16 -080098 """
Luis Lozanoc75fd052016-02-19 17:37:01 -080099 mo = re.search(TRYBOT_IMAGE_RE, trybot_image)
100 assert mo
Ting-Yuan Huange5819872016-12-15 14:22:26 -0800101 dirname = IMAGE_DIR.replace('\\', '').format(**mo.groupdict())
102 version = buildbot_utils.GetGSContent(self._chromeos_root,
103 dirname + '/LATEST-master')
104 return dirname + '/' + version
cmtice46093e52014-12-09 14:59:16 -0800105
Luis Lozanoc75fd052016-02-19 17:37:01 -0800106 def _GetNonAFDOImageName(self, trybot_image):
107 """Given a trybot artifact name, get corresponding non-AFDO image name.
108
109 We get the non-AFDO image from the PFQ builders. This image
110 is not generated for all the boards and, the closest PFQ image
111 was the one build for the previous ChromeOS version (the chrome
112 used in the current version is the one validated in the previous
113 version).
114 The previous ChromeOS does not always exist either. So, we try
115 a couple of versions before.
Luis Lozano783954f2015-12-21 18:06:29 -0800116
117 Args:
118 trybot_image: artifact name such as
119 'trybot-daisy-release/R40-6394.0.0-b1389'
120
121 Returns:
122 Corresponding chrome PFQ image name, e.g.
Luis Lozanoc75fd052016-02-19 17:37:01 -0800123 'daisy-chrome-pfq/R40-6393.0.0-rc1'.
Luis Lozano783954f2015-12-21 18:06:29 -0800124 """
Luis Lozanoc75fd052016-02-19 17:37:01 -0800125 mo = re.search(TRYBOT_IMAGE_RE, trybot_image)
126 assert mo
127 image_dict = mo.groupdict()
128 image_dict['image_type'] = 'chrome-pfq'
129 for _ in xrange(2):
130 image_dict['tip'] = str(int(image_dict['tip']) - 1)
131 nonafdo_image = PFQ_IMAGE_FS.replace('\\', '').format(**image_dict)
132 if buildbot_utils.DoesImageExist(self._chromeos_root, nonafdo_image):
133 return nonafdo_image
134 return ''
Luis Lozano783954f2015-12-21 18:06:29 -0800135
cmtice46093e52014-12-09 14:59:16 -0800136 def _FinishSetup(self):
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800137 """Make sure testing_rsa file is properly set up."""
cmtice46093e52014-12-09 14:59:16 -0800138 # Fix protections on ssh key
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800139 command = ('chmod 600 /var/cache/chromeos-cache/distfiles/target'
140 '/chrome-src-internal/src/third_party/chromite/ssh_keys'
141 '/testing_rsa')
cmtice46093e52014-12-09 14:59:16 -0800142 ret_val = self._ce.ChrootRunCommand(self._chromeos_root, command)
cmtice7f3190b2015-05-22 14:14:51 -0700143 if ret_val != 0:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800144 raise RuntimeError('chmod for testing_rsa failed')
cmtice46093e52014-12-09 14:59:16 -0800145
Luis Lozano783954f2015-12-21 18:06:29 -0800146 def _TestImages(self, trybot_image, vanilla_image, nonafdo_image):
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800147 """Create crosperf experiment file.
cmtice46093e52014-12-09 14:59:16 -0800148
Luis Lozano783954f2015-12-21 18:06:29 -0800149 Given the names of the trybot, vanilla and non-AFDO images, create the
cmtice46093e52014-12-09 14:59:16 -0800150 appropriate crosperf experiment file and launch crosperf on it.
151 """
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800152 experiment_file_dir = os.path.join(self._chromeos_root, '..', self._weekday)
153 experiment_file_name = '%s_toolchain_experiment.txt' % self._board
Yunlian Jiang2f563562015-08-28 13:54:04 -0700154
Manoj Guptad575b8a2017-03-08 10:51:28 -0800155 compiler_string = 'llvm'
Manoj Guptac4110352016-12-28 13:47:12 -0800156 if USE_LLVM_NEXT_PATCH in self._patches_string:
157 experiment_file_name = '%s_llvm_next_experiment.txt' % self._board
158 compiler_string = 'llvm_next'
Yunlian Jiang2f563562015-08-28 13:54:04 -0700159
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800160 experiment_file = os.path.join(experiment_file_dir, experiment_file_name)
cmtice46093e52014-12-09 14:59:16 -0800161 experiment_header = """
162 board: %s
163 remote: %s
Luis Lozanoe1efeb82015-06-16 16:35:44 -0700164 retries: 1
cmtice46093e52014-12-09 14:59:16 -0800165 """ % (self._board, self._remotes)
166 experiment_tests = """
Luis Lozano1489d642015-12-08 10:08:19 -0800167 benchmark: all_toolchain_perf {
cmtice46093e52014-12-09 14:59:16 -0800168 suite: telemetry_Crosperf
Ting-Yuan Huang83323642017-02-17 12:20:02 -0800169 iterations: 0
cmtice46093e52014-12-09 14:59:16 -0800170 }
Caroline Ticee82513b2016-10-27 12:45:15 -0700171
172 benchmark: page_cycler_v2.typical_25 {
173 suite: telemetry_Crosperf
Ting-Yuan Huang83323642017-02-17 12:20:02 -0800174 iterations: 0
Caroline Ticee82513b2016-10-27 12:45:15 -0700175 run_local: False
176 retries: 0
177 }
cmtice46093e52014-12-09 14:59:16 -0800178 """
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800179
180 with open(experiment_file, 'w') as f:
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800181 f.write(experiment_header)
182 f.write(experiment_tests)
cmtice46093e52014-12-09 14:59:16 -0800183
184 # Now add vanilla to test file.
185 official_image = """
186 vanilla_image {
187 chromeos_root: %s
188 build: %s
Caroline Ticeddde5052015-09-23 09:43:35 -0700189 compiler: gcc
cmtice46093e52014-12-09 14:59:16 -0800190 }
191 """ % (self._chromeos_root, vanilla_image)
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800192 f.write(official_image)
cmtice46093e52014-12-09 14:59:16 -0800193
Luis Lozano783954f2015-12-21 18:06:29 -0800194 # Now add non-AFDO image to test file.
Luis Lozano439f2b72016-01-08 11:56:02 -0800195 if nonafdo_image:
196 official_nonafdo_image = """
Luis Lozano783954f2015-12-21 18:06:29 -0800197 nonafdo_image {
198 chromeos_root: %s
199 build: %s
200 compiler: gcc
201 }
202 """ % (self._chromeos_root, nonafdo_image)
Luis Lozano439f2b72016-01-08 11:56:02 -0800203 f.write(official_nonafdo_image)
Luis Lozano783954f2015-12-21 18:06:29 -0800204
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800205 label_string = '%s_trybot_image' % compiler_string
Caroline Tice80eab982015-11-04 14:03:14 -0800206
Manoj Gupta3594db82017-01-31 11:48:57 -0800207 # Reuse autotest files from vanilla image for trybot images
208 autotest_files = os.path.join('/tmp', vanilla_image, 'autotest_files')
cmtice46093e52014-12-09 14:59:16 -0800209 experiment_image = """
Caroline Tice80eab982015-11-04 14:03:14 -0800210 %s {
cmtice46093e52014-12-09 14:59:16 -0800211 chromeos_root: %s
212 build: %s
Manoj Gupta3594db82017-01-31 11:48:57 -0800213 autotest_path: %s
Caroline Ticeddde5052015-09-23 09:43:35 -0700214 compiler: %s
cmtice46093e52014-12-09 14:59:16 -0800215 }
Caroline Tice80eab982015-11-04 14:03:14 -0800216 """ % (label_string, self._chromeos_root, trybot_image,
Manoj Gupta3594db82017-01-31 11:48:57 -0800217 autotest_files, compiler_string)
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800218 f.write(experiment_image)
cmtice46093e52014-12-09 14:59:16 -0800219
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800220 crosperf = os.path.join(TOOLCHAIN_DIR, 'crosperf', 'crosperf')
Han Shen43494292015-09-14 10:26:40 -0700221 noschedv2_opts = '--noschedv2' if self._noschedv2 else ''
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800222 command = ('{crosperf} --no_email=True --results_dir={r_dir} '
223 '--json_report=True {noschedv2_opts} {exp_file}').format(
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800224 crosperf=crosperf,
225 r_dir=self._reports_dir,
226 noschedv2_opts=noschedv2_opts,
227 exp_file=experiment_file)
cmticeaa700b02015-06-12 13:26:47 -0700228
cmtice46093e52014-12-09 14:59:16 -0800229 ret = self._ce.RunCommand(command)
cmtice7f3190b2015-05-22 14:14:51 -0700230 if ret != 0:
Manoj Guptaaee96b72016-10-24 13:43:28 -0700231 raise RuntimeError('Crosperf execution error!')
Caroline Ticeebbc3da2015-09-03 10:27:20 -0700232 else:
233 # Copy json report to pending archives directory.
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800234 command = 'cp %s/*.json %s/.' % (self._reports_dir, PENDING_ARCHIVES_DIR)
Caroline Ticeebbc3da2015-09-03 10:27:20 -0700235 ret = self._ce.RunCommand(command)
cmtice7f3190b2015-05-22 14:14:51 -0700236 return
cmtice46093e52014-12-09 14:59:16 -0800237
cmtice7f3190b2015-05-22 14:14:51 -0700238 def _SendEmail(self):
239 """Find email message generated by crosperf and send it."""
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800240 filename = os.path.join(self._reports_dir, 'msg_body.html')
cmtice7f3190b2015-05-22 14:14:51 -0700241 if (os.path.exists(filename) and
242 os.path.exists(os.path.expanduser(MAIL_PROGRAM))):
Manoj Guptad575b8a2017-03-08 10:51:28 -0800243 email_title = 'buildbot llvm test results'
Manoj Guptac4110352016-12-28 13:47:12 -0800244 if USE_LLVM_NEXT_PATCH in self._patches_string:
245 email_title = 'buildbot llvm_next test results'
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800246 command = ('cat %s | %s -s "%s, %s" -team -html' %
247 (filename, MAIL_PROGRAM, email_title, self._board))
cmtice7f3190b2015-05-22 14:14:51 -0700248 self._ce.RunCommand(command)
cmtice46093e52014-12-09 14:59:16 -0800249
250 def DoAll(self):
Yunlian Jiang14cf5962015-12-11 15:50:14 -0800251 """Main function inside ToolchainComparator class.
cmtice46093e52014-12-09 14:59:16 -0800252
253 Launch trybot, get image names, create crosperf experiment file, run
254 crosperf, and copy images into seven-day report directories.
255 """
cmticece5ffa42015-02-12 15:18:43 -0800256 date_str = datetime.date.today()
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800257 description = 'master_%s_%s_%s' % (self._patches_string, self._build,
Han Shenfe054f12015-02-18 15:00:13 -0800258 date_str)
Yunlian Jiange84ea3d2016-12-12 11:07:40 -0800259 build_id, trybot_image = buildbot_utils.GetTrybotImage(
Manoj Guptaaee96b72016-10-24 13:43:28 -0700260 self._chromeos_root,
261 self._build,
262 self._patches,
263 description,
264 other_flags=['--notests'],
265 build_toolchain=True)
cmtice46093e52014-12-09 14:59:16 -0800266
Yunlian Jiange84ea3d2016-12-12 11:07:40 -0800267 print('trybot_url: \
Manoj Guptac4110352016-12-28 13:47:12 -0800268 https://uberchromegw.corp.google.com/i/chromiumos.tryserver/builders/release/builds/%s'
269 % build_id)
cmticed54f9802015-02-05 11:04:11 -0800270 if len(trybot_image) == 0:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800271 self._l.LogError('Unable to find trybot_image for %s!' % description)
Luis Lozano7f20acb2015-11-04 17:15:08 -0800272 return 1
Luis Lozano783954f2015-12-21 18:06:29 -0800273
Luis Lozanoc75fd052016-02-19 17:37:01 -0800274 vanilla_image = self._GetVanillaImageName(trybot_image)
275 nonafdo_image = self._GetNonAFDOImageName(trybot_image)
Luis Lozano783954f2015-12-21 18:06:29 -0800276
277 print('trybot_image: %s' % trybot_image)
278 print('vanilla_image: %s' % vanilla_image)
279 print('nonafdo_image: %s' % nonafdo_image)
Luis Lozanoc75fd052016-02-19 17:37:01 -0800280
cmtice46093e52014-12-09 14:59:16 -0800281 if os.getlogin() == ROLE_ACCOUNT:
282 self._FinishSetup()
283
Luis Lozano783954f2015-12-21 18:06:29 -0800284 self._TestImages(trybot_image, vanilla_image, nonafdo_image)
cmtice7f3190b2015-05-22 14:14:51 -0700285 self._SendEmail()
cmtice46093e52014-12-09 14:59:16 -0800286 return 0
287
288
289def Main(argv):
290 """The main function."""
291
292 # Common initializations
293 command_executer.InitCommandExecuter()
Caroline Ticeeddb0632016-04-14 09:19:02 -0700294 parser = argparse.ArgumentParser()
Manoj Guptaaee96b72016-10-24 13:43:28 -0700295 parser.add_argument(
296 '--remote', dest='remote', help='Remote machines to run tests on.')
297 parser.add_argument(
298 '--board', dest='board', default='x86-zgb', help='The target board.')
299 parser.add_argument(
300 '--chromeos_root',
301 dest='chromeos_root',
302 help='The chromeos root from which to run tests.')
303 parser.add_argument(
304 '--weekday',
305 default='',
306 dest='weekday',
307 help='The day of the week for which to run tests.')
308 parser.add_argument(
309 '--patch',
310 dest='patches',
311 help='The patches to use for the testing, '
312 "seprate the patch numbers with ',' "
313 'for more than one patches.')
314 parser.add_argument(
315 '--noschedv2',
316 dest='noschedv2',
317 action='store_true',
318 default=False,
319 help='Pass --noschedv2 to crosperf.')
Han Shen36413122015-08-28 11:05:40 -0700320
Caroline Ticeeddb0632016-04-14 09:19:02 -0700321 options = parser.parse_args(argv[1:])
cmtice46093e52014-12-09 14:59:16 -0800322 if not options.board:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800323 print('Please give a board.')
cmtice46093e52014-12-09 14:59:16 -0800324 return 1
325 if not options.remote:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800326 print('Please give at least one remote machine.')
cmtice46093e52014-12-09 14:59:16 -0800327 return 1
328 if not options.chromeos_root:
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800329 print('Please specify the ChromeOS root directory.')
cmtice46093e52014-12-09 14:59:16 -0800330 return 1
Yunlian Jiange52838c2015-08-20 14:32:37 -0700331
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800332 fc = ToolchainComparator(options.board, options.remote, options.chromeos_root,
Manoj Guptad575b8a2017-03-08 10:51:28 -0800333 options.weekday, options.patches, options.noschedv2)
cmtice46093e52014-12-09 14:59:16 -0800334 return fc.DoAll()
335
336
Luis Lozanof2a3ef42015-12-15 13:49:30 -0800337if __name__ == '__main__':
cmtice46093e52014-12-09 14:59:16 -0800338 retval = Main(sys.argv)
339 sys.exit(retval)