blob: 8b942345167d114ca21e87c611146688276f531c [file] [log] [blame]
Derek Beckettf73baca2020-08-19 15:08:47 -07001# Lint as: python2, python3
J. Richard Barnette24adbf42012-04-11 15:04:53 -07002# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Dale Curtisaa5eedb2011-08-23 16:18:52 -07003# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
Derek Beckettf73baca2020-08-19 15:08:47 -07006from __future__ import absolute_import
7from __future__ import division
8from __future__ import print_function
9
J. Richard Barnette1d78b012012-05-15 13:56:30 -070010import logging
Dan Shi0f466e82013-02-22 15:44:58 -080011import os
Simran Basid5e5e272012-09-24 15:23:59 -070012import re
Vincent Palatindf2372c2016-10-07 17:03:00 +020013import sys
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070014import time
15
mussa584b4462014-06-20 15:13:28 -070016import common
J. Richard Barnette45e93de2012-04-11 17:24:15 -070017from autotest_lib.client.bin import utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070018from autotest_lib.client.common_lib import autotemp
Richard Barnette0c73ffc2012-11-19 15:21:18 -080019from autotest_lib.client.common_lib import error
20from autotest_lib.client.common_lib import global_config
J. Richard Barnette91137f02016-03-10 16:52:26 -080021from autotest_lib.client.common_lib import hosts
Dan Shi549fb822015-03-24 18:01:11 -070022from autotest_lib.client.common_lib import lsbrelease_utils
Otabek Kasimov6825b762020-06-23 23:42:44 -070023from autotest_lib.client.common_lib import utils as common_utils
Greg Edelstona7b05d12020-04-01 16:00:51 -060024from autotest_lib.client.common_lib.cros import cros_config
Richard Barnette03a0c132012-11-05 12:40:35 -080025from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070026from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000027from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080028from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080029from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070030from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070031from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050032from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070033from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Garry Wang1a493d82020-08-31 21:01:19 -070034from autotest_lib.server.cros.device_health_profile import device_health_profile
Garry Wanga2e78172020-09-09 23:49:07 -070035from autotest_lib.server.cros.device_health_profile import profile_constants
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070036from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070037from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070038from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080039from autotest_lib.server.hosts import chameleon_host
Otabek Kasimov832d9162020-07-27 19:24:57 -070040from autotest_lib.server.hosts import cros_constants
Richard Barnetted31580e2018-05-14 19:58:00 +000041from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080042from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070043from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070044from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070045from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080046from autotest_lib.site_utils.rpm_control_system import rpm_client
Otabek Kasimov808cd832020-05-28 18:27:46 -070047from autotest_lib.site_utils.admin_audit import constants as audit_const
Otabek Kasimov27bb2862020-08-10 14:40:45 -070048from autotest_lib.site_utils.admin_audit import verifiers as audit_verify
Derek Beckettf73baca2020-08-19 15:08:47 -070049from six.moves import zip
Simran Basid5e5e272012-09-24 15:23:59 -070050
Simran Basi382506b2016-09-13 14:58:15 -070051# In case cros_host is being ran via SSP on an older Moblab version with an
52# older chromite version.
53try:
54 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080055except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080056 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080057
Simran Basid5e5e272012-09-24 15:23:59 -070058
Dan Shib8540a52015-07-16 14:18:23 -070059CONFIG = global_config.global_config
60
beepsc87ff602013-07-31 21:53:00 -070061class FactoryImageCheckerException(error.AutoservError):
62 """Exception raised when an image is a factory image."""
63 pass
64
65
Fang Deng0ca40e22013-08-27 17:47:44 -070066class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070067 """Chromium OS specific subclass of Host."""
68
Simran Basi5ace6f22016-01-06 17:30:44 -080069 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
70
Scott Zawalski62bacae2013-03-05 10:40:32 -050071 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070072
Richard Barnette03a0c132012-11-05 12:40:35 -080073 # Timeout values (in seconds) associated with various Chrome OS
74 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070075 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080076 # In general, a good rule of thumb is that the timeout can be up
77 # to twice the typical measured value on the slowest platform.
78 # The times here have not necessarily been empirically tested to
79 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070080 #
81 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080082 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
83 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080084 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070085 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080086 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070087 # screen delay, time to start the network on the DUT, and the
88 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070089 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080090 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080091 # network.
beepsf079cfb2013-09-18 17:49:51 -070092 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080093 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
94 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070095
96 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080097 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080098 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070099 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -0700100 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -0700101 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -0800102 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -0700103
Dan Shica503482015-03-30 17:23:25 -0700104 # Minimum OS version that supports server side packaging. Older builds may
105 # not have server side package built or with Autotest code change to support
106 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -0700107 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -0700108 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700109
Dana Goyettec172b172020-07-29 16:26:15 -0700110 USE_FSFREEZE = CONFIG.get_config_value(
Dana Goyette6242cb32020-09-23 11:02:57 -0700111 'CROS', 'enable_fs_freeze', type=bool, default=False)
Dana Goyettec172b172020-07-29 16:26:15 -0700112
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800113 # REBOOT_TIMEOUT: How long to wait for a reboot.
114 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700115 # We have a long timeout to ensure we don't flakily fail due to other
116 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700117 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
118 # return from reboot' bug is solved.
119 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700120
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800121 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
122 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700123 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
124 # since changing servo role will reset USB state
125 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800126 _USB_POWER_TIMEOUT = 5
127 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700128 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800129
Fang Dengdeba14f2014-11-14 11:54:09 -0800130 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
131 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700132
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800133 # Constants used in ping_wait_up() and ping_wait_down().
134 #
135 # _PING_WAIT_COUNT is the approximate number of polling
136 # cycles to use when waiting for a host state change.
137 #
138 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
139 # for arguments to the internal _ping_wait_for_status()
140 # method.
141 _PING_WAIT_COUNT = 40
142 _PING_STATUS_DOWN = False
143 _PING_STATUS_UP = True
144
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800145 # Allowed values for the power_method argument.
146
Garry Wang5e5538a2019-04-08 15:36:18 -0700147 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
148 # DUTs except those with servo_v4 CCD.
149 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
150 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800151 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
152 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
153 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700154 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800155 POWER_CONTROL_SERVO = 'servoj10'
156 POWER_CONTROL_MANUAL = 'manual'
157
158 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700159 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800160 POWER_CONTROL_SERVO,
161 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800162
Simran Basi5e6339a2013-03-21 11:34:32 -0700163 _RPM_OUTLET_CHANGED = 'outlet_changed'
164
Dan Shi9cb0eec2014-06-03 09:04:50 -0700165 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700166 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700167 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700168
Brent Peterson1cb623a2020-01-09 13:14:28 -0800169 # Regular expression for extracting EC version string
170 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
171
172 # Regular expression for extracting BIOS version string
173 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
174
Brent Petersonc70a1832020-01-24 15:54:35 -0800175 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700176 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800177
J. Richard Barnette964fba02012-10-24 17:34:29 -0700178 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800179 def check_host(host, timeout=10):
180 """
181 Check if the given host is a chrome-os host.
182
183 @param host: An ssh host representing a device.
184 @param timeout: The timeout for the run command.
185
186 @return: True if the host device is chromeos.
187
beeps46dadc92013-11-07 14:07:10 -0800188 """
189 try:
Allen Liad719c12017-06-27 23:48:04 +0000190 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700191 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700192 '! grep -q moblab /etc/lsb-release && '
193 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700194 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700195 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000196 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700197 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
198 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800199 return not (
200 lsbrelease_utils.is_jetstream(
201 lsb_release_content=lsb_release_content) or
202 lsbrelease_utils.is_gce_board(
203 lsb_release_content=lsb_release_content))
204
beeps46dadc92013-11-07 14:07:10 -0800205 except (error.AutoservRunError, error.AutoservSSHTimeout):
206 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700207
208 return False
beeps46dadc92013-11-07 14:07:10 -0800209
210
211 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800212 def get_chameleon_arguments(args_dict):
213 """Extract chameleon options from `args_dict` and return the result.
214
215 Recommended usage:
216 ~~~~~~~~
217 args_dict = utils.args_to_dict(args)
218 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
219 host = hosts.create_host(machine, chameleon_args=chameleon_args)
220 ~~~~~~~~
221
222 @param args_dict Dictionary from which to extract the chameleon
223 arguments.
224 """
Sam McNally66594ca2019-12-09 12:45:44 +1100225 chameleon_args = {key: args_dict[key]
226 for key in ('chameleon_host', 'chameleon_port')
227 if key in args_dict}
228 if 'chameleon_ssh_port' in args_dict:
229 chameleon_args['port'] = int(args_dict['chameleon_ssh_port'])
230 return chameleon_args
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800231
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800232 @staticmethod
233 def get_btpeer_arguments(args_dict):
234 """Extract btpeer options from `args_dict` and return the result.
235
236 This is used to parse details of Bluetooth peer.
237 Recommended usage:
238 ~~~~~~~~
239 args_dict = utils.args_to_dict(args)
240 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700241 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800242 ~~~~~~~~
243
244 @param args_dict: Dictionary from which to extract the btpeer
245 arguments.
246 """
247 if 'btpeer_host_list' in args_dict:
248 result = []
249 for btpeer in args_dict['btpeer_host_list'].split(','):
250 result.append({key: value for key,value in
251 zip(('btpeer_host','btpeer_port'),
252 btpeer.split(':'))})
253 return result
254 else:
Anand K Mistrye8933092020-08-05 14:49:41 +1000255 return {key: args_dict[key]
256 for key in ('btpeer_host', 'btpeer_port', 'btpeer_ssh_port')
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800257 if key in args_dict}
258
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800259
260 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700261 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800262 """Extract chameleon options from `args_dict` and return the result.
263
264 Recommended usage:
265 ~~~~~~~~
266 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700267 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
268 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800269 ~~~~~~~~
270
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700271 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800272 arguments.
273 """
Allen Li083866b2016-08-18 10:07:10 -0700274 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700275 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700276 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800277
278
279 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800280 def get_servo_arguments(args_dict):
281 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800282
283 Recommended usage:
284 ~~~~~~~~
285 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700286 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800287 host = hosts.create_host(machine, servo_args=servo_args)
288 ~~~~~~~~
289
290 @param args_dict Dictionary from which to extract the servo
291 arguments.
292 """
Garry Wang11b5e872020-03-11 15:14:08 -0700293 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
294 servo_constants.SERVO_PORT_ATTR,
Otabek Kasimov382c3bb2020-10-28 13:22:45 -0700295 servo_constants.SERVO_SERIAL_ATTR,
Garry Wang11b5e872020-03-11 15:14:08 -0700296 servo_constants.SERVO_BOARD_ATTR,
297 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200298 servo_args = {key: args_dict[key]
299 for key in servo_attrs
300 if key in args_dict}
301 return (
302 None
Garry Wang11b5e872020-03-11 15:14:08 -0700303 if servo_constants.SERVO_HOST_ATTR in servo_args
304 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200305 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700306
J. Richard Barnette964fba02012-10-24 17:34:29 -0700307
J. Richard Barnette91137f02016-03-10 16:52:26 -0800308 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700309 pdtester_args=None, try_lab_servo=False,
Shijin Abraham78ce4402020-09-08 22:04:27 -0700310 try_servo_repair=False, ssh_verbosity_flag='',
311 ssh_options='', *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800312 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700313
Fang Denge545abb2014-12-30 18:43:47 -0800314 This method will attempt to create the test-assistant object
315 (chameleon/servo) when it is needed by the test. Check
316 the docstring of chameleon_host.create_chameleon_host and
317 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700318
Fang Denge545abb2014-12-30 18:43:47 -0800319 @param hostname: Hostname of the dut.
320 @param chameleon_args: A dictionary that contains args for creating
321 a ChameleonHost. See chameleon_host for details.
322 @param servo_args: A dictionary that contains args for creating
323 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700324 @param try_lab_servo: When true, indicates that an attempt should
325 be made to create a ServoHost for a DUT in
326 the test lab, even if not required by
327 `servo_args`. See servo_host for details.
328 @param try_servo_repair: If a servo host is created, check it
329 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800330 See servo_host for details.
331 @param ssh_verbosity_flag: String, to pass to the ssh command to control
332 verbosity.
333 @param ssh_options: String, other ssh options to pass to the ssh
334 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700335 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700336 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700337 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800338 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700339 # hold special dut_state for repair process
340 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700341 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700342 # self.env is a dictionary of environment variable settings
343 # to be exported for commands run on the host.
344 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
345 # errors that might happen.
346 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700347 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700348 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700349 _servo_host, servo_state = servo_host.create_servo_host(
350 dut=self,
351 servo_args=servo_args,
352 try_lab_servo=try_lab_servo,
353 try_servo_repair=try_servo_repair,
354 dut_host_info=self.host_info_store.get())
355 self.set_servo_host(_servo_host, servo_state)
Garry Wang1a493d82020-08-31 21:01:19 -0700356 self.health_profile = None
Garry Wang5e5538a2019-04-08 15:36:18 -0700357 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700358
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800359 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800360 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700361 dut=self.hostname,
362 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800363 if self._chameleon_host:
364 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800365 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800366 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700367
Shijin Abraham78ce4402020-09-08 22:04:27 -0700368 # Bluetooth peers will be populated by the test if needed
369 self._btpeer_host_list = []
370 self.btpeer_list = []
371 self.btpeer = None
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800372
howardchung83e55272019-08-08 14:08:05 +0800373 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700374 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700375 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800376
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700377 if self._pdtester_host:
378 self.pdtester_servo = self._pdtester_host.get_servo()
379 logging.info('pdtester_servo: %r', self.pdtester_servo)
380 # Create the pdtester object used to access the ec uart
381 self.pdtester = pdtester.PDTester(self.pdtester_servo,
382 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800383 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700384 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800385
Fang Deng5d518f42013-08-02 14:04:32 -0700386
Shijin Abraham78ce4402020-09-08 22:04:27 -0700387 def initialize_btpeer(self, btpeer_args=[]):
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800388 """ Initialize the Bluetooth peers
389
390 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
391 is chameleon host on Raspberry Pi.
392 @param btpeer_args: A dictionary that contains args for creating
393 a ChameleonHost. See chameleon_host for details.
394
395 """
Shijin Abraham78ce4402020-09-08 22:04:27 -0700396 logging.debug('Attempting to initialize bluetooth peers if available')
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700397 try:
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700398 if type(btpeer_args) is list:
399 btpeer_args_list = btpeer_args
400 else:
401 btpeer_args_list = [btpeer_args]
402
403 self._btpeer_host_list = chameleon_host.create_btpeer_host(
404 dut=self.hostname, btpeer_args_list=btpeer_args_list)
405 logging.debug('Bluetooth peer hosts are %s',
406 self._btpeer_host_list)
407 self.btpeer_list = [_host.create_chameleon_board() for _host in
408 self._btpeer_host_list if _host is not None]
409
410 if len(self.btpeer_list) > 0:
411 self.btpeer = self.btpeer_list[0]
412
413 logging.debug('After initialize_btpeer btpeer_list %s '
414 'btpeer_host_list is %s and btpeer is %s',
415 self.btpeer_list, self._btpeer_host_list,
416 self.btpeer)
417 except Exception as e:
418 logging.error('Exception %s in initialize_btpeer', str(e))
419
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800420
421
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000422 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700423 """Get latest stable cros image name from AFE.
424
425 Use the board name from the info store. Should that fail, try to
426 retrieve the board name from the host's installed image itself.
427
428 @returns: current stable cros image name for this host.
429 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700430 info = self.host_info_store.get()
431 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700432 logging.warn('No board label value found. Trying to infer '
433 'from the host itself.')
434 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700435 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700436 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
437 logging.error('Also failed to get the board name from the DUT '
438 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700439 raise error.AutoservError('Cannot determine board of the DUT'
440 ' while getting repair image name.')
441 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500442
443
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700444 def host_version_prefix(self, image):
445 """Return version label prefix.
446
447 In case the CrOS provisioning version is something other than the
448 standard CrOS version e.g. CrOS TH version, this function will
449 find the prefix from provision.py.
450
451 @param image: The image name to find its version prefix.
452 @returns: A prefix string for the image type.
453 """
454 return provision.get_version_label_prefix(image)
455
Andrew Luo3332ab22020-04-28 16:42:03 -0700456 def stage_build_to_usb(self, build):
457 """Stage the current ChromeOS image on the USB stick connected to the
458 servo.
459
460 @param build: The build to download and send to USB.
461 """
462 if not self.servo:
463 raise error.TestError('Host %s does not have servo.' %
464 self.hostname)
465
466 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700467
468 try:
469 self.servo.image_to_servo_usb(update_url)
470 finally:
471 # servo.image_to_servo_usb turned the DUT off, so turn it back on
472 logging.debug('Turn DUT power back on.')
473 self.servo.get_power_state_controller().power_on()
474
Andrew Luo3332ab22020-04-28 16:42:03 -0700475 logging.debug('ChromeOS image %s is staged on the USB stick.',
476 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700477
beepsdae65fd2013-07-26 16:24:41 -0700478 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700479 """
480 Make sure job_repo_url of this host is valid.
481
joychen03eaad92013-06-26 09:55:21 -0700482 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700483 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
484 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
485 download and extract it. If the devserver embedded in the url is
486 unresponsive, update the job_repo_url of the host after staging it on
487 another devserver.
488
489 @param job_repo_url: A url pointing to the devserver where the autotest
490 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700491 @param tag: The tag from the server job, in the format
492 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700493
494 @raises DevServerException: If we could not resolve a devserver.
495 @raises AutoservError: If we're unable to save the new job_repo_url as
496 a result of choosing a new devserver because the old one failed to
497 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700498 @raises urllib2.URLError: If the devserver embedded in job_repo_url
499 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700500 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800501 info = self.host_info_store.get()
502 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700503 if not job_repo_url:
504 logging.warning('No job repo url set on host %s', self.hostname)
505 return
506
507 logging.info('Verifying job repo url %s', job_repo_url)
508 devserver_url, image_name = tools.get_devserver_build_from_package_url(
509 job_repo_url)
510
beeps0c865032013-07-30 11:37:06 -0700511 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700512
513 logging.info('Staging autotest artifacts for %s on devserver %s',
514 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700515
516 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700517 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700518 stage_time = time.time() - start_time
519
520 # Record how much of the verification time comes from a devserver
521 # restage. If we're doing things right we should not see multiple
522 # devservers for a given board/build/branch path.
523 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800524 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700525 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800526 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700527 pass
528 else:
beeps0c865032013-07-30 11:37:06 -0700529 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700530 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700531 stats_key = {
532 'board': board,
533 'build_type': build_type,
534 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700535 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700536 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800537
538 monarch_fields = {
539 'board': board,
540 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800541 'branch': branch,
542 'dev_server': devserver,
543 }
544 metrics.Counter(
545 'chromeos/autotest/provision/verify_url'
546 ).increment(fields=monarch_fields)
547 metrics.SecondsDistribution(
548 'chromeos/autotest/provision/verify_url_duration'
549 ).add(stage_time, fields=monarch_fields)
550
551
Dan Shicf4d2032015-03-12 15:04:21 -0700552 def stage_server_side_package(self, image=None):
553 """Stage autotest server-side package on devserver.
554
555 @param image: Full path of an OS image to install or a build name.
556
557 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700558
559 @raise: error.AutoservError if fail to locate the build to test with, or
560 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700561 """
Dan Shid37736b2016-07-06 15:10:29 -0700562 # If enable_drone_in_restricted_subnet is False, do not set hostname
563 # in devserver.resolve call, so a devserver in non-restricted subnet
564 # is picked to stage autotest server package for drone to download.
565 hostname = self.hostname
566 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
567 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700568 if image:
569 image_name = tools.get_build_from_image(image)
570 if not image_name:
571 raise error.AutoservError(
572 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700573 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700574 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700575 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800576 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700577 if job_repo_url:
578 devserver_url, image_name = (
579 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700580 # If enable_drone_in_restricted_subnet is True, use the
581 # existing devserver. Otherwise, resolve a new one in
582 # non-restricted subnet.
583 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
584 ds = dev_server.ImageServer(devserver_url)
585 else:
586 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800587 elif info.build is not None:
588 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700589 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700590 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800591 raise error.AutoservError(
592 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700593 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700594
595 # Get the OS version of the build, for any build older than
596 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
597 match = re.match('.*/R\d+-(\d+)\.', image_name)
598 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700599 raise error.AutoservError(
600 'Build %s is older than %s. Server side packaging is '
601 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700602
Dan Shicf4d2032015-03-12 15:04:21 -0700603 ds.stage_artifacts(image_name, ['autotest_server_package'])
604 return '%s/static/%s/%s' % (ds.url(), image_name,
605 'autotest_server_package.tar.bz2')
606
607
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700608 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700609 """Stage a build on a devserver and return the update_url.
610
611 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700612 @param artifact: a string like 'test_image'. Requests
613 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700614 @returns a tuple of (image_name, URL) like
615 (lumpy-release/R27-3837.0.0,
616 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700617 """
618 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000619 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700620 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800621 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700622 devserver.stage_artifacts(image_name, [artifact])
623 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700624 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700625 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700626 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700627 else:
628 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700629
630
beepse539be02013-07-31 21:57:39 -0700631 def stage_factory_image_for_servo(self, image_name):
632 """Stage a build on a devserver and return the update_url.
633
634 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700635
beepse539be02013-07-31 21:57:39 -0700636 @return: An update URL, eg:
637 http://<devserver>/static/canary-channel/\
638 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700639
640 @raises: ValueError if the factory artifact name is missing from
641 the config.
642
beepse539be02013-07-31 21:57:39 -0700643 """
644 if not image_name:
645 logging.error('Need an image_name to stage a factory image.')
646 return
647
Dan Shib8540a52015-07-16 14:18:23 -0700648 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700649 'CROS', 'factory_artifact', type=str, default='')
650 if not factory_artifact:
651 raise ValueError('Cannot retrieve the factory artifact name from '
652 'autotest config, and hence cannot stage factory '
653 'artifacts.')
654
beepse539be02013-07-31 21:57:39 -0700655 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800656 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700657 devserver.stage_artifacts(
658 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700659 [factory_artifact],
660 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700661
662 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
663
664
Laurence Goodby778c9a42017-05-24 19:24:07 -0700665 def prepare_for_update(self):
666 """Prepares the DUT for an update.
667
668 Subclasses may override this to perform any special actions
669 required before updating.
670 """
Laurence Goodby468de252017-06-08 17:22:53 -0700671 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700672
673
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800674 def _clear_fw_version_labels(self, rw_only):
675 """Clear firmware version labels from the machine.
676
677 @param rw_only: True to only clear fwrw_version; otherewise, clear
678 both fwro_version and fwrw_version.
679 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700680 info = self.host_info_store.get()
681 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800682 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700683 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
684 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700685
686
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800687 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700688 """Add firmware version label to the machine.
689
690 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800691 @param rw_only: True to only add fwrw_version; otherwise, add both
692 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700693
694 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700695 info = self.host_info_store.get()
696 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800697 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700698 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
699 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700700
701
Namyoon Woo33f38852020-04-13 17:26:58 -0700702 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800703 """Search for the latest package release version from the image archive,
704 and return it.
705
Namyoon Woo33f38852020-04-13 17:26:58 -0700706 @param platform: platform name, a.k.a. board or model
707 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800708
Namyoon Woo33f38852020-04-13 17:26:58 -0700709 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
710 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800711 or None if LATEST release file does not exist.
712 """
713
Namyoon Woo33f38852020-04-13 17:26:58 -0700714 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800715
Namyoon Woo33f38852020-04-13 17:26:58 -0700716 # Search the image path in reference board archive as well.
717 # For example, bob has its binary image under its reference board (gru)
718 # image archive.
719 if ref_board:
720 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800721
Namyoon Woo33f38852020-04-13 17:26:58 -0700722 for board in platforms:
723 # Read 'LATEST-1.0.0' file
724 branch_dir = provision.FW_BRANCH_GLOB % board
725 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
726 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800727
Namyoon Woo33f38852020-04-13 17:26:58 -0700728 try:
729 # The result could be one or more.
730 result = utils.system_output('gsutil ls -d ' + latest_file)
731
732 candidates = re.findall('gs://.*', result)
733
734 # Found the directory candidates. No need to check the other
735 # board name cadidates. Let's break the loop.
736 break
737 except error.CmdError:
738 # It doesn't exist. Let's move on to the next item.
739 pass
740 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800741 logging.error('No LATEST release info is available.')
742 return None
743
Namyoon Woo406c7d42020-01-24 15:57:11 -0800744 for cand_dir in candidates:
745 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800746
Namyoon Woo406c7d42020-01-24 15:57:11 -0800747 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700748 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800749 try:
750 # Check if release_path does exist.
751 release = utils.system_output('gsutil ls -d ' + release_path)
752 # Now 'release' has a full directory path: e.g.
753 # gs://chromeos-image-archive/firmware-octopus-11297.B-
754 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
755
756 # Remove "gs://chromeos-image-archive".
757 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
758
759 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
760 return release.strip('/')
761 except error.CmdError:
762 # The directory might not exist. Let's try next candidate.
763 pass
764 else:
765 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800766
Brent Peterson1cb623a2020-01-09 13:14:28 -0800767 @staticmethod
768 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800769 """Get version string from binary image using regular expression.
770
771 @param image: Binary image to search
772 @param version_regex: Regular expression to search for
773
774 @return Version string
775
776 @raises TestFail if no version string is found in image
777 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800778 with open(image, 'rb') as f:
779 image_data = f.read()
Derek Beckett98345552020-08-31 16:07:22 -0700780 match = re.findall(version_regex,
781 image_data.decode('ISO-8859-1', errors='ignore'))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800782 if match:
783 return match[0]
784 else:
785 raise error.TestFail('Failed to read version from %s.' % image)
786
787
Garry Wangad2a1712020-03-26 15:06:43 -0700788 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800789 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700790 try_scp=False, install_ec=True, install_bios=True,
791 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700792 """Install firmware to the DUT.
793
794 Use stateful update if the DUT is already running the same build.
795 Stateful update does not update kernel and tends to run much faster
796 than a full reimage. If the DUT is running a different build, or it
797 failed to do a stateful update, full update, including kernel update,
798 will be applied to the DUT.
799
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800800 Once a host enters firmware_install its fw[ro|rw]_version label will
801 be removed. After the firmware is updated successfully, a new
802 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700803
804 @param build: The build version to which we want to provision the
805 firmware of the machine,
806 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800807 @param rw_only: True to only install firmware to its RW portions. Keep
808 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700809 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800810 @param local_tarball: Path to local firmware image for installing
811 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800812 @param verify_version: True to verify EC and BIOS versions after
813 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800814 @param try_scp: False to always program using servo, true to try copying
815 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700816 @param install_ec: True to install EC FW, and False to skip it.
817 @param install_bios: True to install BIOS, and False to skip it.
818 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700819
820 TODO(dshi): After bug 381718 is fixed, update here with corresponding
821 exceptions that could be raised.
822
823 """
824 if not self.servo:
825 raise error.TestError('Host %s does not have servo.' %
826 self.hostname)
827
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700828 # Get the DUT board name from AFE.
829 info = self.host_info_store.get()
830 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700831 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800832
833 if board is None or board == '':
834 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700835
Namyoon Woo382e5892020-05-20 16:48:40 -0700836 # if board_as argument is passed, then use it instead of the original
837 # board name.
838 if board_as:
839 board = board_as
840
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700841 if model is None or model == '':
Namyoon Woofb16eae2020-08-14 10:02:39 -0700842 try:
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700843 model = self.get_platform()
Namyoon Woofb16eae2020-08-14 10:02:39 -0700844 except Exception as e:
845 logging.warn('Dut is unresponsive: %s', str(e))
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700846
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800847 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700848 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800849 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700850 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800851
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700852 try:
853 ds = dev_server.ImageServer.resolve(build, self.hostname)
854 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800855
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700856 if not dest:
857 tmpd = autotemp.tempdir(unique_id='fwimage')
858 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800859
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700860 # Download firmware image
861 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
862 local_tarball = os.path.join(dest, os.path.basename(fwurl))
863 ds.download_file(fwurl, local_tarball)
864 except Exception as e:
865 raise error.TestError('Failed to download firmware package: %s'
866 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700867
Namyoon Woo382e5892020-05-20 16:48:40 -0700868 ec_image = None
869 if install_ec:
870 # Extract EC image from tarball
871 logging.info('Extracting EC image.')
872 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700873 logging.info('Extracted: %s', ec_image)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800874
Namyoon Woo382e5892020-05-20 16:48:40 -0700875 bios_image = None
876 if install_bios:
877 # Extract BIOS image from tarball
878 logging.info('Extracting BIOS image.')
879 bios_image = self.servo.extract_bios_image(board, model,
880 local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700881 logging.info('Extracted: %s', bios_image)
Namyoon Woo382e5892020-05-20 16:48:40 -0700882
883 if not bios_image and not ec_image:
884 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800885
Brent Petersonc70a1832020-01-24 15:54:35 -0800886 # Clear firmware version labels
887 self._clear_fw_version_labels(rw_only)
888
889 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800890 try:
Garry Wang50e4a492020-08-05 12:29:57 -0700891 # Check if copying to DUT is enabled and DUT is available
892 if try_scp and self.is_up():
Brent Petersonc70a1832020-01-24 15:54:35 -0800893 # DUT is available, make temp firmware directory to store images
894 logging.info('Making temp folder.')
895 dest_folder = '/tmp/firmware'
896 self.run('mkdir -p ' + dest_folder)
897
Namyoon Woo68b68082020-06-02 13:13:14 -0700898 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800899
Namyoon Woo382e5892020-05-20 16:48:40 -0700900 if bios_image:
901 # Send BIOS firmware image to DUT
902 logging.info('Sending BIOS firmware.')
903 dest_bios_path = os.path.join(dest_folder,
904 os.path.basename(bios_image))
905 self.send_file(bios_image, dest_bios_path)
906
907 # Initialize firmware update command for BIOS image
908 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800909
910 # Send EC firmware image to DUT when EC image was found
911 if ec_image:
912 logging.info('Sending EC firmware.')
913 dest_ec_path = os.path.join(dest_folder,
914 os.path.basename(ec_image))
915 self.send_file(ec_image, dest_ec_path)
916
917 # Add EC image to firmware update command
918 fw_cmd += ' -e %s' % dest_ec_path
919
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700920 # Make sure command is allowed to finish even if ssh fails.
921 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
922
Brent Peterson669edf42020-02-07 15:07:54 -0800923 # Update firmware on DUT
924 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700925 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700926 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700927 except error.AutoservRunError as e:
928 if e.result_obj.exit_status != 255:
929 raise
930 elif ec_image:
931 logging.warn("DUT network dropped during update"
932 " (often caused by EC resetting USB)")
933 else:
934 logging.error("DUT network dropped during update"
935 " (unexpected, since no EC image)")
936 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800937 else:
938 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800939 if ec_image:
940 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700941 if bios_image:
942 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800943 if utils.host_is_in_lab_zone(self.hostname):
944 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800945
946 # Reboot and wait for DUT after installing firmware
947 logging.info('Rebooting DUT.')
948 self.servo.get_power_state_controller().reset()
949 time.sleep(self.servo.BOOT_DELAY)
950 self.test_wait_for_boot()
951
952 # When enabled verify EC and BIOS firmware version after programming
953 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800954 # Check programmed EC firmware when EC image was found
955 if ec_image:
956 logging.info('Checking EC firmware version.')
957 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800958 ec_version_prefix = dest_ec_version.split('_', 1)[0]
959 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800960 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800961 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800962 if dest_ec_version != image_ec_version:
963 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700964 'Failed to update EC firmware, version %s '
965 '(expected %s)' % (dest_ec_version,
966 image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800967
Namyoon Woo382e5892020-05-20 16:48:40 -0700968 if bios_image:
969 # Check programmed BIOS firmware against expected version
970 logging.info('Checking BIOS firmware version.')
971 dest_bios_version = self.get_firmware_version()
972 bios_version_prefix = dest_bios_version.split('.', 1)[0]
973 bios_regex = self._BIOS_REGEX % bios_version_prefix
974 image_bios_version = self.get_version_from_image(bios_image,
975 bios_regex)
976 if dest_bios_version != image_bios_version:
977 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700978 'Failed to update BIOS, version %s '
Namyoon Woo382e5892020-05-20 16:48:40 -0700979 '(expected %s)' % (dest_bios_version,
980 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700981 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700982 if tmpd:
983 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700984
985
beepsf079cfb2013-09-18 17:49:51 -0700986 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
987 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500988 """
989 Re-install the OS on the DUT by:
990 1) installing a test image on a USB storage device attached to the Servo
991 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800992 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700993 3) installing the image with chromeos-install.
994
Scott Zawalski62bacae2013-03-05 10:40:32 -0500995 @param image_url: If specified use as the url to install on the DUT.
996 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700997 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
998 Factory images need a longer usb_boot_timeout than regular
999 cros images.
1000 @param install_timeout: The timeout to use when installing the chromeos
1001 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -08001002
Scott Zawalski62bacae2013-03-05 10:40:32 -05001003 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -07001004
J. Richard Barnette0199cc82014-12-05 17:08:40 -08001005 """
Garry Wang7b0e1b72020-03-25 19:08:59 -07001006 if image_url:
1007 logging.info('Downloading image to USB, then booting from it.'
1008 ' Usb boot timeout = %s', usb_boot_timeout)
1009 else:
1010 logging.info('Booting from USB directly. Usb boot timeout = %s',
1011 usb_boot_timeout)
1012
1013 metrics_field = {'download': bool(image_url)}
1014 metrics.Counter(
1015 'chromeos/autotest/provision/servo_install/download_image'
1016 ).increment(fields=metrics_field)
1017
Allen Li48a13fe2016-11-22 14:10:40 -08001018 with metrics.SecondsTimer(
1019 'chromeos/autotest/provision/servo_install/boot_duration'):
Garry Wang53fc8f32020-09-18 13:30:08 -07001020 need_snk = self.require_snk_mode_in_recovery()
1021 self.servo.install_recovery_image(image_url, snk_mode=need_snk)
Allen Li48a13fe2016-11-22 14:10:40 -08001022 if not self.wait_up(timeout=usb_boot_timeout):
Garry Wang53fc8f32020-09-18 13:30:08 -07001023 if need_snk:
1024 # Attempt to restore servo_v4 role to 'src' mode.
1025 self.servo.set_servo_v4_role('src')
Allen Li48a13fe2016-11-22 14:10:40 -08001026 raise hosts.AutoservRepairError(
1027 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001028 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001029
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001030 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1031 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001032 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001033 try:
1034 self.run('chromeos-tpm-recovery')
1035 except error.AutoservRunError:
1036 logging.warn('chromeos-tpm-recovery is too old.')
1037
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001038
Allen Li48a13fe2016-11-22 14:10:40 -08001039 with metrics.SecondsTimer(
1040 'chromeos/autotest/provision/servo_install/install_duration'):
1041 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001042 try:
1043 self.run('chromeos-install --yes',timeout=install_timeout)
1044 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001045 except Exception as e:
1046 storage_errors = [
1047 'No space left on device',
1048 'I/O error when trying to write primary GPT',
1049 'Input/output error while writing out',
1050 'cannot read GPT header',
Otabek Kasimov2b7e8302020-08-21 09:23:31 -07001051 'can not determine destination device',
1052 'wrong fs type',
1053 'bad superblock on',
Otabek Kasimov808cd832020-05-28 18:27:46 -07001054 ]
1055 has_error = [msg for msg in storage_errors if(msg in str(e))]
1056 if has_error:
1057 info = self.host_info_store.get()
1058 info.set_version_label(
1059 audit_const.DUT_STORAGE_STATE_PREFIX,
1060 audit_const.HW_STATE_NEED_REPLACEMENT)
1061 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001062 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001063 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001064 logging.debug(
1065 'Fail install image from USB; Storage error; %s', e)
1066 raise error.AutoservError(
1067 'Failed to install image from USB due to a suspect '
1068 'disk failure, DUT storage state changed to '
1069 'need_replacement, please check debug log '
1070 'for details.')
1071 else:
Otabek Kasimov27bb2862020-08-10 14:40:45 -07001072 # DUT will be marked for replacement if storage is bad.
1073 audit_verify.VerifyDutStorage(self).verify()
1074
Otabek Kasimov808cd832020-05-28 18:27:46 -07001075 logging.debug('Fail install image from USB; %s', e)
1076 raise error.AutoservError(
1077 'Failed to install image from USB due to unexpected '
1078 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001079 finally:
1080 # We need reset the DUT no matter re-install success or not,
1081 # as we don't want leave the DUT in boot from usb state.
1082 logging.info('Power cycling DUT through servo.')
1083 self.servo.get_power_state_controller().power_off()
1084 self.servo.switch_usbkey('off')
Garry Wang53fc8f32020-09-18 13:30:08 -07001085 if need_snk:
1086 # Attempt to restore servo_v4 role to 'src' mode.
1087 self.servo.set_servo_v4_role('src')
Garry Wang033a31e2020-04-10 17:20:49 -07001088 # N.B. The Servo API requires that we use power_on() here
1089 # for two reasons:
1090 # 1) After turning on a DUT in recovery mode, you must turn
1091 # it off and then on with power_on() once more to
1092 # disable recovery mode (this is a Parrot specific
1093 # requirement).
1094 # 2) After power_off(), the only way to turn on is with
1095 # power_on() (this is a Storm specific requirement).
1096 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001097
1098 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001099 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001100 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1101 'test image after %d seconds' %
1102 self.BOOT_TIMEOUT,
1103 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001104
1105
Garry Wanga2e78172020-09-09 23:49:07 -07001106 def set_servo_host(self, host, servo_state=None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001107 """Set our servo host member, and associated servo.
1108
1109 @param host Our new `ServoHost`.
1110 """
1111 self._servo_host = host
Derek Beckettb66e5c82020-08-12 15:31:02 -07001112 self.servo_pwr_supported = None
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001113 if self._servo_host is not None:
1114 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001115 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001116 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Derek Beckettb66e5c82020-08-12 15:31:02 -07001117 try:
1118 self.servo_pwr_supported = self.servo.has_control('power_state')
1119 except Exception as e:
1120 logging.debug(
1121 "Could not get servo power state due to {}".format(e))
Gregory Nisbet93b23e22020-10-02 20:42:16 +00001122 else:
1123 self.servo = None
Derek Beckettb66e5c82020-08-12 15:31:02 -07001124 self.servo_pwr_supported = False
Otabek Kasimov41301a22020-05-10 15:28:21 -07001125 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001126 self.set_servo_state(servo_state)
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07001127 self._set_servo_topology()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001128
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001129
Richard Barnette9a26ad62016-06-10 12:03:08 -07001130 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001131 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001132 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001133
Richard Barnette9a26ad62016-06-10 12:03:08 -07001134 If the servo object is missing, attempt to repair the servo
1135 host. Repair failures are passed back to the caller.
1136
1137 @raise AutoservError: If there is no servo host for this CrOS
1138 host.
1139 """
1140 if self.servo:
1141 return
1142 if not self._servo_host:
1143 raise error.AutoservError('No servo host for %s.' %
1144 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001145 try:
1146 self._servo_host.repair()
1147 except:
1148 raise
1149 finally:
1150 self.set_servo_host(self._servo_host)
1151
1152
Otabek Kasimov41301a22020-05-10 15:28:21 -07001153 def set_servo_type(self):
1154 """Set servo info labels to dut host_info"""
1155 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001156 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001157 return
1158 servo_type = self.servo.get_servo_type()
1159 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001160 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001161 ' by `dut-control servo_type`! Please file a bug'
1162 ' and inform infra team as we are not expected '
1163 ' to reach this point.')
1164 return
1165 host_info = self.host_info_store.get()
1166 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1167 old_type = host_info.get_label_value(prefix)
1168 if old_type == servo_type:
1169 # do not need update
1170 return
1171 host_info.set_version_label(prefix, servo_type)
1172 self.host_info_store.commit(host_info)
1173 logging.info('ServoHost: servo_type updated to %s '
1174 '(previous: %s)', servo_type, old_type)
1175
1176
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001177 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001178 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001179 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001180 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001181 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001182 old_state = host_info.get_label_value(servo_state_prefix)
1183 if old_state == servo_state:
1184 # do not need update
1185 return
1186 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001187 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001188 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1189 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001190
1191
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001192 def get_servo_state(self):
1193 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001194 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001195 return host_info.get_label_value(servo_state_prefix)
1196
Dana Goyette655af512020-09-03 10:48:23 -07001197 def get_servo_usb_state(self):
1198 """Get the label value indicating the health of the USB drive.
1199
1200 @return: The label value if defined, otherwise '' (empty string).
1201 @rtype: str
1202 """
1203 host_info = self.host_info_store.get()
1204 servo_usb_state_prefix = audit_const.SERVO_USB_STATE_PREFIX
1205 return host_info.get_label_value(servo_usb_state_prefix)
1206
1207 def is_servo_usb_usable(self):
1208 """Check if the servo USB storage device is usable for FAFT.
1209
1210 @return: False if the label indicates a state that will break FAFT.
1211 True if state is okay, or if state is not defined.
1212 @rtype: bool
1213 """
1214 usb_state = self.get_servo_usb_state()
1215 return usb_state in ('', audit_const.HW_STATE_ACCEPTABLE,
1216 audit_const.HW_STATE_NORMAL,
1217 audit_const.HW_STATE_UNKNOWN)
Otabek Kasimov41301a22020-05-10 15:28:21 -07001218
Garry Wang000c6c02020-05-11 21:27:23 -07001219 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1220 if smart_usbhub_detected is None:
1221 # skip the label update here as this indicate we wasn't able
1222 # to confirm usbhub type.
1223 return
1224 host_info = self.host_info_store.get()
1225 if (smart_usbhub_detected ==
1226 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1227 # skip label update if current label match the truth.
1228 return
1229 if smart_usbhub_detected:
1230 logging.info('Adding %s label to host %s',
1231 servo_constants.SMART_USBHUB_LABEL,
1232 self.hostname)
1233 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1234 else:
1235 logging.info('Removing %s label from host %s',
1236 servo_constants.SMART_USBHUB_LABEL,
1237 self.hostname)
1238 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1239 self.host_info_store.commit(host_info)
1240
1241
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001242 def repair(self):
1243 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001244
1245 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001246 not call back to the parent class, but instead relies on
1247 `self._repair_strategy` to coordinate the verification and
1248 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001249 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001250 message = 'Beginning repair for host %s board %s model %s'
1251 info = self.host_info_store.get()
1252 message %= (self.hostname, info.board, info.model)
1253 self.record('INFO', None, None, message)
Garry Wanga2e78172020-09-09 23:49:07 -07001254 profile_state = profile_constants.DUT_STATE_READY
Shijin Abraham78ce4402020-09-08 22:04:27 -07001255 # Initialize bluetooth peers
1256 self.initialize_btpeer()
Garry Wang87af1d02020-05-26 17:55:54 -07001257 try:
1258 self._repair_strategy.repair(self)
1259 except hosts.AutoservVerifyDependencyError as e:
1260 # We don't want flag a DUT as failed if only non-critical
1261 # verifier(s) failed during the repair.
1262 if e.is_critical():
Garry Wanga2e78172020-09-09 23:49:07 -07001263 profile_state = profile_constants.DUT_STATE_REPAIR_FAILED
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001264 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001265 raise
Garry Wanga2e78172020-09-09 23:49:07 -07001266 finally:
1267 self.set_health_profile_dut_state(profile_state)
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001268
Otabek Kasimov8bb09912020-10-01 14:44:57 -07001269 def get_verifier_state(self, tag):
1270 """Return the state of servo verifier.
1271
1272 @returns: bool or None
1273 """
1274 return self._repair_strategy.verifier_is_good(tag)
Richard Barnette82c35912012-11-20 10:09:10 -08001275
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001276 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001277 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001278 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001279
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001280 if self._chameleon_host:
1281 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001282
Garry Wang1a493d82020-08-31 21:01:19 -07001283 if self.health_profile:
1284 try:
1285 self.health_profile.close()
1286 except Exception as e:
1287 logging.warning(
1288 'Failed to finalize device health profile; %s', e)
1289
xixuand6011f12016-12-08 15:01:58 -08001290 if self._servo_host:
1291 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001292
Dan Shi49ca0932014-11-14 11:22:27 -08001293 def get_power_supply_info(self):
1294 """Get the output of power_supply_info.
1295
1296 power_supply_info outputs the info of each power supply, e.g.,
1297 Device: Line Power
1298 online: no
1299 type: Mains
1300 voltage (V): 0
1301 current (A): 0
1302 Device: Battery
1303 state: Discharging
1304 percentage: 95.9276
1305 technology: Li-ion
1306
1307 Above output shows two devices, Line Power and Battery, with details of
1308 each device listed. This function parses the output into a dictionary,
1309 with key being the device name, and value being a dictionary of details
1310 of the device info.
1311
1312 @return: The dictionary of power_supply_info, e.g.,
1313 {'Line Power': {'online': 'yes', 'type': 'main'},
1314 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001315 @raise error.AutoservRunError if power_supply_info tool is not found in
1316 the DUT. Caller should handle this error to avoid false failure
1317 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001318 """
1319 result = self.run('power_supply_info').stdout.strip()
1320 info = {}
1321 device_name = None
1322 device_info = {}
1323 for line in result.split('\n'):
1324 pair = [v.strip() for v in line.split(':')]
1325 if len(pair) != 2:
1326 continue
1327 if pair[0] == 'Device':
1328 if device_name:
1329 info[device_name] = device_info
1330 device_name = pair[1]
1331 device_info = {}
1332 else:
1333 device_info[pair[0]] = pair[1]
1334 if device_name and not device_name in info:
1335 info[device_name] = device_info
1336 return info
1337
1338
1339 def get_battery_percentage(self):
1340 """Get the battery percentage.
1341
1342 @return: The percentage of battery level, value range from 0-100. Return
1343 None if the battery info cannot be retrieved.
1344 """
1345 try:
1346 info = self.get_power_supply_info()
1347 logging.info(info)
1348 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001349 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001350 return None
1351
1352
Philip Chenaf69ead2020-03-27 13:06:42 -07001353 def get_battery_state(self):
1354 """Get the battery charging state.
1355
1356 @return: A string representing the battery charging state. It can be
1357 'Charging', 'Fully charged', or 'Discharging'.
1358 """
1359 try:
1360 info = self.get_power_supply_info()
1361 logging.info(info)
1362 return info['Battery']['state']
1363 except (KeyError, ValueError, error.AutoservRunError):
1364 return None
1365
1366
Daniel Campello8ca25c22019-12-13 16:48:26 -07001367 def get_battery_display_percentage(self):
1368 """Get the battery display percentage.
1369
1370 @return: The display percentage of battery level, value range from
1371 0-100. Return None if the battery info cannot be retrieved.
1372 """
1373 try:
1374 info = self.get_power_supply_info()
1375 logging.info(info)
1376 return float(info['Battery']['display percentage'])
1377 except (KeyError, ValueError, error.AutoservRunError):
1378 return None
1379
1380
Dan Shi49ca0932014-11-14 11:22:27 -08001381 def is_ac_connected(self):
1382 """Check if the dut has power adapter connected and charging.
1383
1384 @return: True if power adapter is connected and charging.
1385 """
1386 try:
1387 info = self.get_power_supply_info()
1388 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001389 except (KeyError, error.AutoservRunError):
1390 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001391
1392
Simran Basi5e6339a2013-03-21 11:34:32 -07001393 def _cleanup_poweron(self):
1394 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001395 info = self.host_info_store.get()
1396 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001397 return
1398 logging.debug('This host has recently interacted with the RPM'
1399 ' Infrastructure. Ensuring power is on.')
1400 try:
1401 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001402 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001403 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001404 logging.error('Failed to turn Power On for this host after '
1405 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001406
1407 battery_percentage = self.get_battery_percentage()
Gregory Nisbet02273172020-07-13 09:26:17 -07001408 if (battery_percentage and
1409 battery_percentage < cros_repair.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001410 raise
1411 elif self.is_ac_connected():
1412 logging.info('The device has power adapter connected and '
1413 'charging. No need to try to turn RPM on '
1414 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001415 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001416 logging.info('Battery level is now at %s%%. The device may '
1417 'still have enough power to run test, so no '
1418 'exception will be raised.', battery_percentage)
1419
Simran Basi5e6339a2013-03-21 11:34:32 -07001420
Garry Wangad4d4fd2019-01-30 17:00:38 -08001421 def _remove_rpm_changed_tag(self):
1422 info = self.host_info_store.get()
1423 del info.attributes[self._RPM_OUTLET_CHANGED]
1424 self.host_info_store.commit(info)
1425
1426
1427 def _add_rpm_changed_tag(self):
1428 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001429 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001430 self.host_info_store.commit(info)
1431
1432
1433
beepsc87ff602013-07-31 21:53:00 -07001434 def _is_factory_image(self):
1435 """Checks if the image on the DUT is a factory image.
1436
1437 @return: True if the image on the DUT is a factory image.
1438 False otherwise.
1439 """
1440 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1441 return result.exit_status == 0
1442
1443
1444 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001445 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001446
1447 @raises: FactoryImageCheckerException for factory images, since
1448 we cannot attempt to restart ui on them.
1449 error.AutoservRunError for any other type of error that
1450 occurs while restarting ui.
1451 """
1452 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001453 raise FactoryImageCheckerException('Cannot restart ui on factory '
1454 'images')
beepsc87ff602013-07-31 21:53:00 -07001455
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001456 # TODO(jrbarnette): The command to stop/start the ui job
1457 # should live inside cros_ui, too. However that would seem
1458 # to imply interface changes to the existing start()/restart()
1459 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001460 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001461 self.run('stop ui; start ui')
1462 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001463
1464
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001465 def _start_powerd_if_needed(self):
1466 """Start powerd if it isn't already running."""
1467 self.run('start powerd', ignore_status=True)
1468
1469
xixuana3bbc422017-05-04 15:57:21 -07001470 def _get_lsb_release_content(self):
1471 """Return the content of lsb-release file of host."""
1472 return self.run(
1473 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1474
1475
Dan Shi549fb822015-03-24 18:01:11 -07001476 def get_release_version(self):
1477 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1478
1479 @returns The version string in lsb-release, under attribute
1480 CHROMEOS_RELEASE_VERSION.
1481 """
Dan Shi549fb822015-03-24 18:01:11 -07001482 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001483 lsb_release_content=self._get_lsb_release_content())
1484
1485
Don Garrettb9f35802018-01-22 18:25:40 -08001486 def get_release_builder_path(self):
1487 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1488
1489 @returns The version string in lsb-release, under attribute
1490 CHROMEOS_RELEASE_BUILDER_PATH.
1491 """
1492 return lsbrelease_utils.get_chromeos_release_builder_path(
1493 lsb_release_content=self._get_lsb_release_content())
1494
1495
xixuana3bbc422017-05-04 15:57:21 -07001496 def get_chromeos_release_milestone(self):
1497 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1498 from lsb-release.
1499
1500 @returns The version string in lsb-release, under attribute
1501 CHROMEOS_RELEASE_BUILD_TYPE.
1502 """
1503 return lsbrelease_utils.get_chromeos_release_milestone(
1504 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001505
1506
1507 def verify_cros_version_label(self):
Garry Wangd18e7b32020-08-07 18:31:44 -07001508 """Verify if host's cros-version label match the actual image in dut.
Dan Shi549fb822015-03-24 18:01:11 -07001509
Garry Wangd18e7b32020-08-07 18:31:44 -07001510 @returns True if the label match with image in dut, otherwise False
Dan Shi549fb822015-03-24 18:01:11 -07001511 """
Garry Wangd18e7b32020-08-07 18:31:44 -07001512 os_from_host = self.get_release_builder_path()
1513 info = self.host_info_store.get()
1514 os_from_label = info.get_label_value(self.VERSION_PREFIX)
1515 if not os_from_label:
1516 logging.debug('No existing %s label detected', self.VERSION_PREFIX)
1517 return True
1518
1519 # known cases where the version label will not match the
1520 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1521 # * Tests for the `arc-presubmit` append "-cheetsth" to the label.
1522 if os_from_label.endswith(provision.CHEETS_SUFFIX):
1523 logging.debug('%s label with %s suffix detected, this suffix will'
1524 ' be ignored when comparing label.',
1525 self.VERSION_PREFIX, provision.CHEETS_SUFFIX)
1526 os_from_label = os_from_label[:-len(provision.CHEETS_SUFFIX)]
1527 logging.debug('OS version from host: %s; OS verision cached in '
1528 'label: %s', os_from_host, os_from_label)
1529 return os_from_label == os_from_host
Dan Shi549fb822015-03-24 18:01:11 -07001530
1531
Laurence Goodby778c9a42017-05-24 19:24:07 -07001532 def cleanup_services(self):
1533 """Reinitializes the device for cleanup.
1534
1535 Subclasses may override this to customize the cleanup method.
1536
1537 To indicate failure of the reset, the implementation may raise
1538 any of:
1539 error.AutoservRunError
1540 error.AutotestRunError
1541 FactoryImageCheckerException
1542
1543 @raises error.AutoservRunError
1544 @raises error.AutotestRunError
1545 @raises error.FactoryImageCheckerException
1546 """
1547 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001548 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001549
1550
Gregory Nisbetbb6cc7c2020-09-23 15:55:04 -07001551 def cleanup(self, reboot_cmd=None):
1552 """Cleanup state on device.
1553
1554 @param reboot_cmd: command to use to reboot device
1555 @return nothing
1556 """
MK Ryu35d661e2014-09-25 17:44:10 -07001557 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001558 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001559 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001560 except (error.AutotestRunError, error.AutoservRunError,
1561 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001562 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001563
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001564 # cleanup routines, i.e. reboot the machine.
Gregory Nisbetbb6cc7c2020-09-23 15:55:04 -07001565 super(CrosHost, self).cleanup(reboot_cmd=reboot_cmd)
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001566
Simran Basi5e6339a2013-03-21 11:34:32 -07001567 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001568 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001569 self._cleanup_poweron()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001570
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001571 def reboot(self, **dargs):
1572 """
1573 This function reboots the site host. The more generic
1574 RemoteHost.reboot() performs sync and sleeps for 5
1575 seconds. This is not necessary for Chrome OS devices as the
1576 sync should be finished in a short time during the reboot
1577 command.
1578 """
Gregory Nisbetbb6cc7c2020-09-23 15:55:04 -07001579 if dargs.get('reboot_cmd') is None:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001580 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001581 dargs['reboot_cmd'] = ('sleep 1; '
1582 'reboot & sleep %d; '
1583 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001584 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001585 if 'fastsync' not in dargs:
1586 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001587
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001588 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001589 # Record who called us
1590 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001591 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001592 'dut_host_name' : self.hostname,
1593 'success' : True}
1594 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001595 'caller' : "%s:%s" % (orig.co_filename,
1596 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001597 'success' : True,
1598 'error' : ''}
1599
Vincent Palatin80780b22016-07-27 16:02:37 +02001600 t0 = time.time()
1601 try:
Gregory Nisbetbb6cc7c2020-09-23 15:55:04 -07001602 logging.info("reboot cmd: %s", dargs.get('reboot_cmd'))
Vincent Palatin80780b22016-07-27 16:02:37 +02001603 super(CrosHost, self).reboot(**dargs)
1604 except Exception as e:
1605 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001606 metric_debug_fields['success'] = False
1607 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001608 raise
1609 finally:
1610 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001611 metrics.Counter(
1612 'chromeos/autotest/autoserv/reboot_count').increment(
1613 fields=metric_fields)
1614 metrics.Counter(
1615 'chromeos/autotest/autoserv/reboot_debug').increment(
1616 fields=metric_debug_fields)
1617 metrics.SecondsDistribution(
1618 'chromeos/autotest/autoserv/reboot_duration').add(
1619 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001620
1621
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001622 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001623 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001624 """
1625 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001626
1627 @param suspend_time: How long to suspend as integer seconds.
1628 @param suspend_cmd: Suspend command to execute.
1629 @param allow_early_resume: If False and if device resumes before
1630 |suspend_time|, throw an error.
1631
1632 @exception AutoservSuspendError Host resumed earlier than
1633 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001634 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001635
1636 if suspend_cmd is None:
1637 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001638 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001639 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001640 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001641 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1642 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001643
1644
Simran Basiec564392014-08-25 16:48:09 -07001645 def upstart_status(self, service_name):
1646 """Check the status of an upstart init script.
1647
1648 @param service_name: Service to look up.
1649
1650 @returns True if the service is running, False otherwise.
1651 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001652 return 'start/running' in self.run('status %s' % service_name,
1653 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001654
Tom Hughese9552342018-12-18 14:29:25 -08001655 def upstart_stop(self, service_name):
1656 """Stops an upstart job if it's running.
1657
1658 @param service_name: Service to stop
1659
1660 @returns True if service has been stopped or was already stopped
1661 False otherwise.
1662 """
1663 if not self.upstart_status(service_name):
1664 return True
1665
1666 result = self.run('stop %s' % service_name, ignore_status=True)
1667 if result.exit_status != 0:
1668 return False
1669 return True
1670
1671 def upstart_restart(self, service_name):
1672 """Restarts (or starts) an upstart job.
1673
1674 @param service_name: Service to start/restart
1675
1676 @returns True if service has been started/restarted, False otherwise.
1677 """
1678 cmd = 'start'
1679 if self.upstart_status(service_name):
1680 cmd = 'restart'
1681 cmd = cmd + ' %s' % service_name
1682 result = self.run(cmd)
1683 if result.exit_status != 0:
1684 return False
1685 return True
Simran Basiec564392014-08-25 16:48:09 -07001686
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001687 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001688 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001689
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001690 Tests for the following conditions:
1691 1. All conditions tested by the parent version of this
1692 function.
1693 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001694 3. Sufficient space in /mnt/stateful_partition/encrypted.
1695 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001696
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001697 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001698 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001699 default_kilo_inodes_required = CONFIG.get_config_value(
1700 'SERVER', 'kilo_inodes_required', type=int, default=100)
1701 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1702 kilo_inodes_required = CONFIG.get_config_value(
1703 'SERVER', 'kilo_inodes_required_%s' % board,
1704 type=int, default=default_kilo_inodes_required)
1705 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001706 self.check_diskspace(
1707 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001708 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001709 'SERVER', 'gb_diskspace_required', type=float,
1710 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001711 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1712 # Not all targets build with encrypted stateful support.
1713 if self.path_exists(encrypted_stateful_path):
1714 self.check_diskspace(
1715 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001716 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001717 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1718 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001719
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001720 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001721
beepsc87ff602013-07-31 21:53:00 -07001722 # Factory images don't run update engine,
1723 # goofy controls dbus on these DUTs.
1724 if not self._is_factory_image():
1725 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001726
1727
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001728 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1729 def wait_for_system_services(self):
1730 """Waits for system-services to be running.
1731
1732 Sometimes, update_engine will take a while to update firmware, so we
1733 should give this some time to finish. See crbug.com/765686#c38 for
1734 details.
1735 """
1736 if not self.upstart_status('system-services'):
1737 raise error.AutoservError('Chrome failed to reach login. '
1738 'System services not running.')
1739
1740
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001741 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001742 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001743 message = 'Beginning verify for host %s board %s model %s'
1744 info = self.host_info_store.get()
1745 message %= (self.hostname, info.board, info.model)
1746 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001747 try:
1748 self._repair_strategy.verify(self)
1749 except hosts.AutoservVerifyDependencyError as e:
1750 # We don't want flag a DUT as failed if only non-critical
1751 # verifier(s) failed during the repair.
1752 if e.is_critical():
1753 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001754
1755
Fang Deng96667ca2013-08-01 17:46:18 -07001756 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001757 connect_timeout=None, alive_interval=None,
1758 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001759 """Override default make_ssh_command to use options tuned for Chrome OS.
1760
1761 Tuning changes:
1762 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1763 connection failure. Consistency with remote_access.sh.
1764
Samuel Tan2ce155b2015-06-23 18:24:38 -07001765 - ServerAliveInterval=900; which causes SSH to ping connection every
1766 900 seconds. In conjunction with ServerAliveCountMax ensures
1767 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001768 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001769 the test completed successfully. Later increased from 180 seconds to
1770 900 seconds to account for tests where the DUT is suspended for
1771 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001772
1773 - ServerAliveCountMax=3; consistency with remote_access.sh.
1774
1775 - ConnectAttempts=4; reduce flakiness in connection errors;
1776 consistency with remote_access.sh.
1777
1778 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1779 Host keys change with every new installation, don't waste
1780 memory/space saving them.
1781
1782 - SSH protocol forced to 2; needed for ServerAliveInterval.
1783
1784 @param user User name to use for the ssh connection.
1785 @param port Port on the target host to use for ssh connection.
1786 @param opts Additional options to the ssh command.
1787 @param hosts_file Ignored.
1788 @param connect_timeout Ignored.
1789 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001790 @param alive_count_max Ignored.
1791 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001792 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001793 options = ' '.join([opts, '-o Protocol=2'])
1794 return super(CrosHost, self).make_ssh_command(
1795 user=user, port=port, opts=options, hosts_file='/dev/null',
1796 connect_timeout=30, alive_interval=900, alive_count_max=3,
1797 connection_attempts=4)
1798
1799
Jason Abeleb6f924f2013-11-13 16:01:54 -08001800 def syslog(self, message, tag='autotest'):
1801 """Logs a message to syslog on host.
1802
1803 @param message String message to log into syslog
1804 @param tag String tag prefix for syslog
1805
1806 """
1807 self.run('logger -t "%s" "%s"' % (tag, message))
1808
1809
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001810 def _ping_check_status(self, status):
1811 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001812
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001813 @param status Check the ping status against this value.
1814 @return True iff `status` and the result of ping are the same
1815 (i.e. both True or both False).
1816
1817 """
Abhishek Pandit-Subedi038df162020-09-14 16:37:43 -07001818 ping_val = utils.ping(self.hostname,
1819 tries=1,
1820 deadline=1,
1821 timeout=2,
1822 ignore_timeout=True)
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001823 return not (status ^ (ping_val == 0))
1824
1825 def _ping_wait_for_status(self, status, timeout):
1826 """Wait for the host to have a given status (UP or DOWN).
1827
1828 Status is checked by polling. Polling will not last longer
1829 than the number of seconds in `timeout`. The polling
1830 interval will be long enough that only approximately
1831 _PING_WAIT_COUNT polling cycles will be executed, subject
1832 to a maximum interval of about one minute.
1833
1834 @param status Waiting will stop immediately if `ping` of the
1835 host returns this status.
1836 @param timeout Poll for at most this many seconds.
1837 @return True iff the host status from `ping` matched the
1838 requested status at the time of return.
1839
1840 """
1841 # _ping_check_status() takes about 1 second, hence the
1842 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001843 # FIXME: if the ping command errors then _ping_check_status()
1844 # returns instantly. If timeout is also smaller than twice
1845 # _PING_WAIT_COUNT then the while loop below forks many
1846 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1847 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1848 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001849 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1850 end_time = time.time() + timeout
1851 while time.time() <= end_time:
1852 if self._ping_check_status(status):
1853 return True
1854 if poll_interval > 0:
1855 time.sleep(poll_interval)
1856
1857 # The last thing we did was sleep(poll_interval), so it may
1858 # have been too long since the last `ping`. Check one more
1859 # time, just to be sure.
1860 return self._ping_check_status(status)
1861
1862 def ping_wait_up(self, timeout):
1863 """Wait for the host to respond to `ping`.
1864
1865 N.B. This method is not a reliable substitute for
1866 `wait_up()`, because a host that responds to ping will not
1867 necessarily respond to ssh. This method should only be used
1868 if the target DUT can be considered functional even if it
1869 can't be reached via ssh.
1870
1871 @param timeout Minimum time to allow before declaring the
1872 host to be non-responsive.
1873 @return True iff the host answered to ping before the timeout.
1874
1875 """
1876 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001877
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001878 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001879 """Wait until the host no longer responds to `ping`.
1880
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001881 This function can be used as a slightly faster version of
1882 `wait_down()`, by avoiding potentially long ssh timeouts.
1883
1884 @param timeout Minimum time to allow for the host to become
1885 non-responsive.
1886 @return True iff the host quit answering ping before the
1887 timeout.
1888
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001889 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001890 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001891
Anand K Mistry50f218e2020-07-31 14:50:15 +10001892 def _is_host_port_forwarded(self):
Garry Wanga2e78172020-09-09 23:49:07 -07001893 """Checks if the dut is connected over port forwarding.
Anand K Mistry50f218e2020-07-31 14:50:15 +10001894
1895 N.B. This method does not detect all situations where port forwarding is
1896 occurring. Namely, running autotest on the dut may result in a
1897 false-positive, and port forwarding using a different machine on the
1898 same network will be a false-negative.
1899
1900 @return True if the dut is connected over port forwarding
1901 False otherwise
1902 """
Garry Wanga2e78172020-09-09 23:49:07 -07001903 is_localhost = self.hostname in ['localhost', '127.0.0.1']
1904 is_forwarded = is_localhost and not self.is_default_port
1905 if is_forwarded:
1906 logging.info('Detected DUT connected by port forwarding')
1907 return is_forwarded
Anand K Mistry50f218e2020-07-31 14:50:15 +10001908
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001909 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001910 """Wait for the client to enter low-power sleep mode.
1911
1912 The test for "is asleep" can't distinguish a system that is
1913 powered off; to confirm that the unit was asleep, it is
1914 necessary to force resume, and then call
1915 `test_wait_for_resume()`.
1916
1917 This function is expected to be called from a test as part
1918 of a sequence like the following:
1919
1920 ~~~~~~~~
1921 boot_id = host.get_boot_id()
1922 # trigger sleep on the host
1923 host.test_wait_for_sleep()
1924 # trigger resume on the host
1925 host.test_wait_for_resume(boot_id)
1926 ~~~~~~~~
1927
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001928 @param sleep_timeout time limit in seconds to allow the host sleep.
1929
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001930 @exception TestFail The host did not go to sleep within
1931 the allowed time.
1932 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001933 if sleep_timeout is None:
1934 sleep_timeout = self.SLEEP_TIMEOUT
1935
Anand K Mistry50f218e2020-07-31 14:50:15 +10001936 # If the dut is accessed over SSH port-forwarding, `ping` is not useful
1937 # for detecting the dut is down since a ping to localhost will always
1938 # succeed. In this case, fall back to wait_down() which uses SSH.
1939 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07001940 success = self.wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10001941 else:
Garry Wanga2e78172020-09-09 23:49:07 -07001942 success = self.ping_wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10001943
1944 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001945 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001946 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001947
1948
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001949 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001950 """Wait for the client to resume from low-power sleep mode.
1951
1952 The `old_boot_id` parameter should be the value from
1953 `get_boot_id()` obtained prior to entering sleep mode. A
1954 `TestFail` exception is raised if the boot id changes.
1955
1956 See @ref test_wait_for_sleep for more on this function's
1957 usage.
1958
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001959 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001960 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001961 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001962
1963 @exception TestFail The host did not respond within the
1964 allowed time.
1965 @exception TestFail The host responded, but the boot id test
1966 indicated a reboot rather than a sleep
1967 cycle.
1968 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001969 if resume_timeout is None:
1970 resume_timeout = self.RESUME_TIMEOUT
1971
1972 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001973 raise error.TestFail(
1974 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001975 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001976 else:
1977 new_boot_id = self.get_boot_id()
1978 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001979 logging.error('client rebooted (old boot %s, new boot %s)',
1980 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001981 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001982 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001983
1984
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001985 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001986 """Wait for the client to shut down.
1987
1988 The test for "has shut down" can't distinguish a system that
1989 is merely asleep; to confirm that the unit was down, it is
1990 necessary to force boot, and then call test_wait_for_boot().
1991
1992 This function is expected to be called from a test as part
1993 of a sequence like the following:
1994
1995 ~~~~~~~~
1996 boot_id = host.get_boot_id()
1997 # trigger shutdown on the host
1998 host.test_wait_for_shutdown()
1999 # trigger boot on the host
2000 host.test_wait_for_boot(boot_id)
2001 ~~~~~~~~
2002
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002003 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002004 @exception TestFail The host did not shut down within the
2005 allowed time.
2006 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002007 if shutdown_timeout is None:
2008 shutdown_timeout = self.SHUTDOWN_TIMEOUT
2009
Anand K Mistry50f218e2020-07-31 14:50:15 +10002010 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07002011 success = self.wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002012 else:
Garry Wanga2e78172020-09-09 23:49:07 -07002013 success = self.ping_wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002014
2015 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002016 raise error.TestFail(
2017 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002018 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002019
2020
2021 def test_wait_for_boot(self, old_boot_id=None):
2022 """Wait for the client to boot from cold power.
2023
2024 The `old_boot_id` parameter should be the value from
2025 `get_boot_id()` obtained prior to shutting down. A
2026 `TestFail` exception is raised if the boot id does not
2027 change. The boot id test is omitted if `old_boot_id` is not
2028 specified.
2029
2030 See @ref test_wait_for_shutdown for more on this function's
2031 usage.
2032
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002033 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002034 shut down.
2035
2036 @exception TestFail The host did not respond within the
2037 allowed time.
2038 @exception TestFail The host responded, but the boot id test
2039 indicated that there was no reboot.
2040 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002041 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002042 raise error.TestFail(
2043 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002044 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002045 elif old_boot_id:
2046 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002047 logging.error('client not rebooted (boot %s)',
2048 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002049 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002050 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07002051
2052
2053 @staticmethod
2054 def check_for_rpm_support(hostname):
2055 """For a given hostname, return whether or not it is powered by an RPM.
2056
Simran Basi1df55112013-09-06 11:25:09 -07002057 @param hostname: hostname to check for rpm support.
2058
Simran Basid5e5e272012-09-24 15:23:59 -07002059 @return None if this host does not follows the defined naming format
2060 for RPM powered DUT's in the lab. If it does follow the format,
2061 it returns a regular expression MatchObject instead.
2062 """
Fang Dengbaff9082015-01-06 13:46:15 -08002063 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002064
2065
2066 def has_power(self):
2067 """For this host, return whether or not it is powered by an RPM.
2068
2069 @return True if this host is in the CROS lab and follows the defined
2070 naming format.
2071 """
Fang Deng0ca40e22013-08-27 17:47:44 -07002072 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002073
2074
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002075 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07002076 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002077
2078 @param state Specifies which power state to set to DUT
2079 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002080 use. By default "RPM" or "CCD" will be used based
2081 on servo type. Valid values from
2082 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002083
2084 """
2085 ACCEPTABLE_STATES = ['ON', 'OFF']
2086
Garry Wang5e5538a2019-04-08 15:36:18 -07002087 if not power_method:
2088 power_method = self.get_default_power_method()
2089
2090 state = state.upper()
2091 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002092 raise error.TestError('State must be one of: %s.'
2093 % (ACCEPTABLE_STATES,))
2094
2095 if power_method == self.POWER_CONTROL_SERVO:
2096 logging.info('Setting servo port J10 to %s', state)
2097 self.servo.set('prtctl3_pwren', state.lower())
2098 time.sleep(self._USB_POWER_TIMEOUT)
2099 elif power_method == self.POWER_CONTROL_MANUAL:
2100 logging.info('You have %d seconds to set the AC power to %s.',
2101 self._POWER_CYCLE_TIMEOUT, state)
2102 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07002103 elif power_method == self.POWER_CONTROL_CCD:
2104 servo_role = 'src' if state == 'ON' else 'snk'
2105 logging.info('servo ccd power pass through detected,'
2106 ' changing servo_role to %s.', servo_role)
2107 self.servo.set_servo_v4_role(servo_role)
2108 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07002109 # Make sure we don't leave DUT with no power(servo_role=snk)
2110 # when DUT is not pingable, as we raise a exception here
2111 # that may break a power cycle in the middle.
2112 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07002113 raise error.AutoservError(
2114 'DUT failed to regain network connection after %d seconds.'
2115 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002116 else:
2117 if not self.has_power():
2118 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002119 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07002120 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07002121
2122
Garry Wang5e5538a2019-04-08 15:36:18 -07002123 def power_off(self, power_method=None):
2124 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002125
2126 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002127 use. By default "RPM" or "CCD" will be used based
2128 on servo type. Valid values from
2129 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002130
2131 """
Derek Beckettb66e5c82020-08-12 15:31:02 -07002132 self._sync_if_up()
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002133 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002134
Derek Beckettb66e5c82020-08-12 15:31:02 -07002135 def _check_supported(self):
2136 """Throw an error if dts mode control is not supported."""
2137 if not self.servo_pwr_supported:
2138 raise error.TestFail('power_state controls not supported')
2139
2140 def _sync_if_up(self):
2141 """Run sync on the DUT and wait for completion if the DUT is up.
2142
2143 Additionally, try to sync and ignore status if its not up.
2144
2145 Useful prior to reboots to ensure files are written to disc.
2146
2147 """
2148 if self.is_up_fast():
2149 self.run("sync")
2150 return
2151 # If it is not up, attempt to sync in the rare event the DUT is up but
2152 # doesn't respond to a ping. Ignore any errors.
2153 try:
2154 self.run("sync", ignore_status=True, timeout=1)
2155 except Exception:
2156 pass
2157
2158 def power_off_via_servo(self):
2159 """Force the DUT to power off.
2160
2161 The DUT is guaranteed to be off at the end of this call,
2162 regardless of its previous state, provided that there is
2163 working EC and boot firmware. There is no requirement for
2164 working OS software.
2165
2166 """
2167 self._check_supported()
2168 self._sync_if_up()
2169 self.servo.set_nocheck('power_state', 'off')
2170
2171 def power_on_via_servo(self, rec_mode='on'):
2172 """Force the DUT to power on.
2173
2174 Prior to calling this function, the DUT must be powered off,
2175 e.g. with a call to `power_off()`.
2176
2177 At power on, recovery mode is set as specified by the
2178 corresponding argument. When booting with recovery mode on, it
2179 is the caller's responsibility to unplug/plug in a bootable
2180 external storage device.
2181
2182 If the DUT requires a delay after powering on but before
2183 processing inputs such as USB stick insertion, the delay is
2184 handled by this method; the caller is not responsible for such
2185 delays.
2186
2187 @param rec_mode Setting of recovery mode to be applied at
2188 power on. default: REC_OFF aka 'off'
2189
2190 """
2191 self._check_supported()
2192 self.servo.set_nocheck('power_state', rec_mode)
2193
2194 def reset_via_servo(self):
2195 """Force the DUT to reset.
2196
2197 The DUT is guaranteed to be on at the end of this call,
2198 regardless of its previous state, provided that there is
2199 working OS software. This also guarantees that the EC has
2200 been restarted.
2201
2202 """
2203 self._check_supported()
2204 self._sync_if_up()
2205 self.servo.set_nocheck('power_state', 'reset')
2206
Simran Basid5e5e272012-09-24 15:23:59 -07002207
Garry Wang5e5538a2019-04-08 15:36:18 -07002208 def power_on(self, power_method=None):
2209 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002210
2211 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002212 use. By default "RPM" or "CCD" will be used based
2213 on servo type. Valid values from
2214 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002215
2216 """
2217 self._set_power('ON', power_method)
2218
2219
Garry Wang5e5538a2019-04-08 15:36:18 -07002220 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002221 """Cycle power to this host by turning it OFF, then ON.
2222
2223 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002224 use. By default "RPM" or "CCD" will be used based
2225 on servo type. Valid values from
2226 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002227
2228 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002229 if not power_method:
2230 power_method = self.get_default_power_method()
2231
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002232 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002233 self.POWER_CONTROL_MANUAL,
2234 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002235 self.power_off(power_method=power_method)
2236 time.sleep(self._POWER_CYCLE_TIMEOUT)
2237 self.power_on(power_method=power_method)
2238 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002239 self._add_rpm_changed_tag()
2240 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002241
2242
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002243 def get_platform_from_fwid(self):
2244 """Determine the platform from the crossystem fwid.
2245
2246 @returns a string representing this host's platform.
2247 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002248 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002249 crossystem = utils.Crossystem(self)
2250 crossystem.init()
2251 # Extract fwid value and use the leading part as the platform id.
2252 # fwid generally follow the format of {platform}.{firmware version}
2253 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2254 platform = crossystem.fwid().split('.')[0].lower()
2255 # Newer platforms start with 'Google_' while the older ones do not.
2256 return platform.replace('google_', '')
2257
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002258
Simran Basic6f1f7a2012-10-16 10:47:46 -07002259 def get_platform(self):
2260 """Determine the correct platform label for this host.
2261
2262 @returns a string representing this host's platform.
2263 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002264 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2265 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002266 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002267 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002268 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002269 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002270
2271
Greg Edelstona7b05d12020-04-01 16:00:51 -06002272 def get_model_from_cros_config(self):
2273 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002274
Greg Edelstona7b05d12020-04-01 16:00:51 -06002275 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002276 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002277 return cros_config.call_cros_config_get_output('/ name',
2278 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002279
2280
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002281 def get_architecture(self):
2282 """Determine the correct architecture label for this host.
2283
2284 @returns a string representing this host's architecture.
2285 """
2286 crossystem = utils.Crossystem(self)
2287 crossystem.init()
2288 return crossystem.arch()
2289
2290
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002291 def get_chrome_version(self):
2292 """Gets the Chrome version number and milestone as strings.
2293
2294 Invokes "chrome --version" to get the version number and milestone.
2295
2296 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2297 current Chrome version number as a string (in the form "W.X.Y.Z")
2298 and "milestone" is the first component of the version number
2299 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2300 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2301 of "chrome --version" and the milestone will be the empty string.
2302
2303 """
MK Ryu35d661e2014-09-25 17:44:10 -07002304 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002305 return utils.parse_chrome_version(version_string)
2306
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002307
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002308 def get_ec_version(self):
2309 """Get the ec version as strings.
2310
2311 @returns a string representing this host's ec version.
2312 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002313 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002314 result = self.run(command, ignore_status=True)
2315 if result.exit_status != 0:
2316 return ''
2317 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002318
2319
2320 def get_firmware_version(self):
2321 """Get the firmware version as strings.
2322
2323 @returns a string representing this host's firmware version.
2324 """
2325 crossystem = utils.Crossystem(self)
2326 crossystem.init()
2327 return crossystem.fwid()
2328
2329
2330 def get_hardware_revision(self):
2331 """Get the hardware revision as strings.
2332
2333 @returns a string representing this host's hardware revision.
2334 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002335 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002336 result = self.run(command, ignore_status=True)
2337 if result.exit_status != 0:
2338 return ''
2339 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002340
2341
2342 def get_kernel_version(self):
2343 """Get the kernel version as strings.
2344
2345 @returns a string representing this host's kernel version.
2346 """
2347 return self.run('uname -r').stdout.strip()
2348
2349
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002350 def get_cpu_name(self):
2351 """Get the cpu name as strings.
2352
2353 @returns a string representing this host's cpu name.
2354 """
2355
2356 # Try get cpu name from device tree first
2357 if self.path_exists('/proc/device-tree/compatible'):
2358 command = ' | '.join(
2359 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2360 'tail -1'])
2361 return self.run(command).stdout.strip().replace(',', ' ')
2362
2363 # Get cpu name from uname -p
2364 command = 'uname -p'
2365 ret = self.run(command).stdout.strip()
2366
2367 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2368 # Try get cpu name from /proc/cpuinfo instead
2369 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2370 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2371 self = self.run(command).stdout.strip()
2372
2373 # Remove bloat from CPU name, for example
2374 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2375 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2376 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2377 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2378 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2379 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2380
2381
2382 def get_screen_resolution(self):
2383 """Get the screen(s) resolution as strings.
2384 In case of more than 1 monitor, return resolution for each monitor
2385 separate with plus sign.
2386
2387 @returns a string representing this host's screen(s) resolution.
2388 """
2389 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2390 ret = self.run(command, ignore_status=True)
2391 # We might have Chromebox without a screen
2392 if ret.exit_status != 0:
2393 return ''
2394 return ret.stdout.strip().replace('\n', '+')
2395
2396
2397 def get_mem_total_gb(self):
2398 """Get total memory available in the system in GiB (2^20).
2399
2400 @returns an integer representing total memory
2401 """
2402 mem_total_kb = self.read_from_meminfo('MemTotal')
2403 kb_in_gb = float(2 ** 20)
2404 return int(round(mem_total_kb / kb_in_gb))
2405
2406
2407 def get_disk_size_gb(self):
2408 """Get size of disk in GB (10^9)
2409
2410 @returns an integer representing size of disk, 0 in Error Case
2411 """
2412 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2413 result = self.run(command, ignore_status=True)
2414 if result.exit_status != 0:
2415 return 0
2416 _, _, block, _ = re.split(r' +', result.stdout.strip())
2417 byte_per_block = 1024.0
2418 disk_kb_in_gb = 1e9
2419 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2420
2421
2422 def get_battery_size(self):
2423 """Get size of battery in Watt-hour via sysfs
2424
2425 This method assumes that battery support voltage_min_design and
2426 charge_full_design sysfs.
2427
2428 @returns a float representing Battery size, 0 if error.
2429 """
2430 # sysfs report data in micro scale
2431 battery_scale = 1e6
2432
2433 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2434 result = self.run(command, ignore_status=True)
2435 if result.exit_status != 0:
2436 return 0
2437 voltage = float(result.stdout.strip()) / battery_scale
2438
2439 command = 'cat /sys/class/power_supply/*/charge_full_design'
2440 result = self.run(command, ignore_status=True)
2441 if result.exit_status != 0:
2442 return 0
2443 amphereHour = float(result.stdout.strip()) / battery_scale
2444
2445 return voltage * amphereHour
2446
2447
2448 def get_low_battery_shutdown_percent(self):
2449 """Get the percent-based low-battery shutdown threshold.
2450
2451 @returns a float representing low-battery shutdown percent, 0 if error.
2452 """
2453 ret = 0.0
2454 try:
2455 command = 'check_powerd_config --low_battery_shutdown_percent'
2456 ret = float(self.run(command).stdout)
2457 except error.CmdError:
2458 logging.debug("Can't run %s", command)
2459 except ValueError:
2460 logging.debug("Didn't get number from %s", command)
2461
2462 return ret
2463
2464
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002465 def has_hammer(self):
2466 """Check whether DUT has hammer device or not.
2467
2468 @returns boolean whether device has hammer or not
2469 """
2470 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2471 return self.run(command, ignore_status=True).exit_status == 0
2472
2473
Niranjan Kumar34618872017-05-31 12:57:09 -07002474 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002475 """Returns True if the specified switch was provided to Chrome.
2476
2477 @param switch The chrome switch to search for.
2478 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002479
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002480 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2481 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002482
2483
2484 def oobe_triggers_update(self):
2485 """Returns True if this host has an OOBE flow during which
2486 it will perform an update check and perhaps an update.
2487 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2488 As more such flows are developed, code handling them needs
2489 to be added here.
2490
2491 @return Boolean indicating whether this host's OOBE triggers an update.
2492 """
2493 return self.is_chrome_switch_present(
2494 '--enterprise-enable-zero-touch-enrollment=hands-off')
2495
2496
Kevin Chenga2619dc2016-03-28 11:42:08 -07002497 # TODO(kevcheng): change this to just return the board without the
2498 # 'board:' prefix and fix up all the callers. Also look into removing the
2499 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002500 def get_board(self):
2501 """Determine the correct board label for this host.
2502
2503 @returns a string representing this host's board.
2504 """
2505 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2506 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002507 return (ds_constants.BOARD_PREFIX +
2508 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002509
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002510 def get_channel(self):
2511 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002512
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002513 @returns: a string represeting this host's build channel.
2514 (stable, dev, beta). None on fail.
2515 """
2516 return lsbrelease_utils.get_chromeos_channel(
2517 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002518
Kevin Chenga328da62016-03-31 10:49:04 -07002519 def get_power_supply(self):
2520 """
2521 Determine what type of power supply the host has
2522
2523 @returns a string representing this host's power supply.
2524 'power:battery' when the device has a battery intended for
2525 extended use
2526 'power:AC_primary' when the device has a battery not intended
2527 for extended use (for moving the machine, etc)
2528 'power:AC_only' when the device has no battery at all.
2529 """
2530 psu = self.run(command='mosys psu type', ignore_status=True)
2531 if psu.exit_status:
2532 # The psu command for mosys is not included for all platforms. The
2533 # assumption is that the device will have a battery if the command
2534 # is not found.
2535 return 'power:battery'
2536
2537 psu_str = psu.stdout.strip()
2538 if psu_str == 'unknown':
2539 return None
2540
2541 return 'power:%s' % psu_str
2542
2543
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002544 def has_battery(self):
2545 """Determine if DUT has a battery.
2546
2547 Returns:
2548 Boolean, False if known not to have battery, True otherwise.
2549 """
2550 rv = True
2551 power_supply = self.get_power_supply()
2552 if power_supply == 'power:battery':
2553 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2554 board_type = self.get_board_type()
2555 if board_type in _NO_BATTERY_BOARD_TYPE:
2556 logging.warn('Do NOT believe type %s has battery. '
2557 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002558 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002559 ignore_status=True)
2560 logging.debug(psu)
2561 rv = False
2562 elif power_supply == 'power:AC_only':
2563 rv = False
2564
2565 return rv
2566
2567
Kevin Chenga328da62016-03-31 10:49:04 -07002568 def get_servo(self):
2569 """Determine if the host has a servo attached.
2570
2571 If the host has a working servo attached, it should have a servo label.
2572
2573 @return: string 'servo' if the host has servo attached. Otherwise,
2574 returns None.
2575 """
2576 return 'servo' if self._servo_host else None
2577
2578
Kevin Chenga328da62016-03-31 10:49:04 -07002579 def has_internal_display(self):
2580 """Determine if the device under test is equipped with an internal
2581 display.
2582
2583 @return: 'internal_display' if one is present; None otherwise.
2584 """
2585 from autotest_lib.client.cros.graphics import graphics_utils
2586 from autotest_lib.client.common_lib import utils as common_utils
2587
2588 def __system_output(cmd):
2589 return self.run(cmd).stdout
2590
2591 def __read_file(remote_path):
2592 return self.run('cat %s' % remote_path).stdout
2593
2594 # Hijack the necessary client functions so that we can take advantage
2595 # of the client lib here.
2596 # FIXME: find a less hacky way than this
2597 original_system_output = utils.system_output
2598 original_read_file = common_utils.read_file
2599 utils.system_output = __system_output
2600 common_utils.read_file = __read_file
2601 try:
2602 return ('internal_display' if graphics_utils.has_internal_display()
2603 else None)
2604 finally:
2605 utils.system_output = original_system_output
2606 common_utils.read_file = original_read_file
2607
2608
Dan Shi85276d42014-04-08 22:11:45 -07002609 def is_boot_from_usb(self):
2610 """Check if DUT is boot from USB.
2611
2612 @return: True if DUT is boot from usb.
2613 """
2614 device = self.run('rootdev -s -d').stdout.strip()
2615 removable = int(self.run('cat /sys/block/%s/removable' %
2616 os.path.basename(device)).stdout.strip())
2617 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002618
Otabek Kasimov77a40332020-10-20 15:40:03 -07002619 def is_boot_from_external_device(self):
2620 """Check if DUT is boot from external storage.
2621
2622 @return: True if DUT is boot from external storage.
2623 """
2624 boot_device = self.run('rootdev -s -d', ignore_status=True,
2625 timeout=60).stdout.strip()
2626 if not boot_device:
2627 logging.debug('Boot storage not detected on the host.')
2628 return False
2629 main_storage_cmd = ('. /usr/sbin/write_gpt.sh;'
2630 ' . /usr/share/misc/chromeos-common.sh;'
2631 ' load_base_vars; get_fixed_dst_drive')
2632 main_storage = self.run(main_storage_cmd,
2633 ignore_status=True,
2634 timeout=60).stdout.strip()
2635 if not main_storage:
2636 logging.debug('Main storage not detected on the host.')
2637 return False
2638 if boot_device == main_storage:
2639 logging.debug('Device booted from main storage.')
2640 return False
2641 logging.debug('Device booted from external storage storage.')
2642 return True
Helen Zhang17dae2b2014-11-11 09:25:52 -08002643
2644 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002645 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002646
2647 @param key: meminfo requested
2648
2649 @return the memory value as a string
2650
2651 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002652 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2653 logging.debug('%s', meminfo)
2654 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002655
2656
Rohit Makasana98e696f2016-06-03 18:48:10 -07002657 def get_cpu_arch(self):
2658 """Returns CPU arch of the device.
2659
2660 @return CPU architecture of the DUT.
2661 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002662 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002663 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2664 ignore_status=True).stdout:
2665 return 'x86_64'
2666 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2667 ignore_status=True).stdout:
2668 return 'arm'
2669 return 'i386'
2670
2671
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002672 def get_board_type(self):
2673 """
2674 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002675 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2676
2677 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002678 """
Danny Chan471a8d12015-08-18 14:57:41 -07002679 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2680 ignore_status=True).stdout
2681 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002682 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002683 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002684
2685
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002686 def get_arc_version(self):
2687 """Return ARC version installed on the DUT.
2688
2689 @returns ARC version as string if the CrOS build has ARC, else None.
2690 """
2691 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2692 ignore_status=True).stdout
2693 if arc_version:
2694 return arc_version.split('=')[-1].strip()
2695 return None
2696
2697
Gilad Arnolda76bef02015-09-29 13:55:15 -07002698 def get_os_type(self):
2699 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002700
2701
Kevin Chenga2619dc2016-03-28 11:42:08 -07002702 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002703 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002704 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002705
2706
2707 def get_default_power_method(self):
2708 """
2709 Get the default power method for power_on/off/cycle() methods.
2710 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2711 """
2712 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002713 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002714 if self.servo and self.servo.supports_built_in_pd_control():
2715 self._default_power_method = self.POWER_CONTROL_CCD
2716 else:
2717 logging.debug('Either servo is unitialized or the servo '
2718 'setup does not support pd controls. Falling '
2719 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002720 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002721
2722
2723 def find_usb_devices(self, idVendor, idProduct):
2724 """
2725 Get usb device sysfs name for specific device.
2726
2727 @param idVendor Vendor ID to search in sysfs directory.
2728 @param idProduct Product ID to search in sysfs directory.
2729
2730 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2731 """
2732 # Look for matching file and cut at position 7 to get dir name.
2733 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2734
2735 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2736 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2737
2738 # Use uniq -d to print duplicate line from both command
2739 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2740
2741 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2742
2743
2744 def bind_usb_device(self, usb_node):
2745 """
2746 Bind usb device
2747
2748 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2749 """
2750 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2751 self.run(cmd, ignore_status=True)
2752
2753
2754 def unbind_usb_device(self, usb_node):
2755 """
2756 Unbind usb device
2757
2758 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2759 """
2760 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2761 self.run(cmd, ignore_status=True)
2762
2763
2764 def get_wlan_ip(self):
2765 """
2766 Get ip address of wlan interface.
2767
2768 @return ip address of wlan or empty string if wlan is not connected.
2769 """
2770 cmds = [
2771 'iw dev', # List wlan physical device
2772 'grep Interface', # Grep only interface name
2773 'cut -f 2 -d" "', # Cut the name part
2774 'xargs ifconfig', # Feed it to ifconfig to get ip
2775 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2776 'cut -f 2 -d " "' # Cut the ip part
2777 ]
2778 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002779
2780 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2781 """
2782 Connect to wifi network
2783
2784 @param ssid SSID of the wifi network.
2785 @param passphrase Passphrase of the wifi network. None if not existed.
2786 @param security Security of the wifi network. Default to "psk" if
2787 passphase is given without security. Possible values
2788 are "none", "psk", "802_1x".
2789
2790 @return True if succeed, False if not.
2791 """
2792 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2793 if passphrase:
2794 cmd += ' ' + passphrase
2795 if security:
2796 cmd += ' ' + security
2797 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002798
2799 def get_device_repair_state(self):
2800 """Get device repair state"""
2801 return self._device_repair_state
2802
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002803 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002804 """Set device repair state.
2805
2806 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002807 file in result directory. The file will be read by Lucifer. The
2808 file will not be created if result directory not specified.
2809
2810 @params state: The new state for the device.
2811 @params resultdir: The path to result directory. If path not provided
2812 will be attempt to get retrieve it from job
2813 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002814 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002815 resultdir = resultdir or getattr(self.job, 'resultdir', '')
2816 if resultdir:
2817 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07002818 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002819 logging.info('Set device state as %s. '
2820 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07002821 else:
2822 logging.debug('Cannot write the device state due missing info '
2823 'about result dir.')
2824 self._device_repair_state = state
2825
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002826 def set_device_needs_replacement(self, resultdir=None):
2827 """Set device as required replacement.
2828
2829 @params resultdir: The path to result directory. If path not provided
2830 will be attempt to get retrieve it from job
2831 if present.
2832 """
2833 self.set_device_repair_state(
2834 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
2835 resultdir=resultdir)
2836
2837 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002838 """Check if device require manual attention to be fixed.
2839
2840 The state 'needs_manual_repair' can be set when auto repair cannot
2841 fix the device due hardware or cable issues.
2842 """
2843 # ignore the logic if state present
2844 # state can be set by any cros repair actions
Otabek Kasimovc9812582020-10-08 18:52:52 -07002845 if self.get_device_repair_state() or not self._repair_strategy:
Otabek Kasimov6825b762020-06-23 23:42:44 -07002846 return
Otabek Kasimovc9812582020-10-08 18:52:52 -07002847 dut_ssh_verifier = self._repair_strategy.verifier_is_good('ssh')
2848 if dut_ssh_verifier == hosts.VERIFY_SUCCESS:
2849 # DUT us sshable and we still have many options to repair it.
2850 return
2851 # when we cannot ssh to the DUT and we have hardware or set-up
2852 # issues with servo then we need request manual repair for this DUT.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002853 servo_state_required_manual_fix = [
Otabek Kasimovc9812582020-10-08 18:52:52 -07002854 servo_constants.SERVO_STATE_DUT_NOT_CONNECTED,
2855 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
Otabek Kasimov6825b762020-06-23 23:42:44 -07002856 ]
2857 if self.get_servo_state() in servo_state_required_manual_fix:
Otabek Kasimovc9812582020-10-08 18:52:52 -07002858 logging.info(
2859 'DUT required manual repair because it is not sshable'
2860 ' and possible have setup issue with Servo. Please verify'
2861 ' all connections and present of devices.')
2862 self.set_device_repair_state(
2863 cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)
Otabek Kasimov42506d02020-07-29 14:44:57 -07002864
2865 def is_file_system_writable(self, testdirs=None):
2866 """Check is the file systems are writable.
2867
2868 The standard linux response to certain unexpected file system errors
2869 (including hardware errors in block devices) is to change the file
2870 system status to read-only. This checks that that hasn't happened.
2871
2872 @param testdirs: List of directories to check. If no data provided
2873 then '/mnt/stateful_partition' and '/var/tmp'
2874 directories will be checked.
2875
2876 @returns boolean whether file-system writable.
2877 """
2878 def _check_dir(testdir):
2879 # check if we can create a file
2880 filename = os.path.join(testdir, 'writable_my_test_file')
2881 command = 'touch %s && rm %s' % (filename, filename)
2882 rv = self.run(command=command,
2883 timeout=30,
2884 ignore_status=True)
2885 is_writable = rv.exit_status == 0
2886 if not is_writable:
2887 logging.info('Cannot create a file in "%s"!'
2888 ' Probably the FS is read-only', testdir)
2889 logging.info("FileSystem is not writable!")
2890 return False
2891 return True
2892
2893 if not testdirs or len(testdirs) == 0:
2894 # N.B. Order matters here: Encrypted stateful is loop-mounted
2895 # from a file in unencrypted stateful, so we don't test for
2896 # errors in encrypted stateful if unencrypted fails.
2897 testdirs = ['/mnt/stateful_partition', '/var/tmp']
2898
2899 for dir in testdirs:
2900 # loop will be stopped if any directory fill fail the check
2901 try:
2902 if not _check_dir(dir):
2903 return False
2904 except Exception as e:
2905 # here expected only timeout error, all other will
2906 # be catch by 'ignore_status=True'
2907 logging.debug('Fail to check %s to write in it', dir)
2908 return False
2909 return True
Garry Wang1a493d82020-08-31 21:01:19 -07002910
Dana Goyettec172b172020-07-29 16:26:15 -07002911 def blocking_sync(self, freeze_for_reset=False):
2912 """Sync root device and internal device, via script.
2913
2914 The actual calls end up logged by the run() call, since they're printed
2915 to stdout/stderr in the script.
2916
2917 @param freeze_for_reset: if True, prepare for reset by blocking writes
2918 (only if enable_fs_sync_fsfreeze=True)
2919 """
2920
2921 if freeze_for_reset and self.USE_FSFREEZE:
2922 logging.info('Blocking sync and freeze')
2923 elif freeze_for_reset:
2924 logging.info('Blocking sync for reset')
2925 else:
2926 logging.info('Blocking sync')
2927
2928 # client/bin is installed on the DUT as /usr/local/autotest/bin
2929 sync_cmd = '/usr/local/autotest/bin/fs_sync.py'
2930 if freeze_for_reset and self.USE_FSFREEZE:
2931 sync_cmd += ' --freeze'
2932 return self.run(sync_cmd)
2933
Garry Wang1a493d82020-08-31 21:01:19 -07002934 def setup_device_health_profile(self):
2935 """Setup device health profile for repair/provision task to consume.
2936 """
Garry Wanga2e78172020-09-09 23:49:07 -07002937 if self.health_profile:
2938 logging.info('Device health profile has already been initialized.')
Garry Wang1a493d82020-08-31 21:01:19 -07002939 if not self._servo_host:
2940 logging.info('Servohost is not instantiated, skip device'
2941 ' health profile setup...')
2942 return
2943 # Also skip setup health profile if it's a task runs locally.
2944 if self._servo_host.is_localhost():
2945 logging.info('Servohost is a localhost, skip device'
2946 ' health profile setup...')
2947 return
2948 try:
2949 self.health_profile = device_health_profile.DeviceHealthProfile(
2950 self, self._servo_host)
2951 except Exception as e:
2952 logging.warning('Failed to setup device health profile; %s', e)
Garry Wanga2e78172020-09-09 23:49:07 -07002953
2954 def set_health_profile_dut_state(self, state):
2955 if not self.health_profile:
2956 logging.debug('Device health profile is not initialized, skip'
2957 ' set dut state.')
2958 return
2959 reset_counters = state in profile_constants.STATES_NEED_RESET_COUNTER
2960 self.health_profile.update_dut_state(state, reset_counters)
Garry Wang53fc8f32020-09-18 13:30:08 -07002961
2962 def require_snk_mode_in_recovery(self):
2963 """Check whether we need to switch servo_v4 role to snk when
2964 booting into recovery mode. (See crbug.com/1129165)
2965 """
2966 info = self.host_info_store.get()
2967 if info.get_label_value('power') != 'battery':
2968 logging.info(
2969 '%s does not has battery, snk mode is not needed'
2970 ' for recovery.', self.hostname)
2971 return False
2972 if not self.servo.supports_built_in_pd_control():
2973 logging.info('Power delivery is not supported on this servo, snk'
2974 ' mode is not needed for recovery.')
2975 return False
2976 try:
2977 #TODO(xianuowang@) move MIN_BATTERY_LEVEL to cros_constant
2978 battery_percent = self.servo.get('battery_charge_percent')
2979 if battery_percent < cros_repair.MIN_BATTERY_LEVEL:
2980 logging.info(
2981 'Current battery level %s%% below %s%% threshold, we'
2982 ' will attempt to boot host in recovery mode without'
2983 ' changing servo to snk mode. Please note the host may'
2984 ' not able to see usb drive in recovery mode later due'
2985 ' to servo not in snk mode.', battery_percent,
2986 cros_repair.MIN_BATTERY_LEVEL)
2987 return False
2988 except Exception as e:
2989 logging.info(
2990 'Unexpected error occurred when getting'
2991 ' battery_charge_percent from servo; %s', str(e))
2992 return False
2993 return True
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07002994
2995 def _set_servo_topology(self):
2996 """Set servo-topology info to the host-info."""
2997 logging.debug('Try to save servo topology to host-info.')
2998 if not self._servo_host:
2999 logging.info('Servo host is not initilized.')
3000 return
3001 if not self._servo_host.is_servo_topology_supported():
3002 logging.info('Servo-topology is not supported.')
3003 return
3004 servo_topology = self._servo_host.get_topology()
3005 if not servo_topology or servo_topology.is_empty():
3006 logging.info('Servo topology is empty')
3007 return
3008 servo_topology.save(self.host_info_store)