blob: 3352350e061b3898c31373111795dbb74dcb0f83 [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
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800110 # REBOOT_TIMEOUT: How long to wait for a reboot.
111 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700112 # We have a long timeout to ensure we don't flakily fail due to other
113 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700114 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
115 # return from reboot' bug is solved.
116 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700117
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800118 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
119 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700120 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
121 # since changing servo role will reset USB state
122 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800123 _USB_POWER_TIMEOUT = 5
124 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700125 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800126
Fang Dengdeba14f2014-11-14 11:54:09 -0800127 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
128 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700129
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800130 # Constants used in ping_wait_up() and ping_wait_down().
131 #
132 # _PING_WAIT_COUNT is the approximate number of polling
133 # cycles to use when waiting for a host state change.
134 #
135 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
136 # for arguments to the internal _ping_wait_for_status()
137 # method.
138 _PING_WAIT_COUNT = 40
139 _PING_STATUS_DOWN = False
140 _PING_STATUS_UP = True
141
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800142 # Allowed values for the power_method argument.
143
Garry Wang5e5538a2019-04-08 15:36:18 -0700144 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
145 # DUTs except those with servo_v4 CCD.
146 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
147 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800148 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
149 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
150 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700151 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800152 POWER_CONTROL_SERVO = 'servoj10'
153 POWER_CONTROL_MANUAL = 'manual'
154
155 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700156 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800157 POWER_CONTROL_SERVO,
158 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800159
Simran Basi5e6339a2013-03-21 11:34:32 -0700160 _RPM_OUTLET_CHANGED = 'outlet_changed'
161
Dan Shi9cb0eec2014-06-03 09:04:50 -0700162 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700163 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700164 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700165
Brent Peterson1cb623a2020-01-09 13:14:28 -0800166 # Regular expression for extracting EC version string
167 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
168
169 # Regular expression for extracting BIOS version string
170 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
171
Brent Petersonc70a1832020-01-24 15:54:35 -0800172 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700173 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800174
J. Richard Barnette964fba02012-10-24 17:34:29 -0700175 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800176 def check_host(host, timeout=10):
177 """
178 Check if the given host is a chrome-os host.
179
180 @param host: An ssh host representing a device.
181 @param timeout: The timeout for the run command.
182
183 @return: True if the host device is chromeos.
184
beeps46dadc92013-11-07 14:07:10 -0800185 """
186 try:
Allen Liad719c12017-06-27 23:48:04 +0000187 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700188 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700189 '! grep -q moblab /etc/lsb-release && '
190 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700191 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700192 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000193 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700194 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
195 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800196 return not (
197 lsbrelease_utils.is_jetstream(
198 lsb_release_content=lsb_release_content) or
199 lsbrelease_utils.is_gce_board(
200 lsb_release_content=lsb_release_content))
201
beeps46dadc92013-11-07 14:07:10 -0800202 except (error.AutoservRunError, error.AutoservSSHTimeout):
203 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700204
205 return False
beeps46dadc92013-11-07 14:07:10 -0800206
207
208 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800209 def get_chameleon_arguments(args_dict):
210 """Extract chameleon options from `args_dict` and return the result.
211
212 Recommended usage:
213 ~~~~~~~~
214 args_dict = utils.args_to_dict(args)
215 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
216 host = hosts.create_host(machine, chameleon_args=chameleon_args)
217 ~~~~~~~~
218
219 @param args_dict Dictionary from which to extract the chameleon
220 arguments.
221 """
Sam McNally66594ca2019-12-09 12:45:44 +1100222 chameleon_args = {key: args_dict[key]
223 for key in ('chameleon_host', 'chameleon_port')
224 if key in args_dict}
225 if 'chameleon_ssh_port' in args_dict:
226 chameleon_args['port'] = int(args_dict['chameleon_ssh_port'])
227 return chameleon_args
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800228
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800229 @staticmethod
230 def get_btpeer_arguments(args_dict):
231 """Extract btpeer options from `args_dict` and return the result.
232
233 This is used to parse details of Bluetooth peer.
234 Recommended usage:
235 ~~~~~~~~
236 args_dict = utils.args_to_dict(args)
237 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700238 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800239 ~~~~~~~~
240
241 @param args_dict: Dictionary from which to extract the btpeer
242 arguments.
243 """
244 if 'btpeer_host_list' in args_dict:
245 result = []
246 for btpeer in args_dict['btpeer_host_list'].split(','):
247 result.append({key: value for key,value in
248 zip(('btpeer_host','btpeer_port'),
249 btpeer.split(':'))})
250 return result
251 else:
Anand K Mistrye8933092020-08-05 14:49:41 +1000252 return {key: args_dict[key]
253 for key in ('btpeer_host', 'btpeer_port', 'btpeer_ssh_port')
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800254 if key in args_dict}
255
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800256
257 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700258 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800259 """Extract chameleon options from `args_dict` and return the result.
260
261 Recommended usage:
262 ~~~~~~~~
263 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700264 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
265 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800266 ~~~~~~~~
267
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700268 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800269 arguments.
270 """
Allen Li083866b2016-08-18 10:07:10 -0700271 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700272 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700273 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800274
275
276 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800277 def get_servo_arguments(args_dict):
278 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800279
280 Recommended usage:
281 ~~~~~~~~
282 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700283 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800284 host = hosts.create_host(machine, servo_args=servo_args)
285 ~~~~~~~~
286
287 @param args_dict Dictionary from which to extract the servo
288 arguments.
289 """
Garry Wang11b5e872020-03-11 15:14:08 -0700290 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
291 servo_constants.SERVO_PORT_ATTR,
292 servo_constants.SERVO_BOARD_ATTR,
293 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200294 servo_args = {key: args_dict[key]
295 for key in servo_attrs
296 if key in args_dict}
297 return (
298 None
Garry Wang11b5e872020-03-11 15:14:08 -0700299 if servo_constants.SERVO_HOST_ATTR in servo_args
300 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200301 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700302
J. Richard Barnette964fba02012-10-24 17:34:29 -0700303
J. Richard Barnette91137f02016-03-10 16:52:26 -0800304 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700305 pdtester_args=None, try_lab_servo=False,
Shijin Abraham78ce4402020-09-08 22:04:27 -0700306 try_servo_repair=False, ssh_verbosity_flag='',
307 ssh_options='', *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800308 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700309
Fang Denge545abb2014-12-30 18:43:47 -0800310 This method will attempt to create the test-assistant object
311 (chameleon/servo) when it is needed by the test. Check
312 the docstring of chameleon_host.create_chameleon_host and
313 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700314
Fang Denge545abb2014-12-30 18:43:47 -0800315 @param hostname: Hostname of the dut.
316 @param chameleon_args: A dictionary that contains args for creating
317 a ChameleonHost. See chameleon_host for details.
318 @param servo_args: A dictionary that contains args for creating
319 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700320 @param try_lab_servo: When true, indicates that an attempt should
321 be made to create a ServoHost for a DUT in
322 the test lab, even if not required by
323 `servo_args`. See servo_host for details.
324 @param try_servo_repair: If a servo host is created, check it
325 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800326 See servo_host for details.
327 @param ssh_verbosity_flag: String, to pass to the ssh command to control
328 verbosity.
329 @param ssh_options: String, other ssh options to pass to the ssh
330 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700331 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700332 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700333 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800334 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700335 # hold special dut_state for repair process
336 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700337 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700338 # self.env is a dictionary of environment variable settings
339 # to be exported for commands run on the host.
340 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
341 # errors that might happen.
342 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700343 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700344 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700345 _servo_host, servo_state = servo_host.create_servo_host(
346 dut=self,
347 servo_args=servo_args,
348 try_lab_servo=try_lab_servo,
349 try_servo_repair=try_servo_repair,
350 dut_host_info=self.host_info_store.get())
351 self.set_servo_host(_servo_host, servo_state)
Garry Wang1a493d82020-08-31 21:01:19 -0700352 self.health_profile = None
Garry Wang5e5538a2019-04-08 15:36:18 -0700353 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700354
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800355 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800356 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700357 dut=self.hostname,
358 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800359 if self._chameleon_host:
360 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800361 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800362 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700363
Shijin Abraham78ce4402020-09-08 22:04:27 -0700364 # Bluetooth peers will be populated by the test if needed
365 self._btpeer_host_list = []
366 self.btpeer_list = []
367 self.btpeer = None
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800368
howardchung83e55272019-08-08 14:08:05 +0800369 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700370 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700371 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800372
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700373 if self._pdtester_host:
374 self.pdtester_servo = self._pdtester_host.get_servo()
375 logging.info('pdtester_servo: %r', self.pdtester_servo)
376 # Create the pdtester object used to access the ec uart
377 self.pdtester = pdtester.PDTester(self.pdtester_servo,
378 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800379 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700380 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800381
Fang Deng5d518f42013-08-02 14:04:32 -0700382
Shijin Abraham78ce4402020-09-08 22:04:27 -0700383 def initialize_btpeer(self, btpeer_args=[]):
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800384 """ Initialize the Bluetooth peers
385
386 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
387 is chameleon host on Raspberry Pi.
388 @param btpeer_args: A dictionary that contains args for creating
389 a ChameleonHost. See chameleon_host for details.
390
391 """
Shijin Abraham78ce4402020-09-08 22:04:27 -0700392 logging.debug('Attempting to initialize bluetooth peers if available')
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700393 try:
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700394 if type(btpeer_args) is list:
395 btpeer_args_list = btpeer_args
396 else:
397 btpeer_args_list = [btpeer_args]
398
399 self._btpeer_host_list = chameleon_host.create_btpeer_host(
400 dut=self.hostname, btpeer_args_list=btpeer_args_list)
401 logging.debug('Bluetooth peer hosts are %s',
402 self._btpeer_host_list)
403 self.btpeer_list = [_host.create_chameleon_board() for _host in
404 self._btpeer_host_list if _host is not None]
405
406 if len(self.btpeer_list) > 0:
407 self.btpeer = self.btpeer_list[0]
408
409 logging.debug('After initialize_btpeer btpeer_list %s '
410 'btpeer_host_list is %s and btpeer is %s',
411 self.btpeer_list, self._btpeer_host_list,
412 self.btpeer)
413 except Exception as e:
414 logging.error('Exception %s in initialize_btpeer', str(e))
415
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800416
417
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000418 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700419 """Get latest stable cros image name from AFE.
420
421 Use the board name from the info store. Should that fail, try to
422 retrieve the board name from the host's installed image itself.
423
424 @returns: current stable cros image name for this host.
425 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700426 info = self.host_info_store.get()
427 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700428 logging.warn('No board label value found. Trying to infer '
429 'from the host itself.')
430 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700431 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700432 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
433 logging.error('Also failed to get the board name from the DUT '
434 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700435 raise error.AutoservError('Cannot determine board of the DUT'
436 ' while getting repair image name.')
437 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500438
439
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700440 def host_version_prefix(self, image):
441 """Return version label prefix.
442
443 In case the CrOS provisioning version is something other than the
444 standard CrOS version e.g. CrOS TH version, this function will
445 find the prefix from provision.py.
446
447 @param image: The image name to find its version prefix.
448 @returns: A prefix string for the image type.
449 """
450 return provision.get_version_label_prefix(image)
451
Andrew Luo3332ab22020-04-28 16:42:03 -0700452 def stage_build_to_usb(self, build):
453 """Stage the current ChromeOS image on the USB stick connected to the
454 servo.
455
456 @param build: The build to download and send to USB.
457 """
458 if not self.servo:
459 raise error.TestError('Host %s does not have servo.' %
460 self.hostname)
461
462 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700463
464 try:
465 self.servo.image_to_servo_usb(update_url)
466 finally:
467 # servo.image_to_servo_usb turned the DUT off, so turn it back on
468 logging.debug('Turn DUT power back on.')
469 self.servo.get_power_state_controller().power_on()
470
Andrew Luo3332ab22020-04-28 16:42:03 -0700471 logging.debug('ChromeOS image %s is staged on the USB stick.',
472 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700473
beepsdae65fd2013-07-26 16:24:41 -0700474 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700475 """
476 Make sure job_repo_url of this host is valid.
477
joychen03eaad92013-06-26 09:55:21 -0700478 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700479 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
480 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
481 download and extract it. If the devserver embedded in the url is
482 unresponsive, update the job_repo_url of the host after staging it on
483 another devserver.
484
485 @param job_repo_url: A url pointing to the devserver where the autotest
486 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700487 @param tag: The tag from the server job, in the format
488 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700489
490 @raises DevServerException: If we could not resolve a devserver.
491 @raises AutoservError: If we're unable to save the new job_repo_url as
492 a result of choosing a new devserver because the old one failed to
493 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700494 @raises urllib2.URLError: If the devserver embedded in job_repo_url
495 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700496 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800497 info = self.host_info_store.get()
498 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700499 if not job_repo_url:
500 logging.warning('No job repo url set on host %s', self.hostname)
501 return
502
503 logging.info('Verifying job repo url %s', job_repo_url)
504 devserver_url, image_name = tools.get_devserver_build_from_package_url(
505 job_repo_url)
506
beeps0c865032013-07-30 11:37:06 -0700507 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700508
509 logging.info('Staging autotest artifacts for %s on devserver %s',
510 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700511
512 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700513 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700514 stage_time = time.time() - start_time
515
516 # Record how much of the verification time comes from a devserver
517 # restage. If we're doing things right we should not see multiple
518 # devservers for a given board/build/branch path.
519 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800520 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700521 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800522 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700523 pass
524 else:
beeps0c865032013-07-30 11:37:06 -0700525 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700526 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700527 stats_key = {
528 'board': board,
529 'build_type': build_type,
530 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700531 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700532 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800533
534 monarch_fields = {
535 'board': board,
536 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800537 'branch': branch,
538 'dev_server': devserver,
539 }
540 metrics.Counter(
541 'chromeos/autotest/provision/verify_url'
542 ).increment(fields=monarch_fields)
543 metrics.SecondsDistribution(
544 'chromeos/autotest/provision/verify_url_duration'
545 ).add(stage_time, fields=monarch_fields)
546
547
Dan Shicf4d2032015-03-12 15:04:21 -0700548 def stage_server_side_package(self, image=None):
549 """Stage autotest server-side package on devserver.
550
551 @param image: Full path of an OS image to install or a build name.
552
553 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700554
555 @raise: error.AutoservError if fail to locate the build to test with, or
556 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700557 """
Dan Shid37736b2016-07-06 15:10:29 -0700558 # If enable_drone_in_restricted_subnet is False, do not set hostname
559 # in devserver.resolve call, so a devserver in non-restricted subnet
560 # is picked to stage autotest server package for drone to download.
561 hostname = self.hostname
562 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
563 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700564 if image:
565 image_name = tools.get_build_from_image(image)
566 if not image_name:
567 raise error.AutoservError(
568 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700569 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700570 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700571 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800572 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700573 if job_repo_url:
574 devserver_url, image_name = (
575 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700576 # If enable_drone_in_restricted_subnet is True, use the
577 # existing devserver. Otherwise, resolve a new one in
578 # non-restricted subnet.
579 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
580 ds = dev_server.ImageServer(devserver_url)
581 else:
582 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800583 elif info.build is not None:
584 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700585 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700586 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800587 raise error.AutoservError(
588 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700589 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700590
591 # Get the OS version of the build, for any build older than
592 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
593 match = re.match('.*/R\d+-(\d+)\.', image_name)
594 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700595 raise error.AutoservError(
596 'Build %s is older than %s. Server side packaging is '
597 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700598
Dan Shicf4d2032015-03-12 15:04:21 -0700599 ds.stage_artifacts(image_name, ['autotest_server_package'])
600 return '%s/static/%s/%s' % (ds.url(), image_name,
601 'autotest_server_package.tar.bz2')
602
603
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700604 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700605 """Stage a build on a devserver and return the update_url.
606
607 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700608 @param artifact: a string like 'test_image'. Requests
609 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700610 @returns a tuple of (image_name, URL) like
611 (lumpy-release/R27-3837.0.0,
612 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700613 """
614 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000615 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700616 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800617 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700618 devserver.stage_artifacts(image_name, [artifact])
619 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700620 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700621 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700622 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700623 else:
624 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700625
626
beepse539be02013-07-31 21:57:39 -0700627 def stage_factory_image_for_servo(self, image_name):
628 """Stage a build on a devserver and return the update_url.
629
630 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700631
beepse539be02013-07-31 21:57:39 -0700632 @return: An update URL, eg:
633 http://<devserver>/static/canary-channel/\
634 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700635
636 @raises: ValueError if the factory artifact name is missing from
637 the config.
638
beepse539be02013-07-31 21:57:39 -0700639 """
640 if not image_name:
641 logging.error('Need an image_name to stage a factory image.')
642 return
643
Dan Shib8540a52015-07-16 14:18:23 -0700644 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700645 'CROS', 'factory_artifact', type=str, default='')
646 if not factory_artifact:
647 raise ValueError('Cannot retrieve the factory artifact name from '
648 'autotest config, and hence cannot stage factory '
649 'artifacts.')
650
beepse539be02013-07-31 21:57:39 -0700651 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800652 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700653 devserver.stage_artifacts(
654 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700655 [factory_artifact],
656 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700657
658 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
659
660
Laurence Goodby778c9a42017-05-24 19:24:07 -0700661 def prepare_for_update(self):
662 """Prepares the DUT for an update.
663
664 Subclasses may override this to perform any special actions
665 required before updating.
666 """
Laurence Goodby468de252017-06-08 17:22:53 -0700667 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700668
669
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800670 def _clear_fw_version_labels(self, rw_only):
671 """Clear firmware version labels from the machine.
672
673 @param rw_only: True to only clear fwrw_version; otherewise, clear
674 both fwro_version and fwrw_version.
675 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700676 info = self.host_info_store.get()
677 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800678 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700679 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
680 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700681
682
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800683 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700684 """Add firmware version label to the machine.
685
686 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800687 @param rw_only: True to only add fwrw_version; otherwise, add both
688 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700689
690 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700691 info = self.host_info_store.get()
692 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800693 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700694 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
695 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700696
697
Namyoon Woo33f38852020-04-13 17:26:58 -0700698 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800699 """Search for the latest package release version from the image archive,
700 and return it.
701
Namyoon Woo33f38852020-04-13 17:26:58 -0700702 @param platform: platform name, a.k.a. board or model
703 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800704
Namyoon Woo33f38852020-04-13 17:26:58 -0700705 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
706 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800707 or None if LATEST release file does not exist.
708 """
709
Namyoon Woo33f38852020-04-13 17:26:58 -0700710 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800711
Namyoon Woo33f38852020-04-13 17:26:58 -0700712 # Search the image path in reference board archive as well.
713 # For example, bob has its binary image under its reference board (gru)
714 # image archive.
715 if ref_board:
716 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800717
Namyoon Woo33f38852020-04-13 17:26:58 -0700718 for board in platforms:
719 # Read 'LATEST-1.0.0' file
720 branch_dir = provision.FW_BRANCH_GLOB % board
721 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
722 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800723
Namyoon Woo33f38852020-04-13 17:26:58 -0700724 try:
725 # The result could be one or more.
726 result = utils.system_output('gsutil ls -d ' + latest_file)
727
728 candidates = re.findall('gs://.*', result)
729
730 # Found the directory candidates. No need to check the other
731 # board name cadidates. Let's break the loop.
732 break
733 except error.CmdError:
734 # It doesn't exist. Let's move on to the next item.
735 pass
736 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800737 logging.error('No LATEST release info is available.')
738 return None
739
Namyoon Woo406c7d42020-01-24 15:57:11 -0800740 for cand_dir in candidates:
741 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800742
Namyoon Woo406c7d42020-01-24 15:57:11 -0800743 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700744 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800745 try:
746 # Check if release_path does exist.
747 release = utils.system_output('gsutil ls -d ' + release_path)
748 # Now 'release' has a full directory path: e.g.
749 # gs://chromeos-image-archive/firmware-octopus-11297.B-
750 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
751
752 # Remove "gs://chromeos-image-archive".
753 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
754
755 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
756 return release.strip('/')
757 except error.CmdError:
758 # The directory might not exist. Let's try next candidate.
759 pass
760 else:
761 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800762
Brent Peterson1cb623a2020-01-09 13:14:28 -0800763 @staticmethod
764 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800765 """Get version string from binary image using regular expression.
766
767 @param image: Binary image to search
768 @param version_regex: Regular expression to search for
769
770 @return Version string
771
772 @raises TestFail if no version string is found in image
773 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800774 with open(image, 'rb') as f:
775 image_data = f.read()
Derek Beckett98345552020-08-31 16:07:22 -0700776 match = re.findall(version_regex,
777 image_data.decode('ISO-8859-1', errors='ignore'))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800778 if match:
779 return match[0]
780 else:
781 raise error.TestFail('Failed to read version from %s.' % image)
782
783
Garry Wangad2a1712020-03-26 15:06:43 -0700784 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800785 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700786 try_scp=False, install_ec=True, install_bios=True,
787 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700788 """Install firmware to the DUT.
789
790 Use stateful update if the DUT is already running the same build.
791 Stateful update does not update kernel and tends to run much faster
792 than a full reimage. If the DUT is running a different build, or it
793 failed to do a stateful update, full update, including kernel update,
794 will be applied to the DUT.
795
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800796 Once a host enters firmware_install its fw[ro|rw]_version label will
797 be removed. After the firmware is updated successfully, a new
798 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700799
800 @param build: The build version to which we want to provision the
801 firmware of the machine,
802 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800803 @param rw_only: True to only install firmware to its RW portions. Keep
804 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700805 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800806 @param local_tarball: Path to local firmware image for installing
807 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800808 @param verify_version: True to verify EC and BIOS versions after
809 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800810 @param try_scp: False to always program using servo, true to try copying
811 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700812 @param install_ec: True to install EC FW, and False to skip it.
813 @param install_bios: True to install BIOS, and False to skip it.
814 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700815
816 TODO(dshi): After bug 381718 is fixed, update here with corresponding
817 exceptions that could be raised.
818
819 """
820 if not self.servo:
821 raise error.TestError('Host %s does not have servo.' %
822 self.hostname)
823
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700824 # Get the DUT board name from AFE.
825 info = self.host_info_store.get()
826 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700827 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800828
829 if board is None or board == '':
830 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700831
Namyoon Woo382e5892020-05-20 16:48:40 -0700832 # if board_as argument is passed, then use it instead of the original
833 # board name.
834 if board_as:
835 board = board_as
836
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700837 if model is None or model == '':
Namyoon Woofb16eae2020-08-14 10:02:39 -0700838 try:
839 model = self.get_platform_from_fwid()
840 except Exception as e:
841 logging.warn('Dut is unresponsive: %s', str(e))
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700842
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800843 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700844 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800845 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700846 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800847
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700848 try:
849 ds = dev_server.ImageServer.resolve(build, self.hostname)
850 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800851
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700852 if not dest:
853 tmpd = autotemp.tempdir(unique_id='fwimage')
854 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800855
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700856 # Download firmware image
857 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
858 local_tarball = os.path.join(dest, os.path.basename(fwurl))
859 ds.download_file(fwurl, local_tarball)
860 except Exception as e:
861 raise error.TestError('Failed to download firmware package: %s'
862 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700863
Namyoon Woo382e5892020-05-20 16:48:40 -0700864 ec_image = None
865 if install_ec:
866 # Extract EC image from tarball
867 logging.info('Extracting EC image.')
868 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800869
Namyoon Woo382e5892020-05-20 16:48:40 -0700870 bios_image = None
871 if install_bios:
872 # Extract BIOS image from tarball
873 logging.info('Extracting BIOS image.')
874 bios_image = self.servo.extract_bios_image(board, model,
875 local_tarball)
876
877 if not bios_image and not ec_image:
878 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800879
Brent Petersonc70a1832020-01-24 15:54:35 -0800880 # Clear firmware version labels
881 self._clear_fw_version_labels(rw_only)
882
883 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800884 try:
Garry Wang50e4a492020-08-05 12:29:57 -0700885 # Check if copying to DUT is enabled and DUT is available
886 if try_scp and self.is_up():
Brent Petersonc70a1832020-01-24 15:54:35 -0800887 # DUT is available, make temp firmware directory to store images
888 logging.info('Making temp folder.')
889 dest_folder = '/tmp/firmware'
890 self.run('mkdir -p ' + dest_folder)
891
Namyoon Woo68b68082020-06-02 13:13:14 -0700892 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800893
Namyoon Woo382e5892020-05-20 16:48:40 -0700894 if bios_image:
895 # Send BIOS firmware image to DUT
896 logging.info('Sending BIOS firmware.')
897 dest_bios_path = os.path.join(dest_folder,
898 os.path.basename(bios_image))
899 self.send_file(bios_image, dest_bios_path)
900
901 # Initialize firmware update command for BIOS image
902 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800903
904 # Send EC firmware image to DUT when EC image was found
905 if ec_image:
906 logging.info('Sending EC firmware.')
907 dest_ec_path = os.path.join(dest_folder,
908 os.path.basename(ec_image))
909 self.send_file(ec_image, dest_ec_path)
910
911 # Add EC image to firmware update command
912 fw_cmd += ' -e %s' % dest_ec_path
913
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700914 # Make sure command is allowed to finish even if ssh fails.
915 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
916
Brent Peterson669edf42020-02-07 15:07:54 -0800917 # Update firmware on DUT
918 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700919 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700920 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700921 except error.AutoservRunError as e:
922 if e.result_obj.exit_status != 255:
923 raise
924 elif ec_image:
925 logging.warn("DUT network dropped during update"
926 " (often caused by EC resetting USB)")
927 else:
928 logging.error("DUT network dropped during update"
929 " (unexpected, since no EC image)")
930 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800931 else:
932 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800933 if ec_image:
934 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700935 if bios_image:
936 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800937 if utils.host_is_in_lab_zone(self.hostname):
938 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800939
940 # Reboot and wait for DUT after installing firmware
941 logging.info('Rebooting DUT.')
942 self.servo.get_power_state_controller().reset()
943 time.sleep(self.servo.BOOT_DELAY)
944 self.test_wait_for_boot()
945
946 # When enabled verify EC and BIOS firmware version after programming
947 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800948 # Check programmed EC firmware when EC image was found
949 if ec_image:
950 logging.info('Checking EC firmware version.')
951 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800952 ec_version_prefix = dest_ec_version.split('_', 1)[0]
953 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800954 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800955 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800956 if dest_ec_version != image_ec_version:
957 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700958 'Failed to update EC firmware, version %s '
959 '(expected %s)' % (dest_ec_version,
960 image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800961
Namyoon Woo382e5892020-05-20 16:48:40 -0700962 if bios_image:
963 # Check programmed BIOS firmware against expected version
964 logging.info('Checking BIOS firmware version.')
965 dest_bios_version = self.get_firmware_version()
966 bios_version_prefix = dest_bios_version.split('.', 1)[0]
967 bios_regex = self._BIOS_REGEX % bios_version_prefix
968 image_bios_version = self.get_version_from_image(bios_image,
969 bios_regex)
970 if dest_bios_version != image_bios_version:
971 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700972 'Failed to update BIOS, version %s '
Namyoon Woo382e5892020-05-20 16:48:40 -0700973 '(expected %s)' % (dest_bios_version,
974 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700975 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700976 if tmpd:
977 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700978
979
beepsf079cfb2013-09-18 17:49:51 -0700980 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
981 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500982 """
983 Re-install the OS on the DUT by:
984 1) installing a test image on a USB storage device attached to the Servo
985 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800986 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700987 3) installing the image with chromeos-install.
988
Scott Zawalski62bacae2013-03-05 10:40:32 -0500989 @param image_url: If specified use as the url to install on the DUT.
990 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700991 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
992 Factory images need a longer usb_boot_timeout than regular
993 cros images.
994 @param install_timeout: The timeout to use when installing the chromeos
995 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800996
Scott Zawalski62bacae2013-03-05 10:40:32 -0500997 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700998
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800999 """
Garry Wang7b0e1b72020-03-25 19:08:59 -07001000 if image_url:
1001 logging.info('Downloading image to USB, then booting from it.'
1002 ' Usb boot timeout = %s', usb_boot_timeout)
1003 else:
1004 logging.info('Booting from USB directly. Usb boot timeout = %s',
1005 usb_boot_timeout)
1006
1007 metrics_field = {'download': bool(image_url)}
1008 metrics.Counter(
1009 'chromeos/autotest/provision/servo_install/download_image'
1010 ).increment(fields=metrics_field)
1011
Allen Li48a13fe2016-11-22 14:10:40 -08001012 with metrics.SecondsTimer(
1013 'chromeos/autotest/provision/servo_install/boot_duration'):
Garry Wang53fc8f32020-09-18 13:30:08 -07001014 need_snk = self.require_snk_mode_in_recovery()
1015 self.servo.install_recovery_image(image_url, snk_mode=need_snk)
Allen Li48a13fe2016-11-22 14:10:40 -08001016 if not self.wait_up(timeout=usb_boot_timeout):
Garry Wang53fc8f32020-09-18 13:30:08 -07001017 if need_snk:
1018 # Attempt to restore servo_v4 role to 'src' mode.
1019 self.servo.set_servo_v4_role('src')
Allen Li48a13fe2016-11-22 14:10:40 -08001020 raise hosts.AutoservRepairError(
1021 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001022 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001023
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001024 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1025 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001026 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001027 try:
1028 self.run('chromeos-tpm-recovery')
1029 except error.AutoservRunError:
1030 logging.warn('chromeos-tpm-recovery is too old.')
1031
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001032
Allen Li48a13fe2016-11-22 14:10:40 -08001033 with metrics.SecondsTimer(
1034 'chromeos/autotest/provision/servo_install/install_duration'):
1035 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001036 try:
1037 self.run('chromeos-install --yes',timeout=install_timeout)
1038 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001039 except Exception as e:
1040 storage_errors = [
1041 'No space left on device',
1042 'I/O error when trying to write primary GPT',
1043 'Input/output error while writing out',
1044 'cannot read GPT header',
Otabek Kasimov2b7e8302020-08-21 09:23:31 -07001045 'can not determine destination device',
1046 'wrong fs type',
1047 'bad superblock on',
Otabek Kasimov808cd832020-05-28 18:27:46 -07001048 ]
1049 has_error = [msg for msg in storage_errors if(msg in str(e))]
1050 if has_error:
1051 info = self.host_info_store.get()
1052 info.set_version_label(
1053 audit_const.DUT_STORAGE_STATE_PREFIX,
1054 audit_const.HW_STATE_NEED_REPLACEMENT)
1055 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001056 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001057 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001058 logging.debug(
1059 'Fail install image from USB; Storage error; %s', e)
1060 raise error.AutoservError(
1061 'Failed to install image from USB due to a suspect '
1062 'disk failure, DUT storage state changed to '
1063 'need_replacement, please check debug log '
1064 'for details.')
1065 else:
Otabek Kasimov27bb2862020-08-10 14:40:45 -07001066 # DUT will be marked for replacement if storage is bad.
1067 audit_verify.VerifyDutStorage(self).verify()
1068
Otabek Kasimov808cd832020-05-28 18:27:46 -07001069 logging.debug('Fail install image from USB; %s', e)
1070 raise error.AutoservError(
1071 'Failed to install image from USB due to unexpected '
1072 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001073 finally:
1074 # We need reset the DUT no matter re-install success or not,
1075 # as we don't want leave the DUT in boot from usb state.
1076 logging.info('Power cycling DUT through servo.')
1077 self.servo.get_power_state_controller().power_off()
1078 self.servo.switch_usbkey('off')
Garry Wang53fc8f32020-09-18 13:30:08 -07001079 if need_snk:
1080 # Attempt to restore servo_v4 role to 'src' mode.
1081 self.servo.set_servo_v4_role('src')
Garry Wang033a31e2020-04-10 17:20:49 -07001082 # N.B. The Servo API requires that we use power_on() here
1083 # for two reasons:
1084 # 1) After turning on a DUT in recovery mode, you must turn
1085 # it off and then on with power_on() once more to
1086 # disable recovery mode (this is a Parrot specific
1087 # requirement).
1088 # 2) After power_off(), the only way to turn on is with
1089 # power_on() (this is a Storm specific requirement).
1090 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001091
1092 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001093 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001094 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1095 'test image after %d seconds' %
1096 self.BOOT_TIMEOUT,
1097 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001098
1099
Garry Wanga2e78172020-09-09 23:49:07 -07001100 def set_servo_host(self, host, servo_state=None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001101 """Set our servo host member, and associated servo.
1102
1103 @param host Our new `ServoHost`.
1104 """
1105 self._servo_host = host
Derek Beckett2fb9aa02020-08-12 15:31:02 -07001106 self.servo = None
1107 self.servo_pwr_supported = None
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001108 if self._servo_host is not None:
1109 self.servo = self._servo_host.get_servo()
Derek Beckett2fb9aa02020-08-12 15:31:02 -07001110 self.servo_pwr_supported = self.servo.has_control('power_state')
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001111 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001112 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Otabek Kasimov41301a22020-05-10 15:28:21 -07001113 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001114 self.set_servo_state(servo_state)
1115
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001116
Richard Barnette9a26ad62016-06-10 12:03:08 -07001117 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001118 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001119 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001120
Richard Barnette9a26ad62016-06-10 12:03:08 -07001121 If the servo object is missing, attempt to repair the servo
1122 host. Repair failures are passed back to the caller.
1123
1124 @raise AutoservError: If there is no servo host for this CrOS
1125 host.
1126 """
1127 if self.servo:
1128 return
1129 if not self._servo_host:
1130 raise error.AutoservError('No servo host for %s.' %
1131 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001132 try:
1133 self._servo_host.repair()
1134 except:
1135 raise
1136 finally:
1137 self.set_servo_host(self._servo_host)
1138
1139
Otabek Kasimov41301a22020-05-10 15:28:21 -07001140 def set_servo_type(self):
1141 """Set servo info labels to dut host_info"""
1142 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001143 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001144 return
1145 servo_type = self.servo.get_servo_type()
1146 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001147 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001148 ' by `dut-control servo_type`! Please file a bug'
1149 ' and inform infra team as we are not expected '
1150 ' to reach this point.')
1151 return
1152 host_info = self.host_info_store.get()
1153 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1154 old_type = host_info.get_label_value(prefix)
1155 if old_type == servo_type:
1156 # do not need update
1157 return
1158 host_info.set_version_label(prefix, servo_type)
1159 self.host_info_store.commit(host_info)
1160 logging.info('ServoHost: servo_type updated to %s '
1161 '(previous: %s)', servo_type, old_type)
1162
1163
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001164 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001165 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001166 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001167 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001168 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001169 old_state = host_info.get_label_value(servo_state_prefix)
1170 if old_state == servo_state:
1171 # do not need update
1172 return
1173 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001174 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001175 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1176 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001177
1178
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001179 def get_servo_state(self):
1180 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 return host_info.get_label_value(servo_state_prefix)
1183
Dana Goyette655af512020-09-03 10:48:23 -07001184 def get_servo_usb_state(self):
1185 """Get the label value indicating the health of the USB drive.
1186
1187 @return: The label value if defined, otherwise '' (empty string).
1188 @rtype: str
1189 """
1190 host_info = self.host_info_store.get()
1191 servo_usb_state_prefix = audit_const.SERVO_USB_STATE_PREFIX
1192 return host_info.get_label_value(servo_usb_state_prefix)
1193
1194 def is_servo_usb_usable(self):
1195 """Check if the servo USB storage device is usable for FAFT.
1196
1197 @return: False if the label indicates a state that will break FAFT.
1198 True if state is okay, or if state is not defined.
1199 @rtype: bool
1200 """
1201 usb_state = self.get_servo_usb_state()
1202 return usb_state in ('', audit_const.HW_STATE_ACCEPTABLE,
1203 audit_const.HW_STATE_NORMAL,
1204 audit_const.HW_STATE_UNKNOWN)
Otabek Kasimov41301a22020-05-10 15:28:21 -07001205
Garry Wang000c6c02020-05-11 21:27:23 -07001206 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1207 if smart_usbhub_detected is None:
1208 # skip the label update here as this indicate we wasn't able
1209 # to confirm usbhub type.
1210 return
1211 host_info = self.host_info_store.get()
1212 if (smart_usbhub_detected ==
1213 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1214 # skip label update if current label match the truth.
1215 return
1216 if smart_usbhub_detected:
1217 logging.info('Adding %s label to host %s',
1218 servo_constants.SMART_USBHUB_LABEL,
1219 self.hostname)
1220 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1221 else:
1222 logging.info('Removing %s label from host %s',
1223 servo_constants.SMART_USBHUB_LABEL,
1224 self.hostname)
1225 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1226 self.host_info_store.commit(host_info)
1227
1228
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001229 def repair(self):
1230 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001231
1232 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001233 not call back to the parent class, but instead relies on
1234 `self._repair_strategy` to coordinate the verification and
1235 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001236 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001237 message = 'Beginning repair for host %s board %s model %s'
1238 info = self.host_info_store.get()
1239 message %= (self.hostname, info.board, info.model)
1240 self.record('INFO', None, None, message)
Garry Wanga2e78172020-09-09 23:49:07 -07001241 profile_state = profile_constants.DUT_STATE_READY
Shijin Abraham78ce4402020-09-08 22:04:27 -07001242 # Initialize bluetooth peers
1243 self.initialize_btpeer()
Garry Wang87af1d02020-05-26 17:55:54 -07001244 try:
1245 self._repair_strategy.repair(self)
1246 except hosts.AutoservVerifyDependencyError as e:
1247 # We don't want flag a DUT as failed if only non-critical
1248 # verifier(s) failed during the repair.
1249 if e.is_critical():
Garry Wanga2e78172020-09-09 23:49:07 -07001250 profile_state = profile_constants.DUT_STATE_REPAIR_FAILED
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001251 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001252 raise
Garry Wanga2e78172020-09-09 23:49:07 -07001253 finally:
1254 self.set_health_profile_dut_state(profile_state)
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001255
Richard Barnette82c35912012-11-20 10:09:10 -08001256
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001257 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001258 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001259 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001260
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001261 if self._chameleon_host:
1262 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001263
Garry Wang1a493d82020-08-31 21:01:19 -07001264 if self.health_profile:
1265 try:
1266 self.health_profile.close()
1267 except Exception as e:
1268 logging.warning(
1269 'Failed to finalize device health profile; %s', e)
1270
xixuand6011f12016-12-08 15:01:58 -08001271 if self._servo_host:
1272 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001273
Dan Shi49ca0932014-11-14 11:22:27 -08001274 def get_power_supply_info(self):
1275 """Get the output of power_supply_info.
1276
1277 power_supply_info outputs the info of each power supply, e.g.,
1278 Device: Line Power
1279 online: no
1280 type: Mains
1281 voltage (V): 0
1282 current (A): 0
1283 Device: Battery
1284 state: Discharging
1285 percentage: 95.9276
1286 technology: Li-ion
1287
1288 Above output shows two devices, Line Power and Battery, with details of
1289 each device listed. This function parses the output into a dictionary,
1290 with key being the device name, and value being a dictionary of details
1291 of the device info.
1292
1293 @return: The dictionary of power_supply_info, e.g.,
1294 {'Line Power': {'online': 'yes', 'type': 'main'},
1295 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001296 @raise error.AutoservRunError if power_supply_info tool is not found in
1297 the DUT. Caller should handle this error to avoid false failure
1298 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001299 """
1300 result = self.run('power_supply_info').stdout.strip()
1301 info = {}
1302 device_name = None
1303 device_info = {}
1304 for line in result.split('\n'):
1305 pair = [v.strip() for v in line.split(':')]
1306 if len(pair) != 2:
1307 continue
1308 if pair[0] == 'Device':
1309 if device_name:
1310 info[device_name] = device_info
1311 device_name = pair[1]
1312 device_info = {}
1313 else:
1314 device_info[pair[0]] = pair[1]
1315 if device_name and not device_name in info:
1316 info[device_name] = device_info
1317 return info
1318
1319
1320 def get_battery_percentage(self):
1321 """Get the battery percentage.
1322
1323 @return: The percentage of battery level, value range from 0-100. Return
1324 None if the battery info cannot be retrieved.
1325 """
1326 try:
1327 info = self.get_power_supply_info()
1328 logging.info(info)
1329 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001330 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001331 return None
1332
1333
Philip Chenaf69ead2020-03-27 13:06:42 -07001334 def get_battery_state(self):
1335 """Get the battery charging state.
1336
1337 @return: A string representing the battery charging state. It can be
1338 'Charging', 'Fully charged', or 'Discharging'.
1339 """
1340 try:
1341 info = self.get_power_supply_info()
1342 logging.info(info)
1343 return info['Battery']['state']
1344 except (KeyError, ValueError, error.AutoservRunError):
1345 return None
1346
1347
Daniel Campello8ca25c22019-12-13 16:48:26 -07001348 def get_battery_display_percentage(self):
1349 """Get the battery display percentage.
1350
1351 @return: The display percentage of battery level, value range from
1352 0-100. Return None if the battery info cannot be retrieved.
1353 """
1354 try:
1355 info = self.get_power_supply_info()
1356 logging.info(info)
1357 return float(info['Battery']['display percentage'])
1358 except (KeyError, ValueError, error.AutoservRunError):
1359 return None
1360
1361
Dan Shi49ca0932014-11-14 11:22:27 -08001362 def is_ac_connected(self):
1363 """Check if the dut has power adapter connected and charging.
1364
1365 @return: True if power adapter is connected and charging.
1366 """
1367 try:
1368 info = self.get_power_supply_info()
1369 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001370 except (KeyError, error.AutoservRunError):
1371 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001372
1373
Simran Basi5e6339a2013-03-21 11:34:32 -07001374 def _cleanup_poweron(self):
1375 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001376 info = self.host_info_store.get()
1377 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001378 return
1379 logging.debug('This host has recently interacted with the RPM'
1380 ' Infrastructure. Ensuring power is on.')
1381 try:
1382 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001383 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001384 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001385 logging.error('Failed to turn Power On for this host after '
1386 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001387
1388 battery_percentage = self.get_battery_percentage()
Gregory Nisbet02273172020-07-13 09:26:17 -07001389 if (battery_percentage and
1390 battery_percentage < cros_repair.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001391 raise
1392 elif self.is_ac_connected():
1393 logging.info('The device has power adapter connected and '
1394 'charging. No need to try to turn RPM on '
1395 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001396 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001397 logging.info('Battery level is now at %s%%. The device may '
1398 'still have enough power to run test, so no '
1399 'exception will be raised.', battery_percentage)
1400
Simran Basi5e6339a2013-03-21 11:34:32 -07001401
Garry Wangad4d4fd2019-01-30 17:00:38 -08001402 def _remove_rpm_changed_tag(self):
1403 info = self.host_info_store.get()
1404 del info.attributes[self._RPM_OUTLET_CHANGED]
1405 self.host_info_store.commit(info)
1406
1407
1408 def _add_rpm_changed_tag(self):
1409 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001410 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001411 self.host_info_store.commit(info)
1412
1413
1414
beepsc87ff602013-07-31 21:53:00 -07001415 def _is_factory_image(self):
1416 """Checks if the image on the DUT is a factory image.
1417
1418 @return: True if the image on the DUT is a factory image.
1419 False otherwise.
1420 """
1421 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1422 return result.exit_status == 0
1423
1424
1425 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001426 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001427
1428 @raises: FactoryImageCheckerException for factory images, since
1429 we cannot attempt to restart ui on them.
1430 error.AutoservRunError for any other type of error that
1431 occurs while restarting ui.
1432 """
1433 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001434 raise FactoryImageCheckerException('Cannot restart ui on factory '
1435 'images')
beepsc87ff602013-07-31 21:53:00 -07001436
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001437 # TODO(jrbarnette): The command to stop/start the ui job
1438 # should live inside cros_ui, too. However that would seem
1439 # to imply interface changes to the existing start()/restart()
1440 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001441 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001442 self.run('stop ui; start ui')
1443 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001444
1445
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001446 def _start_powerd_if_needed(self):
1447 """Start powerd if it isn't already running."""
1448 self.run('start powerd', ignore_status=True)
1449
1450
xixuana3bbc422017-05-04 15:57:21 -07001451 def _get_lsb_release_content(self):
1452 """Return the content of lsb-release file of host."""
1453 return self.run(
1454 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1455
1456
Dan Shi549fb822015-03-24 18:01:11 -07001457 def get_release_version(self):
1458 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1459
1460 @returns The version string in lsb-release, under attribute
1461 CHROMEOS_RELEASE_VERSION.
1462 """
Dan Shi549fb822015-03-24 18:01:11 -07001463 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001464 lsb_release_content=self._get_lsb_release_content())
1465
1466
Don Garrettb9f35802018-01-22 18:25:40 -08001467 def get_release_builder_path(self):
1468 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1469
1470 @returns The version string in lsb-release, under attribute
1471 CHROMEOS_RELEASE_BUILDER_PATH.
1472 """
1473 return lsbrelease_utils.get_chromeos_release_builder_path(
1474 lsb_release_content=self._get_lsb_release_content())
1475
1476
xixuana3bbc422017-05-04 15:57:21 -07001477 def get_chromeos_release_milestone(self):
1478 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1479 from lsb-release.
1480
1481 @returns The version string in lsb-release, under attribute
1482 CHROMEOS_RELEASE_BUILD_TYPE.
1483 """
1484 return lsbrelease_utils.get_chromeos_release_milestone(
1485 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001486
1487
1488 def verify_cros_version_label(self):
Garry Wangd18e7b32020-08-07 18:31:44 -07001489 """Verify if host's cros-version label match the actual image in dut.
Dan Shi549fb822015-03-24 18:01:11 -07001490
Garry Wangd18e7b32020-08-07 18:31:44 -07001491 @returns True if the label match with image in dut, otherwise False
Dan Shi549fb822015-03-24 18:01:11 -07001492 """
Garry Wangd18e7b32020-08-07 18:31:44 -07001493 os_from_host = self.get_release_builder_path()
1494 info = self.host_info_store.get()
1495 os_from_label = info.get_label_value(self.VERSION_PREFIX)
1496 if not os_from_label:
1497 logging.debug('No existing %s label detected', self.VERSION_PREFIX)
1498 return True
1499
1500 # known cases where the version label will not match the
1501 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1502 # * Tests for the `arc-presubmit` append "-cheetsth" to the label.
1503 if os_from_label.endswith(provision.CHEETS_SUFFIX):
1504 logging.debug('%s label with %s suffix detected, this suffix will'
1505 ' be ignored when comparing label.',
1506 self.VERSION_PREFIX, provision.CHEETS_SUFFIX)
1507 os_from_label = os_from_label[:-len(provision.CHEETS_SUFFIX)]
1508 logging.debug('OS version from host: %s; OS verision cached in '
1509 'label: %s', os_from_host, os_from_label)
1510 return os_from_label == os_from_host
Dan Shi549fb822015-03-24 18:01:11 -07001511
1512
Laurence Goodby778c9a42017-05-24 19:24:07 -07001513 def cleanup_services(self):
1514 """Reinitializes the device for cleanup.
1515
1516 Subclasses may override this to customize the cleanup method.
1517
1518 To indicate failure of the reset, the implementation may raise
1519 any of:
1520 error.AutoservRunError
1521 error.AutotestRunError
1522 FactoryImageCheckerException
1523
1524 @raises error.AutoservRunError
1525 @raises error.AutotestRunError
1526 @raises error.FactoryImageCheckerException
1527 """
1528 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001529 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001530
1531
beepsc87ff602013-07-31 21:53:00 -07001532 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001533 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001534 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001535 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001536 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001537 except (error.AutotestRunError, error.AutoservRunError,
1538 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001539 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001540
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001541 # cleanup routines, i.e. reboot the machine.
1542 super(CrosHost, self).cleanup()
1543
Simran Basi5e6339a2013-03-21 11:34:32 -07001544 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001545 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001546 self._cleanup_poweron()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001547
1548
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001549 def reboot(self, **dargs):
1550 """
1551 This function reboots the site host. The more generic
1552 RemoteHost.reboot() performs sync and sleeps for 5
1553 seconds. This is not necessary for Chrome OS devices as the
1554 sync should be finished in a short time during the reboot
1555 command.
1556 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001557 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001558 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001559 dargs['reboot_cmd'] = ('sleep 1; '
1560 'reboot & sleep %d; '
1561 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001562 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001563 if 'fastsync' not in dargs:
1564 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001565
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001566 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001567 # Record who called us
1568 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001569 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001570 'dut_host_name' : self.hostname,
1571 'success' : True}
1572 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001573 'caller' : "%s:%s" % (orig.co_filename,
1574 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001575 'success' : True,
1576 'error' : ''}
1577
Vincent Palatin80780b22016-07-27 16:02:37 +02001578 t0 = time.time()
1579 try:
1580 super(CrosHost, self).reboot(**dargs)
1581 except Exception as e:
1582 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001583 metric_debug_fields['success'] = False
1584 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001585 raise
1586 finally:
1587 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001588 metrics.Counter(
1589 'chromeos/autotest/autoserv/reboot_count').increment(
1590 fields=metric_fields)
1591 metrics.Counter(
1592 'chromeos/autotest/autoserv/reboot_debug').increment(
1593 fields=metric_debug_fields)
1594 metrics.SecondsDistribution(
1595 'chromeos/autotest/autoserv/reboot_duration').add(
1596 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001597
1598
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001599 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001600 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001601 """
1602 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001603
1604 @param suspend_time: How long to suspend as integer seconds.
1605 @param suspend_cmd: Suspend command to execute.
1606 @param allow_early_resume: If False and if device resumes before
1607 |suspend_time|, throw an error.
1608
1609 @exception AutoservSuspendError Host resumed earlier than
1610 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001611 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001612
1613 if suspend_cmd is None:
1614 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001615 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001616 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001617 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001618 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1619 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001620
1621
Simran Basiec564392014-08-25 16:48:09 -07001622 def upstart_status(self, service_name):
1623 """Check the status of an upstart init script.
1624
1625 @param service_name: Service to look up.
1626
1627 @returns True if the service is running, False otherwise.
1628 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001629 return 'start/running' in self.run('status %s' % service_name,
1630 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001631
Tom Hughese9552342018-12-18 14:29:25 -08001632 def upstart_stop(self, service_name):
1633 """Stops an upstart job if it's running.
1634
1635 @param service_name: Service to stop
1636
1637 @returns True if service has been stopped or was already stopped
1638 False otherwise.
1639 """
1640 if not self.upstart_status(service_name):
1641 return True
1642
1643 result = self.run('stop %s' % service_name, ignore_status=True)
1644 if result.exit_status != 0:
1645 return False
1646 return True
1647
1648 def upstart_restart(self, service_name):
1649 """Restarts (or starts) an upstart job.
1650
1651 @param service_name: Service to start/restart
1652
1653 @returns True if service has been started/restarted, False otherwise.
1654 """
1655 cmd = 'start'
1656 if self.upstart_status(service_name):
1657 cmd = 'restart'
1658 cmd = cmd + ' %s' % service_name
1659 result = self.run(cmd)
1660 if result.exit_status != 0:
1661 return False
1662 return True
Simran Basiec564392014-08-25 16:48:09 -07001663
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001664 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001665 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001666
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001667 Tests for the following conditions:
1668 1. All conditions tested by the parent version of this
1669 function.
1670 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001671 3. Sufficient space in /mnt/stateful_partition/encrypted.
1672 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001673
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001674 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001675 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001676 default_kilo_inodes_required = CONFIG.get_config_value(
1677 'SERVER', 'kilo_inodes_required', type=int, default=100)
1678 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1679 kilo_inodes_required = CONFIG.get_config_value(
1680 'SERVER', 'kilo_inodes_required_%s' % board,
1681 type=int, default=default_kilo_inodes_required)
1682 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001683 self.check_diskspace(
1684 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001685 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001686 'SERVER', 'gb_diskspace_required', type=float,
1687 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001688 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1689 # Not all targets build with encrypted stateful support.
1690 if self.path_exists(encrypted_stateful_path):
1691 self.check_diskspace(
1692 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001693 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001694 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1695 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001696
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001697 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001698
beepsc87ff602013-07-31 21:53:00 -07001699 # Factory images don't run update engine,
1700 # goofy controls dbus on these DUTs.
1701 if not self._is_factory_image():
1702 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001703
1704
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001705 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1706 def wait_for_system_services(self):
1707 """Waits for system-services to be running.
1708
1709 Sometimes, update_engine will take a while to update firmware, so we
1710 should give this some time to finish. See crbug.com/765686#c38 for
1711 details.
1712 """
1713 if not self.upstart_status('system-services'):
1714 raise error.AutoservError('Chrome failed to reach login. '
1715 'System services not running.')
1716
1717
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001718 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001719 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001720 message = 'Beginning verify for host %s board %s model %s'
1721 info = self.host_info_store.get()
1722 message %= (self.hostname, info.board, info.model)
1723 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001724 try:
1725 self._repair_strategy.verify(self)
1726 except hosts.AutoservVerifyDependencyError as e:
1727 # We don't want flag a DUT as failed if only non-critical
1728 # verifier(s) failed during the repair.
1729 if e.is_critical():
1730 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001731
1732
Fang Deng96667ca2013-08-01 17:46:18 -07001733 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001734 connect_timeout=None, alive_interval=None,
1735 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001736 """Override default make_ssh_command to use options tuned for Chrome OS.
1737
1738 Tuning changes:
1739 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1740 connection failure. Consistency with remote_access.sh.
1741
Samuel Tan2ce155b2015-06-23 18:24:38 -07001742 - ServerAliveInterval=900; which causes SSH to ping connection every
1743 900 seconds. In conjunction with ServerAliveCountMax ensures
1744 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001745 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001746 the test completed successfully. Later increased from 180 seconds to
1747 900 seconds to account for tests where the DUT is suspended for
1748 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001749
1750 - ServerAliveCountMax=3; consistency with remote_access.sh.
1751
1752 - ConnectAttempts=4; reduce flakiness in connection errors;
1753 consistency with remote_access.sh.
1754
1755 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1756 Host keys change with every new installation, don't waste
1757 memory/space saving them.
1758
1759 - SSH protocol forced to 2; needed for ServerAliveInterval.
1760
1761 @param user User name to use for the ssh connection.
1762 @param port Port on the target host to use for ssh connection.
1763 @param opts Additional options to the ssh command.
1764 @param hosts_file Ignored.
1765 @param connect_timeout Ignored.
1766 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001767 @param alive_count_max Ignored.
1768 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001769 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001770 options = ' '.join([opts, '-o Protocol=2'])
1771 return super(CrosHost, self).make_ssh_command(
1772 user=user, port=port, opts=options, hosts_file='/dev/null',
1773 connect_timeout=30, alive_interval=900, alive_count_max=3,
1774 connection_attempts=4)
1775
1776
Jason Abeleb6f924f2013-11-13 16:01:54 -08001777 def syslog(self, message, tag='autotest'):
1778 """Logs a message to syslog on host.
1779
1780 @param message String message to log into syslog
1781 @param tag String tag prefix for syslog
1782
1783 """
1784 self.run('logger -t "%s" "%s"' % (tag, message))
1785
1786
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001787 def _ping_check_status(self, status):
1788 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001789
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001790 @param status Check the ping status against this value.
1791 @return True iff `status` and the result of ping are the same
1792 (i.e. both True or both False).
1793
1794 """
Abhishek Pandit-Subedi1d081e22020-09-22 17:13:46 +00001795 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001796 return not (status ^ (ping_val == 0))
1797
1798 def _ping_wait_for_status(self, status, timeout):
1799 """Wait for the host to have a given status (UP or DOWN).
1800
1801 Status is checked by polling. Polling will not last longer
1802 than the number of seconds in `timeout`. The polling
1803 interval will be long enough that only approximately
1804 _PING_WAIT_COUNT polling cycles will be executed, subject
1805 to a maximum interval of about one minute.
1806
1807 @param status Waiting will stop immediately if `ping` of the
1808 host returns this status.
1809 @param timeout Poll for at most this many seconds.
1810 @return True iff the host status from `ping` matched the
1811 requested status at the time of return.
1812
1813 """
1814 # _ping_check_status() takes about 1 second, hence the
1815 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001816 # FIXME: if the ping command errors then _ping_check_status()
1817 # returns instantly. If timeout is also smaller than twice
1818 # _PING_WAIT_COUNT then the while loop below forks many
1819 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1820 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1821 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001822 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1823 end_time = time.time() + timeout
1824 while time.time() <= end_time:
1825 if self._ping_check_status(status):
1826 return True
1827 if poll_interval > 0:
1828 time.sleep(poll_interval)
1829
1830 # The last thing we did was sleep(poll_interval), so it may
1831 # have been too long since the last `ping`. Check one more
1832 # time, just to be sure.
1833 return self._ping_check_status(status)
1834
1835 def ping_wait_up(self, timeout):
1836 """Wait for the host to respond to `ping`.
1837
1838 N.B. This method is not a reliable substitute for
1839 `wait_up()`, because a host that responds to ping will not
1840 necessarily respond to ssh. This method should only be used
1841 if the target DUT can be considered functional even if it
1842 can't be reached via ssh.
1843
1844 @param timeout Minimum time to allow before declaring the
1845 host to be non-responsive.
1846 @return True iff the host answered to ping before the timeout.
1847
1848 """
1849 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001850
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001851 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001852 """Wait until the host no longer responds to `ping`.
1853
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001854 This function can be used as a slightly faster version of
1855 `wait_down()`, by avoiding potentially long ssh timeouts.
1856
1857 @param timeout Minimum time to allow for the host to become
1858 non-responsive.
1859 @return True iff the host quit answering ping before the
1860 timeout.
1861
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001862 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001863 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001864
Anand K Mistry50f218e2020-07-31 14:50:15 +10001865 def _is_host_port_forwarded(self):
Garry Wanga2e78172020-09-09 23:49:07 -07001866 """Checks if the dut is connected over port forwarding.
Anand K Mistry50f218e2020-07-31 14:50:15 +10001867
1868 N.B. This method does not detect all situations where port forwarding is
1869 occurring. Namely, running autotest on the dut may result in a
1870 false-positive, and port forwarding using a different machine on the
1871 same network will be a false-negative.
1872
1873 @return True if the dut is connected over port forwarding
1874 False otherwise
1875 """
Garry Wanga2e78172020-09-09 23:49:07 -07001876 is_localhost = self.hostname in ['localhost', '127.0.0.1']
1877 is_forwarded = is_localhost and not self.is_default_port
1878 if is_forwarded:
1879 logging.info('Detected DUT connected by port forwarding')
1880 return is_forwarded
Anand K Mistry50f218e2020-07-31 14:50:15 +10001881
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001882 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001883 """Wait for the client to enter low-power sleep mode.
1884
1885 The test for "is asleep" can't distinguish a system that is
1886 powered off; to confirm that the unit was asleep, it is
1887 necessary to force resume, and then call
1888 `test_wait_for_resume()`.
1889
1890 This function is expected to be called from a test as part
1891 of a sequence like the following:
1892
1893 ~~~~~~~~
1894 boot_id = host.get_boot_id()
1895 # trigger sleep on the host
1896 host.test_wait_for_sleep()
1897 # trigger resume on the host
1898 host.test_wait_for_resume(boot_id)
1899 ~~~~~~~~
1900
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001901 @param sleep_timeout time limit in seconds to allow the host sleep.
1902
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001903 @exception TestFail The host did not go to sleep within
1904 the allowed time.
1905 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001906 if sleep_timeout is None:
1907 sleep_timeout = self.SLEEP_TIMEOUT
1908
Anand K Mistry50f218e2020-07-31 14:50:15 +10001909 # If the dut is accessed over SSH port-forwarding, `ping` is not useful
1910 # for detecting the dut is down since a ping to localhost will always
1911 # succeed. In this case, fall back to wait_down() which uses SSH.
1912 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07001913 success = self.wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10001914 else:
Garry Wanga2e78172020-09-09 23:49:07 -07001915 success = self.ping_wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10001916
1917 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001918 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001919 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001920
1921
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001922 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001923 """Wait for the client to resume from low-power sleep mode.
1924
1925 The `old_boot_id` parameter should be the value from
1926 `get_boot_id()` obtained prior to entering sleep mode. A
1927 `TestFail` exception is raised if the boot id changes.
1928
1929 See @ref test_wait_for_sleep for more on this function's
1930 usage.
1931
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001932 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001933 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001934 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001935
1936 @exception TestFail The host did not respond within the
1937 allowed time.
1938 @exception TestFail The host responded, but the boot id test
1939 indicated a reboot rather than a sleep
1940 cycle.
1941 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001942 if resume_timeout is None:
1943 resume_timeout = self.RESUME_TIMEOUT
1944
1945 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001946 raise error.TestFail(
1947 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001948 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001949 else:
1950 new_boot_id = self.get_boot_id()
1951 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001952 logging.error('client rebooted (old boot %s, new boot %s)',
1953 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001954 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001955 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001956
1957
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001958 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001959 """Wait for the client to shut down.
1960
1961 The test for "has shut down" can't distinguish a system that
1962 is merely asleep; to confirm that the unit was down, it is
1963 necessary to force boot, and then call test_wait_for_boot().
1964
1965 This function is expected to be called from a test as part
1966 of a sequence like the following:
1967
1968 ~~~~~~~~
1969 boot_id = host.get_boot_id()
1970 # trigger shutdown on the host
1971 host.test_wait_for_shutdown()
1972 # trigger boot on the host
1973 host.test_wait_for_boot(boot_id)
1974 ~~~~~~~~
1975
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001976 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001977 @exception TestFail The host did not shut down within the
1978 allowed time.
1979 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001980 if shutdown_timeout is None:
1981 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1982
Anand K Mistry50f218e2020-07-31 14:50:15 +10001983 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07001984 success = self.wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10001985 else:
Garry Wanga2e78172020-09-09 23:49:07 -07001986 success = self.ping_wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10001987
1988 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001989 raise error.TestFail(
1990 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001991 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001992
1993
1994 def test_wait_for_boot(self, old_boot_id=None):
1995 """Wait for the client to boot from cold power.
1996
1997 The `old_boot_id` parameter should be the value from
1998 `get_boot_id()` obtained prior to shutting down. A
1999 `TestFail` exception is raised if the boot id does not
2000 change. The boot id test is omitted if `old_boot_id` is not
2001 specified.
2002
2003 See @ref test_wait_for_shutdown for more on this function's
2004 usage.
2005
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002006 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002007 shut down.
2008
2009 @exception TestFail The host did not respond within the
2010 allowed time.
2011 @exception TestFail The host responded, but the boot id test
2012 indicated that there was no reboot.
2013 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002014 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002015 raise error.TestFail(
2016 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002017 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002018 elif old_boot_id:
2019 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002020 logging.error('client not rebooted (boot %s)',
2021 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002022 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002023 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07002024
2025
2026 @staticmethod
2027 def check_for_rpm_support(hostname):
2028 """For a given hostname, return whether or not it is powered by an RPM.
2029
Simran Basi1df55112013-09-06 11:25:09 -07002030 @param hostname: hostname to check for rpm support.
2031
Simran Basid5e5e272012-09-24 15:23:59 -07002032 @return None if this host does not follows the defined naming format
2033 for RPM powered DUT's in the lab. If it does follow the format,
2034 it returns a regular expression MatchObject instead.
2035 """
Fang Dengbaff9082015-01-06 13:46:15 -08002036 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002037
2038
2039 def has_power(self):
2040 """For this host, return whether or not it is powered by an RPM.
2041
2042 @return True if this host is in the CROS lab and follows the defined
2043 naming format.
2044 """
Fang Deng0ca40e22013-08-27 17:47:44 -07002045 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002046
2047
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002048 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07002049 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002050
2051 @param state Specifies which power state to set to DUT
2052 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002053 use. By default "RPM" or "CCD" will be used based
2054 on servo type. Valid values from
2055 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002056
2057 """
2058 ACCEPTABLE_STATES = ['ON', 'OFF']
2059
Garry Wang5e5538a2019-04-08 15:36:18 -07002060 if not power_method:
2061 power_method = self.get_default_power_method()
2062
2063 state = state.upper()
2064 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002065 raise error.TestError('State must be one of: %s.'
2066 % (ACCEPTABLE_STATES,))
2067
2068 if power_method == self.POWER_CONTROL_SERVO:
2069 logging.info('Setting servo port J10 to %s', state)
2070 self.servo.set('prtctl3_pwren', state.lower())
2071 time.sleep(self._USB_POWER_TIMEOUT)
2072 elif power_method == self.POWER_CONTROL_MANUAL:
2073 logging.info('You have %d seconds to set the AC power to %s.',
2074 self._POWER_CYCLE_TIMEOUT, state)
2075 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07002076 elif power_method == self.POWER_CONTROL_CCD:
2077 servo_role = 'src' if state == 'ON' else 'snk'
2078 logging.info('servo ccd power pass through detected,'
2079 ' changing servo_role to %s.', servo_role)
2080 self.servo.set_servo_v4_role(servo_role)
2081 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07002082 # Make sure we don't leave DUT with no power(servo_role=snk)
2083 # when DUT is not pingable, as we raise a exception here
2084 # that may break a power cycle in the middle.
2085 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07002086 raise error.AutoservError(
2087 'DUT failed to regain network connection after %d seconds.'
2088 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002089 else:
2090 if not self.has_power():
2091 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002092 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07002093 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07002094
2095
Garry Wang5e5538a2019-04-08 15:36:18 -07002096 def power_off(self, power_method=None):
2097 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002098
2099 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002100 use. By default "RPM" or "CCD" will be used based
2101 on servo type. Valid values from
2102 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002103
2104 """
Derek Beckett2fb9aa02020-08-12 15:31:02 -07002105 self._sync_if_up()
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002106 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002107
Derek Beckett2fb9aa02020-08-12 15:31:02 -07002108 def _check_supported(self):
2109 """Throw an error if dts mode control is not supported."""
2110 if not self.servo_pwr_supported:
2111 raise error.TestFail('power_state controls not supported')
2112
2113 def _sync_if_up(self):
2114 """Run sync on the DUT and wait for completion if the DUT is up.
2115
2116 Additionally, try to sync and ignore status if its not up.
2117
2118 Useful prior to reboots to ensure files are written to disc.
2119
2120 """
2121 if self.is_up_fast():
2122 self.run("sync")
2123 return
2124 # If it is not up, attempt to sync in the rare event the DUT is up but
2125 # doesn't respond to a ping. Ignore any errors.
2126 try:
2127 self.run("sync", ignore_status=True, timeout=1)
2128 except Exception:
2129 pass
2130
2131 def power_off_via_servo(self):
2132 """Force the DUT to power off.
2133
2134 The DUT is guaranteed to be off at the end of this call,
2135 regardless of its previous state, provided that there is
2136 working EC and boot firmware. There is no requirement for
2137 working OS software.
2138
2139 """
2140 self._check_supported()
2141 self._sync_if_up()
2142 self.servo.set_nocheck('power_state', 'off')
2143
2144 def power_on_via_servo(self, rec_mode='on'):
2145 """Force the DUT to power on.
2146
2147 Prior to calling this function, the DUT must be powered off,
2148 e.g. with a call to `power_off()`.
2149
2150 At power on, recovery mode is set as specified by the
2151 corresponding argument. When booting with recovery mode on, it
2152 is the caller's responsibility to unplug/plug in a bootable
2153 external storage device.
2154
2155 If the DUT requires a delay after powering on but before
2156 processing inputs such as USB stick insertion, the delay is
2157 handled by this method; the caller is not responsible for such
2158 delays.
2159
2160 @param rec_mode Setting of recovery mode to be applied at
2161 power on. default: REC_OFF aka 'off'
2162
2163 """
2164 self._check_supported()
2165 self.servo.set_nocheck('power_state', rec_mode)
2166
2167 def reset_via_servo(self):
2168 """Force the DUT to reset.
2169
2170 The DUT is guaranteed to be on at the end of this call,
2171 regardless of its previous state, provided that there is
2172 working OS software. This also guarantees that the EC has
2173 been restarted.
2174
2175 """
2176 self._check_supported()
2177 self._sync_if_up()
2178 self.servo.set_nocheck('power_state', 'reset')
2179
Simran Basid5e5e272012-09-24 15:23:59 -07002180
Garry Wang5e5538a2019-04-08 15:36:18 -07002181 def power_on(self, power_method=None):
2182 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002183
2184 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002185 use. By default "RPM" or "CCD" will be used based
2186 on servo type. Valid values from
2187 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002188
2189 """
2190 self._set_power('ON', power_method)
2191
2192
Garry Wang5e5538a2019-04-08 15:36:18 -07002193 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002194 """Cycle power to this host by turning it OFF, then ON.
2195
2196 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002197 use. By default "RPM" or "CCD" will be used based
2198 on servo type. Valid values from
2199 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002200
2201 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002202 if not power_method:
2203 power_method = self.get_default_power_method()
2204
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002205 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002206 self.POWER_CONTROL_MANUAL,
2207 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002208 self.power_off(power_method=power_method)
2209 time.sleep(self._POWER_CYCLE_TIMEOUT)
2210 self.power_on(power_method=power_method)
2211 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002212 self._add_rpm_changed_tag()
2213 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002214
2215
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002216 def get_platform_from_fwid(self):
2217 """Determine the platform from the crossystem fwid.
2218
2219 @returns a string representing this host's platform.
2220 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002221 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002222 crossystem = utils.Crossystem(self)
2223 crossystem.init()
2224 # Extract fwid value and use the leading part as the platform id.
2225 # fwid generally follow the format of {platform}.{firmware version}
2226 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2227 platform = crossystem.fwid().split('.')[0].lower()
2228 # Newer platforms start with 'Google_' while the older ones do not.
2229 return platform.replace('google_', '')
2230
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002231
Simran Basic6f1f7a2012-10-16 10:47:46 -07002232 def get_platform(self):
2233 """Determine the correct platform label for this host.
2234
2235 @returns a string representing this host's platform.
2236 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002237 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2238 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002239 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002240 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002241 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002242 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002243
2244
Greg Edelstona7b05d12020-04-01 16:00:51 -06002245 def get_model_from_cros_config(self):
2246 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002247
Greg Edelstona7b05d12020-04-01 16:00:51 -06002248 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002249 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002250 return cros_config.call_cros_config_get_output('/ name',
2251 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002252
2253
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002254 def get_architecture(self):
2255 """Determine the correct architecture label for this host.
2256
2257 @returns a string representing this host's architecture.
2258 """
2259 crossystem = utils.Crossystem(self)
2260 crossystem.init()
2261 return crossystem.arch()
2262
2263
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002264 def get_chrome_version(self):
2265 """Gets the Chrome version number and milestone as strings.
2266
2267 Invokes "chrome --version" to get the version number and milestone.
2268
2269 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2270 current Chrome version number as a string (in the form "W.X.Y.Z")
2271 and "milestone" is the first component of the version number
2272 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2273 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2274 of "chrome --version" and the milestone will be the empty string.
2275
2276 """
MK Ryu35d661e2014-09-25 17:44:10 -07002277 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002278 return utils.parse_chrome_version(version_string)
2279
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002280
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002281 def get_ec_version(self):
2282 """Get the ec version as strings.
2283
2284 @returns a string representing this host's ec version.
2285 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002286 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002287 result = self.run(command, ignore_status=True)
2288 if result.exit_status != 0:
2289 return ''
2290 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002291
2292
2293 def get_firmware_version(self):
2294 """Get the firmware version as strings.
2295
2296 @returns a string representing this host's firmware version.
2297 """
2298 crossystem = utils.Crossystem(self)
2299 crossystem.init()
2300 return crossystem.fwid()
2301
2302
2303 def get_hardware_revision(self):
2304 """Get the hardware revision as strings.
2305
2306 @returns a string representing this host's hardware revision.
2307 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002308 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002309 result = self.run(command, ignore_status=True)
2310 if result.exit_status != 0:
2311 return ''
2312 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002313
2314
2315 def get_kernel_version(self):
2316 """Get the kernel version as strings.
2317
2318 @returns a string representing this host's kernel version.
2319 """
2320 return self.run('uname -r').stdout.strip()
2321
2322
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002323 def get_cpu_name(self):
2324 """Get the cpu name as strings.
2325
2326 @returns a string representing this host's cpu name.
2327 """
2328
2329 # Try get cpu name from device tree first
2330 if self.path_exists('/proc/device-tree/compatible'):
2331 command = ' | '.join(
2332 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2333 'tail -1'])
2334 return self.run(command).stdout.strip().replace(',', ' ')
2335
2336 # Get cpu name from uname -p
2337 command = 'uname -p'
2338 ret = self.run(command).stdout.strip()
2339
2340 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2341 # Try get cpu name from /proc/cpuinfo instead
2342 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2343 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2344 self = self.run(command).stdout.strip()
2345
2346 # Remove bloat from CPU name, for example
2347 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2348 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2349 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2350 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2351 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2352 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2353
2354
2355 def get_screen_resolution(self):
2356 """Get the screen(s) resolution as strings.
2357 In case of more than 1 monitor, return resolution for each monitor
2358 separate with plus sign.
2359
2360 @returns a string representing this host's screen(s) resolution.
2361 """
2362 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2363 ret = self.run(command, ignore_status=True)
2364 # We might have Chromebox without a screen
2365 if ret.exit_status != 0:
2366 return ''
2367 return ret.stdout.strip().replace('\n', '+')
2368
2369
2370 def get_mem_total_gb(self):
2371 """Get total memory available in the system in GiB (2^20).
2372
2373 @returns an integer representing total memory
2374 """
2375 mem_total_kb = self.read_from_meminfo('MemTotal')
2376 kb_in_gb = float(2 ** 20)
2377 return int(round(mem_total_kb / kb_in_gb))
2378
2379
2380 def get_disk_size_gb(self):
2381 """Get size of disk in GB (10^9)
2382
2383 @returns an integer representing size of disk, 0 in Error Case
2384 """
2385 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2386 result = self.run(command, ignore_status=True)
2387 if result.exit_status != 0:
2388 return 0
2389 _, _, block, _ = re.split(r' +', result.stdout.strip())
2390 byte_per_block = 1024.0
2391 disk_kb_in_gb = 1e9
2392 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2393
2394
2395 def get_battery_size(self):
2396 """Get size of battery in Watt-hour via sysfs
2397
2398 This method assumes that battery support voltage_min_design and
2399 charge_full_design sysfs.
2400
2401 @returns a float representing Battery size, 0 if error.
2402 """
2403 # sysfs report data in micro scale
2404 battery_scale = 1e6
2405
2406 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2407 result = self.run(command, ignore_status=True)
2408 if result.exit_status != 0:
2409 return 0
2410 voltage = float(result.stdout.strip()) / battery_scale
2411
2412 command = 'cat /sys/class/power_supply/*/charge_full_design'
2413 result = self.run(command, ignore_status=True)
2414 if result.exit_status != 0:
2415 return 0
2416 amphereHour = float(result.stdout.strip()) / battery_scale
2417
2418 return voltage * amphereHour
2419
2420
2421 def get_low_battery_shutdown_percent(self):
2422 """Get the percent-based low-battery shutdown threshold.
2423
2424 @returns a float representing low-battery shutdown percent, 0 if error.
2425 """
2426 ret = 0.0
2427 try:
2428 command = 'check_powerd_config --low_battery_shutdown_percent'
2429 ret = float(self.run(command).stdout)
2430 except error.CmdError:
2431 logging.debug("Can't run %s", command)
2432 except ValueError:
2433 logging.debug("Didn't get number from %s", command)
2434
2435 return ret
2436
2437
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002438 def has_hammer(self):
2439 """Check whether DUT has hammer device or not.
2440
2441 @returns boolean whether device has hammer or not
2442 """
2443 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2444 return self.run(command, ignore_status=True).exit_status == 0
2445
2446
Niranjan Kumar34618872017-05-31 12:57:09 -07002447 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002448 """Returns True if the specified switch was provided to Chrome.
2449
2450 @param switch The chrome switch to search for.
2451 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002452
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002453 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2454 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002455
2456
2457 def oobe_triggers_update(self):
2458 """Returns True if this host has an OOBE flow during which
2459 it will perform an update check and perhaps an update.
2460 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2461 As more such flows are developed, code handling them needs
2462 to be added here.
2463
2464 @return Boolean indicating whether this host's OOBE triggers an update.
2465 """
2466 return self.is_chrome_switch_present(
2467 '--enterprise-enable-zero-touch-enrollment=hands-off')
2468
2469
Kevin Chenga2619dc2016-03-28 11:42:08 -07002470 # TODO(kevcheng): change this to just return the board without the
2471 # 'board:' prefix and fix up all the callers. Also look into removing the
2472 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002473 def get_board(self):
2474 """Determine the correct board label for this host.
2475
2476 @returns a string representing this host's board.
2477 """
2478 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2479 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002480 return (ds_constants.BOARD_PREFIX +
2481 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002482
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002483 def get_channel(self):
2484 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002485
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002486 @returns: a string represeting this host's build channel.
2487 (stable, dev, beta). None on fail.
2488 """
2489 return lsbrelease_utils.get_chromeos_channel(
2490 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002491
Kevin Chenga328da62016-03-31 10:49:04 -07002492 def get_power_supply(self):
2493 """
2494 Determine what type of power supply the host has
2495
2496 @returns a string representing this host's power supply.
2497 'power:battery' when the device has a battery intended for
2498 extended use
2499 'power:AC_primary' when the device has a battery not intended
2500 for extended use (for moving the machine, etc)
2501 'power:AC_only' when the device has no battery at all.
2502 """
2503 psu = self.run(command='mosys psu type', ignore_status=True)
2504 if psu.exit_status:
2505 # The psu command for mosys is not included for all platforms. The
2506 # assumption is that the device will have a battery if the command
2507 # is not found.
2508 return 'power:battery'
2509
2510 psu_str = psu.stdout.strip()
2511 if psu_str == 'unknown':
2512 return None
2513
2514 return 'power:%s' % psu_str
2515
2516
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002517 def has_battery(self):
2518 """Determine if DUT has a battery.
2519
2520 Returns:
2521 Boolean, False if known not to have battery, True otherwise.
2522 """
2523 rv = True
2524 power_supply = self.get_power_supply()
2525 if power_supply == 'power:battery':
2526 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2527 board_type = self.get_board_type()
2528 if board_type in _NO_BATTERY_BOARD_TYPE:
2529 logging.warn('Do NOT believe type %s has battery. '
2530 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002531 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002532 ignore_status=True)
2533 logging.debug(psu)
2534 rv = False
2535 elif power_supply == 'power:AC_only':
2536 rv = False
2537
2538 return rv
2539
2540
Kevin Chenga328da62016-03-31 10:49:04 -07002541 def get_servo(self):
2542 """Determine if the host has a servo attached.
2543
2544 If the host has a working servo attached, it should have a servo label.
2545
2546 @return: string 'servo' if the host has servo attached. Otherwise,
2547 returns None.
2548 """
2549 return 'servo' if self._servo_host else None
2550
2551
Kevin Chenga328da62016-03-31 10:49:04 -07002552 def has_internal_display(self):
2553 """Determine if the device under test is equipped with an internal
2554 display.
2555
2556 @return: 'internal_display' if one is present; None otherwise.
2557 """
2558 from autotest_lib.client.cros.graphics import graphics_utils
2559 from autotest_lib.client.common_lib import utils as common_utils
2560
2561 def __system_output(cmd):
2562 return self.run(cmd).stdout
2563
2564 def __read_file(remote_path):
2565 return self.run('cat %s' % remote_path).stdout
2566
2567 # Hijack the necessary client functions so that we can take advantage
2568 # of the client lib here.
2569 # FIXME: find a less hacky way than this
2570 original_system_output = utils.system_output
2571 original_read_file = common_utils.read_file
2572 utils.system_output = __system_output
2573 common_utils.read_file = __read_file
2574 try:
2575 return ('internal_display' if graphics_utils.has_internal_display()
2576 else None)
2577 finally:
2578 utils.system_output = original_system_output
2579 common_utils.read_file = original_read_file
2580
2581
Dan Shi85276d42014-04-08 22:11:45 -07002582 def is_boot_from_usb(self):
2583 """Check if DUT is boot from USB.
2584
2585 @return: True if DUT is boot from usb.
2586 """
2587 device = self.run('rootdev -s -d').stdout.strip()
2588 removable = int(self.run('cat /sys/block/%s/removable' %
2589 os.path.basename(device)).stdout.strip())
2590 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002591
2592
2593 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002594 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002595
2596 @param key: meminfo requested
2597
2598 @return the memory value as a string
2599
2600 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002601 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2602 logging.debug('%s', meminfo)
2603 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002604
2605
Rohit Makasana98e696f2016-06-03 18:48:10 -07002606 def get_cpu_arch(self):
2607 """Returns CPU arch of the device.
2608
2609 @return CPU architecture of the DUT.
2610 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002611 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002612 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2613 ignore_status=True).stdout:
2614 return 'x86_64'
2615 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2616 ignore_status=True).stdout:
2617 return 'arm'
2618 return 'i386'
2619
2620
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002621 def get_board_type(self):
2622 """
2623 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002624 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2625
2626 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002627 """
Danny Chan471a8d12015-08-18 14:57:41 -07002628 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2629 ignore_status=True).stdout
2630 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002631 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002632 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002633
2634
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002635 def get_arc_version(self):
2636 """Return ARC version installed on the DUT.
2637
2638 @returns ARC version as string if the CrOS build has ARC, else None.
2639 """
2640 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2641 ignore_status=True).stdout
2642 if arc_version:
2643 return arc_version.split('=')[-1].strip()
2644 return None
2645
2646
Gilad Arnolda76bef02015-09-29 13:55:15 -07002647 def get_os_type(self):
2648 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002649
2650
Kevin Chenga2619dc2016-03-28 11:42:08 -07002651 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002652 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002653 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002654
2655
2656 def get_default_power_method(self):
2657 """
2658 Get the default power method for power_on/off/cycle() methods.
2659 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2660 """
2661 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002662 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002663 if self.servo and self.servo.supports_built_in_pd_control():
2664 self._default_power_method = self.POWER_CONTROL_CCD
2665 else:
2666 logging.debug('Either servo is unitialized or the servo '
2667 'setup does not support pd controls. Falling '
2668 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002669 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002670
2671
2672 def find_usb_devices(self, idVendor, idProduct):
2673 """
2674 Get usb device sysfs name for specific device.
2675
2676 @param idVendor Vendor ID to search in sysfs directory.
2677 @param idProduct Product ID to search in sysfs directory.
2678
2679 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2680 """
2681 # Look for matching file and cut at position 7 to get dir name.
2682 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2683
2684 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2685 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2686
2687 # Use uniq -d to print duplicate line from both command
2688 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2689
2690 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2691
2692
2693 def bind_usb_device(self, usb_node):
2694 """
2695 Bind usb device
2696
2697 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2698 """
2699 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2700 self.run(cmd, ignore_status=True)
2701
2702
2703 def unbind_usb_device(self, usb_node):
2704 """
2705 Unbind usb device
2706
2707 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2708 """
2709 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2710 self.run(cmd, ignore_status=True)
2711
2712
2713 def get_wlan_ip(self):
2714 """
2715 Get ip address of wlan interface.
2716
2717 @return ip address of wlan or empty string if wlan is not connected.
2718 """
2719 cmds = [
2720 'iw dev', # List wlan physical device
2721 'grep Interface', # Grep only interface name
2722 'cut -f 2 -d" "', # Cut the name part
2723 'xargs ifconfig', # Feed it to ifconfig to get ip
2724 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2725 'cut -f 2 -d " "' # Cut the ip part
2726 ]
2727 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002728
2729 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2730 """
2731 Connect to wifi network
2732
2733 @param ssid SSID of the wifi network.
2734 @param passphrase Passphrase of the wifi network. None if not existed.
2735 @param security Security of the wifi network. Default to "psk" if
2736 passphase is given without security. Possible values
2737 are "none", "psk", "802_1x".
2738
2739 @return True if succeed, False if not.
2740 """
2741 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2742 if passphrase:
2743 cmd += ' ' + passphrase
2744 if security:
2745 cmd += ' ' + security
2746 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002747
2748 def get_device_repair_state(self):
2749 """Get device repair state"""
2750 return self._device_repair_state
2751
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002752 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002753 """Set device repair state.
2754
2755 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002756 file in result directory. The file will be read by Lucifer. The
2757 file will not be created if result directory not specified.
2758
2759 @params state: The new state for the device.
2760 @params resultdir: The path to result directory. If path not provided
2761 will be attempt to get retrieve it from job
2762 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002763 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002764 resultdir = resultdir or getattr(self.job, 'resultdir', '')
2765 if resultdir:
2766 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07002767 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002768 logging.info('Set device state as %s. '
2769 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07002770 else:
2771 logging.debug('Cannot write the device state due missing info '
2772 'about result dir.')
2773 self._device_repair_state = state
2774
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002775 def set_device_needs_replacement(self, resultdir=None):
2776 """Set device as required replacement.
2777
2778 @params resultdir: The path to result directory. If path not provided
2779 will be attempt to get retrieve it from job
2780 if present.
2781 """
2782 self.set_device_repair_state(
2783 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
2784 resultdir=resultdir)
2785
2786 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002787 """Check if device require manual attention to be fixed.
2788
2789 The state 'needs_manual_repair' can be set when auto repair cannot
2790 fix the device due hardware or cable issues.
2791 """
2792 # ignore the logic if state present
2793 # state can be set by any cros repair actions
2794 if self.get_device_repair_state():
2795 return
2796
2797 # set need manual attention if servo has hardware issue
2798 servo_state_required_manual_fix = [
2799 servo_constants.SERVO_STATE_NOT_CONNECTED,
2800 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
2801 servo_constants.SERVO_STATE_LID_OPEN_FAILED,
2802 servo_constants.SERVO_STATE_BAD_RIBBON_CABLE,
2803 servo_constants.SERVO_STATE_EC_BROKEN,
2804 ]
2805 if self.get_servo_state() in servo_state_required_manual_fix:
Otabek Kasimovde8eea32020-07-01 12:12:22 -07002806 data = {'host': self.hostname,
Otabek Kasimov832d9162020-07-27 19:24:57 -07002807 'state': cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR}
Otabek Kasimov6825b762020-06-23 23:42:44 -07002808 metrics.Counter(
2809 'chromeos/autotest/repair/special_dut_state'
2810 ).increment(fields=data)
2811 # TODO (otabek) unblock when be sure that we do not have flakiness
Otabek Kasimov832d9162020-07-27 19:24:57 -07002812 # self.set_device_repair_state(
2813 # cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)
Otabek Kasimov42506d02020-07-29 14:44:57 -07002814
2815 def is_file_system_writable(self, testdirs=None):
2816 """Check is the file systems are writable.
2817
2818 The standard linux response to certain unexpected file system errors
2819 (including hardware errors in block devices) is to change the file
2820 system status to read-only. This checks that that hasn't happened.
2821
2822 @param testdirs: List of directories to check. If no data provided
2823 then '/mnt/stateful_partition' and '/var/tmp'
2824 directories will be checked.
2825
2826 @returns boolean whether file-system writable.
2827 """
2828 def _check_dir(testdir):
2829 # check if we can create a file
2830 filename = os.path.join(testdir, 'writable_my_test_file')
2831 command = 'touch %s && rm %s' % (filename, filename)
2832 rv = self.run(command=command,
2833 timeout=30,
2834 ignore_status=True)
2835 is_writable = rv.exit_status == 0
2836 if not is_writable:
2837 logging.info('Cannot create a file in "%s"!'
2838 ' Probably the FS is read-only', testdir)
2839 logging.info("FileSystem is not writable!")
2840 return False
2841 return True
2842
2843 if not testdirs or len(testdirs) == 0:
2844 # N.B. Order matters here: Encrypted stateful is loop-mounted
2845 # from a file in unencrypted stateful, so we don't test for
2846 # errors in encrypted stateful if unencrypted fails.
2847 testdirs = ['/mnt/stateful_partition', '/var/tmp']
2848
2849 for dir in testdirs:
2850 # loop will be stopped if any directory fill fail the check
2851 try:
2852 if not _check_dir(dir):
2853 return False
2854 except Exception as e:
2855 # here expected only timeout error, all other will
2856 # be catch by 'ignore_status=True'
2857 logging.debug('Fail to check %s to write in it', dir)
2858 return False
2859 return True
Garry Wang1a493d82020-08-31 21:01:19 -07002860
2861 def setup_device_health_profile(self):
2862 """Setup device health profile for repair/provision task to consume.
2863 """
Garry Wanga2e78172020-09-09 23:49:07 -07002864 if self.health_profile:
2865 logging.info('Device health profile has already been initialized.')
Garry Wang1a493d82020-08-31 21:01:19 -07002866 if not self._servo_host:
2867 logging.info('Servohost is not instantiated, skip device'
2868 ' health profile setup...')
2869 return
2870 # Also skip setup health profile if it's a task runs locally.
2871 if self._servo_host.is_localhost():
2872 logging.info('Servohost is a localhost, skip device'
2873 ' health profile setup...')
2874 return
2875 try:
2876 self.health_profile = device_health_profile.DeviceHealthProfile(
2877 self, self._servo_host)
2878 except Exception as e:
2879 logging.warning('Failed to setup device health profile; %s', e)
Garry Wanga2e78172020-09-09 23:49:07 -07002880
2881 def set_health_profile_dut_state(self, state):
2882 if not self.health_profile:
2883 logging.debug('Device health profile is not initialized, skip'
2884 ' set dut state.')
2885 return
2886 reset_counters = state in profile_constants.STATES_NEED_RESET_COUNTER
2887 self.health_profile.update_dut_state(state, reset_counters)
Garry Wang53fc8f32020-09-18 13:30:08 -07002888
2889 def require_snk_mode_in_recovery(self):
2890 """Check whether we need to switch servo_v4 role to snk when
2891 booting into recovery mode. (See crbug.com/1129165)
2892 """
2893 info = self.host_info_store.get()
2894 if info.get_label_value('power') != 'battery':
2895 logging.info(
2896 '%s does not has battery, snk mode is not needed'
2897 ' for recovery.', self.hostname)
2898 return False
2899 if not self.servo.supports_built_in_pd_control():
2900 logging.info('Power delivery is not supported on this servo, snk'
2901 ' mode is not needed for recovery.')
2902 return False
2903 try:
2904 #TODO(xianuowang@) move MIN_BATTERY_LEVEL to cros_constant
2905 battery_percent = self.servo.get('battery_charge_percent')
2906 if battery_percent < cros_repair.MIN_BATTERY_LEVEL:
2907 logging.info(
2908 'Current battery level %s%% below %s%% threshold, we'
2909 ' will attempt to boot host in recovery mode without'
2910 ' changing servo to snk mode. Please note the host may'
2911 ' not able to see usb drive in recovery mode later due'
2912 ' to servo not in snk mode.', battery_percent,
2913 cros_repair.MIN_BATTERY_LEVEL)
2914 return False
2915 except Exception as e:
2916 logging.info(
2917 'Unexpected error occurred when getting'
2918 ' battery_charge_percent from servo; %s', str(e))
2919 return False
2920 return True