blob: 2870a5aa5e109994178a6c99613bce59c3e04795 [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
Jeremy Bettis067eb4a2022-03-18 11:15:21 -06009from io import StringIO
10import json
Derek Beckettf73baca2020-08-19 15:08:47 -070011
J. Richard Barnette1d78b012012-05-15 13:56:30 -070012import logging
Dan Shi0f466e82013-02-22 15:44:58 -080013import os
Simran Basid5e5e272012-09-24 15:23:59 -070014import re
Vincent Palatindf2372c2016-10-07 17:03:00 +020015import sys
Otabek Kasimovb7cb8422020-12-23 02:38:32 -080016import six
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070017import time
18
mussa584b4462014-06-20 15:13:28 -070019import common
J. Richard Barnette45e93de2012-04-11 17:24:15 -070020from autotest_lib.client.bin import utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070021from autotest_lib.client.common_lib import autotemp
Richard Barnette0c73ffc2012-11-19 15:21:18 -080022from autotest_lib.client.common_lib import error
23from autotest_lib.client.common_lib import global_config
J. Richard Barnette91137f02016-03-10 16:52:26 -080024from autotest_lib.client.common_lib import hosts
Dan Shi549fb822015-03-24 18:01:11 -070025from autotest_lib.client.common_lib import lsbrelease_utils
Otabek Kasimov6825b762020-06-23 23:42:44 -070026from autotest_lib.client.common_lib import utils as common_utils
Greg Edelstona7b05d12020-04-01 16:00:51 -060027from autotest_lib.client.common_lib.cros import cros_config
Richard Barnette03a0c132012-11-05 12:40:35 -080028from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070029from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000030from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080031from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080032from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070033from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070034from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050035from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070036from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Garry Wang1a493d82020-08-31 21:01:19 -070037from autotest_lib.server.cros.device_health_profile import device_health_profile
Garry Wanga2e78172020-09-09 23:49:07 -070038from autotest_lib.server.cros.device_health_profile import profile_constants
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070039from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070040from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070041from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080042from autotest_lib.server.hosts import chameleon_host
Otabek Kasimov832d9162020-07-27 19:24:57 -070043from autotest_lib.server.hosts import cros_constants
Richard Barnetted31580e2018-05-14 19:58:00 +000044from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080045from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070046from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070047from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070048from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080049from autotest_lib.site_utils.rpm_control_system import rpm_client
Otabek Kasimov808cd832020-05-28 18:27:46 -070050from autotest_lib.site_utils.admin_audit import constants as audit_const
Otabek Kasimov27bb2862020-08-10 14:40:45 -070051from autotest_lib.site_utils.admin_audit import verifiers as audit_verify
Derek Beckettf73baca2020-08-19 15:08:47 -070052from six.moves import zip
Simran Basid5e5e272012-09-24 15:23:59 -070053
Andrew Luo4be621d2020-03-21 07:01:13 -070054
Simran Basi382506b2016-09-13 14:58:15 -070055# In case cros_host is being ran via SSP on an older Moblab version with an
56# older chromite version.
57try:
Mike Frysinger714c5b02020-09-04 23:22:54 -040058 from autotest_lib.utils.frozen_chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080059except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080060 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080061
Simran Basid5e5e272012-09-24 15:23:59 -070062
Dan Shib8540a52015-07-16 14:18:23 -070063CONFIG = global_config.global_config
64
beepsc87ff602013-07-31 21:53:00 -070065class FactoryImageCheckerException(error.AutoservError):
66 """Exception raised when an image is a factory image."""
67 pass
68
69
Fang Deng0ca40e22013-08-27 17:47:44 -070070class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070071 """Chromium OS specific subclass of Host."""
72
Simran Basi5ace6f22016-01-06 17:30:44 -080073 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
74
Scott Zawalski62bacae2013-03-05 10:40:32 -050075 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070076
Richard Barnette03a0c132012-11-05 12:40:35 -080077 # Timeout values (in seconds) associated with various Chrome OS
78 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070079 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080080 # In general, a good rule of thumb is that the timeout can be up
81 # to twice the typical measured value on the slowest platform.
82 # The times here have not necessarily been empirically tested to
83 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070084 #
85 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080086 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
87 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080088 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070089 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080090 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070091 # screen delay, time to start the network on the DUT, and the
92 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070093 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080094 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080095 # network.
beepsf079cfb2013-09-18 17:49:51 -070096 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
Otabek Kasimovaeb47fe2021-01-26 20:53:55 -080097 # ADMIN_INSTALL_TIMEOUT: Time to allow for chromeos-install
98 # used by admin tasks.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080099 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
100 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -0700101
102 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -0800103 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +0800104 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -0700105 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -0700106 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -0700107 INSTALL_TIMEOUT = 480
Otabek Kasimovaeb47fe2021-01-26 20:53:55 -0800108 ADMIN_INSTALL_TIMEOUT = 600
Dan Shi2c88eed2013-11-12 10:18:38 -0800109 POWERWASH_BOOT_TIMEOUT = 60
Jeremy Bettis62942362022-03-14 15:23:48 -0600110 DEVSERVER_DOWNLOAD_TIMEOUT = 600
Chris Sosab76e0ee2013-05-22 16:55:41 -0700111
Dan Shica503482015-03-30 17:23:25 -0700112 # Minimum OS version that supports server side packaging. Older builds may
113 # not have server side package built or with Autotest code change to support
114 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -0700115 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -0700116 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700117
Dana Goyettec172b172020-07-29 16:26:15 -0700118 USE_FSFREEZE = CONFIG.get_config_value(
Dana Goyette6242cb32020-09-23 11:02:57 -0700119 'CROS', 'enable_fs_freeze', type=bool, default=False)
Dana Goyettec172b172020-07-29 16:26:15 -0700120
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800121 # REBOOT_TIMEOUT: How long to wait for a reboot.
122 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700123 # We have a long timeout to ensure we don't flakily fail due to other
124 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700125 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
126 # return from reboot' bug is solved.
127 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700128
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800129 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
130 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700131 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
132 # since changing servo role will reset USB state
133 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800134 _USB_POWER_TIMEOUT = 5
135 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700136 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800137
Fang Dengdeba14f2014-11-14 11:54:09 -0800138 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
139 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700140
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800141 # Constants used in ping_wait_up() and ping_wait_down().
142 #
143 # _PING_WAIT_COUNT is the approximate number of polling
144 # cycles to use when waiting for a host state change.
145 #
146 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
147 # for arguments to the internal _ping_wait_for_status()
148 # method.
149 _PING_WAIT_COUNT = 40
150 _PING_STATUS_DOWN = False
151 _PING_STATUS_UP = True
152
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800153 # Allowed values for the power_method argument.
154
Garry Wang5e5538a2019-04-08 15:36:18 -0700155 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
156 # DUTs except those with servo_v4 CCD.
157 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
158 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800159 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
160 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
161 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700162 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800163 POWER_CONTROL_SERVO = 'servoj10'
164 POWER_CONTROL_MANUAL = 'manual'
165
166 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700167 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800168 POWER_CONTROL_SERVO,
169 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800170
Simran Basi5e6339a2013-03-21 11:34:32 -0700171 _RPM_OUTLET_CHANGED = 'outlet_changed'
172
Dan Shi9cb0eec2014-06-03 09:04:50 -0700173 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700174 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700175 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700176
Brent Peterson1cb623a2020-01-09 13:14:28 -0800177 # Regular expression for extracting EC version string
178 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
179
180 # Regular expression for extracting BIOS version string
181 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
182
Brent Petersonc70a1832020-01-24 15:54:35 -0800183 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700184 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800185
J. Richard Barnette964fba02012-10-24 17:34:29 -0700186 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800187 def check_host(host, timeout=10):
188 """
189 Check if the given host is a chrome-os host.
190
191 @param host: An ssh host representing a device.
192 @param timeout: The timeout for the run command.
193
194 @return: True if the host device is chromeos.
195
beeps46dadc92013-11-07 14:07:10 -0800196 """
197 try:
Allen Liad719c12017-06-27 23:48:04 +0000198 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700199 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700200 '! grep -q moblab /etc/lsb-release && '
Derek Beckett342e3e62021-01-05 17:17:23 -0800201 '! grep -q labstation /etc/lsb-release &&'
202 ' grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
203 ignore_status=True,
Laurence Goodby468de252017-06-08 17:22:53 -0700204 timeout=timeout).stdout
Derek Beckett342e3e62021-01-05 17:17:23 -0800205 if result:
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800206 return not (
207 lsbrelease_utils.is_jetstream(
Derek Beckett342e3e62021-01-05 17:17:23 -0800208 lsb_release_content=result) or
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800209 lsbrelease_utils.is_gce_board(
Derek Beckett342e3e62021-01-05 17:17:23 -0800210 lsb_release_content=result))
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800211
beeps46dadc92013-11-07 14:07:10 -0800212 except (error.AutoservRunError, error.AutoservSSHTimeout):
213 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700214
215 return False
beeps46dadc92013-11-07 14:07:10 -0800216
217
218 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800219 def get_chameleon_arguments(args_dict):
220 """Extract chameleon options from `args_dict` and return the result.
221
222 Recommended usage:
223 ~~~~~~~~
224 args_dict = utils.args_to_dict(args)
225 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
226 host = hosts.create_host(machine, chameleon_args=chameleon_args)
227 ~~~~~~~~
228
229 @param args_dict Dictionary from which to extract the chameleon
230 arguments.
231 """
Sam McNally66594ca2019-12-09 12:45:44 +1100232 chameleon_args = {key: args_dict[key]
233 for key in ('chameleon_host', 'chameleon_port')
234 if key in args_dict}
235 if 'chameleon_ssh_port' in args_dict:
236 chameleon_args['port'] = int(args_dict['chameleon_ssh_port'])
237 return chameleon_args
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800238
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800239 @staticmethod
240 def get_btpeer_arguments(args_dict):
241 """Extract btpeer options from `args_dict` and return the result.
242
243 This is used to parse details of Bluetooth peer.
244 Recommended usage:
245 ~~~~~~~~
246 args_dict = utils.args_to_dict(args)
247 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700248 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800249 ~~~~~~~~
250
251 @param args_dict: Dictionary from which to extract the btpeer
252 arguments.
253 """
254 if 'btpeer_host_list' in args_dict:
255 result = []
256 for btpeer in args_dict['btpeer_host_list'].split(','):
Claire Changd0b19842020-11-04 22:28:45 +0800257 # IPv6 addresses including a port number should be enclosed in
258 # square brackets.
259 delimiter = ']:' if re.search(r':.*:', btpeer) else ':'
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800260 result.append({key: value for key,value in
261 zip(('btpeer_host','btpeer_port'),
Claire Changd0b19842020-11-04 22:28:45 +0800262 btpeer.strip('[]').split(delimiter))})
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800263 return result
264 else:
Anand K Mistrye8933092020-08-05 14:49:41 +1000265 return {key: args_dict[key]
266 for key in ('btpeer_host', 'btpeer_port', 'btpeer_ssh_port')
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800267 if key in args_dict}
268
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800269
270 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700271 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800272 """Extract chameleon options from `args_dict` and return the result.
273
274 Recommended usage:
275 ~~~~~~~~
276 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700277 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
278 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800279 ~~~~~~~~
280
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700281 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800282 arguments.
283 """
Allen Li083866b2016-08-18 10:07:10 -0700284 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700285 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700286 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800287
288
289 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800290 def get_servo_arguments(args_dict):
291 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800292
293 Recommended usage:
294 ~~~~~~~~
295 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700296 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800297 host = hosts.create_host(machine, servo_args=servo_args)
298 ~~~~~~~~
299
300 @param args_dict Dictionary from which to extract the servo
301 arguments.
302 """
Garry Wang11b5e872020-03-11 15:14:08 -0700303 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
Andrew Luo4be621d2020-03-21 07:01:13 -0700304 servo_constants.SERVO_HOST_SSH_PORT_ATTR,
Garry Wang11b5e872020-03-11 15:14:08 -0700305 servo_constants.SERVO_PORT_ATTR,
Otabek Kasimov382c3bb2020-10-28 13:22:45 -0700306 servo_constants.SERVO_SERIAL_ATTR,
Garry Wang11b5e872020-03-11 15:14:08 -0700307 servo_constants.SERVO_BOARD_ATTR,
308 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200309 servo_args = {key: args_dict[key]
310 for key in servo_attrs
311 if key in args_dict}
312 return (
313 None
Garry Wang11b5e872020-03-11 15:14:08 -0700314 if servo_constants.SERVO_HOST_ATTR in servo_args
315 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200316 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700317
J. Richard Barnette964fba02012-10-24 17:34:29 -0700318
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800319 def _initialize(self,
320 hostname,
321 chameleon_args=None,
322 servo_args=None,
323 pdtester_args=None,
324 try_lab_servo=False,
325 try_servo_repair=False,
326 ssh_verbosity_flag='',
327 ssh_options='',
328 try_servo_recovery=False,
329 *args,
330 **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800331 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700332
Fang Denge545abb2014-12-30 18:43:47 -0800333 This method will attempt to create the test-assistant object
334 (chameleon/servo) when it is needed by the test. Check
335 the docstring of chameleon_host.create_chameleon_host and
336 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700337
Fang Denge545abb2014-12-30 18:43:47 -0800338 @param hostname: Hostname of the dut.
339 @param chameleon_args: A dictionary that contains args for creating
340 a ChameleonHost. See chameleon_host for details.
341 @param servo_args: A dictionary that contains args for creating
342 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700343 @param try_lab_servo: When true, indicates that an attempt should
344 be made to create a ServoHost for a DUT in
345 the test lab, even if not required by
346 `servo_args`. See servo_host for details.
347 @param try_servo_repair: If a servo host is created, check it
348 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800349 See servo_host for details.
350 @param ssh_verbosity_flag: String, to pass to the ssh command to control
351 verbosity.
352 @param ssh_options: String, other ssh options to pass to the ssh
353 command.
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800354 @param try_servo_recovery: When True, start servod in recovery mode.
355 See servo_host for details.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700356 """
Andrew Luo4be621d2020-03-21 07:01:13 -0700357 super(CrosHost, self)._initialize(hostname=hostname, *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800358 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700359 # hold special dut_state for repair process
360 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700361 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700362 # self.env is a dictionary of environment variable settings
363 # to be exported for commands run on the host.
364 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
365 # errors that might happen.
366 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700367 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700368 self._ssh_options = ssh_options
Garry Wang1a493d82020-08-31 21:01:19 -0700369 self.health_profile = None
Garry Wang5e5538a2019-04-08 15:36:18 -0700370 self._default_power_method = None
Otabek Kasimov39637412020-11-23 19:09:27 -0800371 dut_health_profile = device_health_profile.DeviceHealthProfile(
372 hostname=self.hostname,
373 host_info=self.host_info_store.get(),
374 result_dir=self.get_result_dir())
Otabek Kasimovb61729d2020-11-24 00:42:43 -0800375
376 # TODO(otabek@): remove when b/171414073 closed
Andrew Luo4be621d2020-03-21 07:01:13 -0700377 if self.use_icmp:
Derek Beckettc7677812021-02-12 14:41:11 -0800378 pingable_before_servo = self.is_up_fast(count=1)
Andrew Luo4be621d2020-03-21 07:01:13 -0700379 if pingable_before_servo:
380 logging.info('DUT is pingable before init Servo.')
381 else:
382 logging.info('Skipping ping to DUT before init Servo.')
Otabek Kasimov39637412020-11-23 19:09:27 -0800383 _servo_host, servo_state = servo_host.create_servo_host(
384 dut=self,
385 servo_args=servo_args,
386 try_lab_servo=try_lab_servo,
387 try_servo_repair=try_servo_repair,
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800388 try_servo_recovery=try_servo_recovery,
Otabek Kasimov39637412020-11-23 19:09:27 -0800389 dut_host_info=self.host_info_store.get(),
390 dut_health_profile=dut_health_profile)
391 if dut_health_profile.is_loaded():
392 logging.info('Device health profile loaded.')
393 # The device profile is located in the servo_host which make it
394 # dependency. If profile is not loaded yet then we do not have it
395 # TODO(otabek@) persist device provide out of servo-host.
396 self.health_profile = dut_health_profile
397 self.set_servo_host(_servo_host, servo_state)
Richard Barnettee519dcd2016-08-15 17:37:17 -0700398
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800399 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800400 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700401 dut=self.hostname,
402 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800403 if self._chameleon_host:
404 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800405 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800406 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700407
Shijin Abraham78ce4402020-09-08 22:04:27 -0700408 # Bluetooth peers will be populated by the test if needed
409 self._btpeer_host_list = []
410 self.btpeer_list = []
411 self.btpeer = None
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800412
howardchung83e55272019-08-08 14:08:05 +0800413 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700414 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700415 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800416
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700417 if self._pdtester_host:
418 self.pdtester_servo = self._pdtester_host.get_servo()
419 logging.info('pdtester_servo: %r', self.pdtester_servo)
420 # Create the pdtester object used to access the ec uart
421 self.pdtester = pdtester.PDTester(self.pdtester_servo,
422 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800423 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700424 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800425
Fang Deng5d518f42013-08-02 14:04:32 -0700426
Shijin Abraham78ce4402020-09-08 22:04:27 -0700427 def initialize_btpeer(self, btpeer_args=[]):
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800428 """ Initialize the Bluetooth peers
429
430 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
431 is chameleon host on Raspberry Pi.
432 @param btpeer_args: A dictionary that contains args for creating
433 a ChameleonHost. See chameleon_host for details.
434
435 """
Shijin Abraham78ce4402020-09-08 22:04:27 -0700436 logging.debug('Attempting to initialize bluetooth peers if available')
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700437 try:
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700438 if type(btpeer_args) is list:
439 btpeer_args_list = btpeer_args
440 else:
441 btpeer_args_list = [btpeer_args]
442
443 self._btpeer_host_list = chameleon_host.create_btpeer_host(
444 dut=self.hostname, btpeer_args_list=btpeer_args_list)
445 logging.debug('Bluetooth peer hosts are %s',
446 self._btpeer_host_list)
447 self.btpeer_list = [_host.create_chameleon_board() for _host in
448 self._btpeer_host_list if _host is not None]
449
450 if len(self.btpeer_list) > 0:
451 self.btpeer = self.btpeer_list[0]
452
453 logging.debug('After initialize_btpeer btpeer_list %s '
454 'btpeer_host_list is %s and btpeer is %s',
455 self.btpeer_list, self._btpeer_host_list,
456 self.btpeer)
457 except Exception as e:
458 logging.error('Exception %s in initialize_btpeer', str(e))
459
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800460
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000461 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700462 """Get latest stable cros image name from AFE.
463
464 Use the board name from the info store. Should that fail, try to
465 retrieve the board name from the host's installed image itself.
466
467 @returns: current stable cros image name for this host.
468 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700469 info = self.host_info_store.get()
470 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700471 logging.warn('No board label value found. Trying to infer '
472 'from the host itself.')
473 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700474 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700475 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
476 logging.error('Also failed to get the board name from the DUT '
477 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700478 raise error.AutoservError('Cannot determine board of the DUT'
479 ' while getting repair image name.')
480 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500481
482
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700483 def host_version_prefix(self, image):
484 """Return version label prefix.
485
486 In case the CrOS provisioning version is something other than the
487 standard CrOS version e.g. CrOS TH version, this function will
488 find the prefix from provision.py.
489
490 @param image: The image name to find its version prefix.
491 @returns: A prefix string for the image type.
492 """
493 return provision.get_version_label_prefix(image)
494
Andrew Luo3332ab22020-04-28 16:42:03 -0700495 def stage_build_to_usb(self, build):
496 """Stage the current ChromeOS image on the USB stick connected to the
497 servo.
498
499 @param build: The build to download and send to USB.
500 """
501 if not self.servo:
502 raise error.TestError('Host %s does not have servo.' %
503 self.hostname)
504
505 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700506
507 try:
508 self.servo.image_to_servo_usb(update_url)
509 finally:
510 # servo.image_to_servo_usb turned the DUT off, so turn it back on
511 logging.debug('Turn DUT power back on.')
512 self.servo.get_power_state_controller().power_on()
513
Andrew Luo3332ab22020-04-28 16:42:03 -0700514 logging.debug('ChromeOS image %s is staged on the USB stick.',
515 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700516
beepsdae65fd2013-07-26 16:24:41 -0700517 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700518 """
519 Make sure job_repo_url of this host is valid.
520
joychen03eaad92013-06-26 09:55:21 -0700521 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700522 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
523 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
524 download and extract it. If the devserver embedded in the url is
525 unresponsive, update the job_repo_url of the host after staging it on
526 another devserver.
527
528 @param job_repo_url: A url pointing to the devserver where the autotest
529 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700530 @param tag: The tag from the server job, in the format
531 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700532
533 @raises DevServerException: If we could not resolve a devserver.
534 @raises AutoservError: If we're unable to save the new job_repo_url as
535 a result of choosing a new devserver because the old one failed to
536 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700537 @raises urllib2.URLError: If the devserver embedded in job_repo_url
538 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700539 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800540 info = self.host_info_store.get()
541 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700542 if not job_repo_url:
543 logging.warning('No job repo url set on host %s', self.hostname)
544 return
545
546 logging.info('Verifying job repo url %s', job_repo_url)
547 devserver_url, image_name = tools.get_devserver_build_from_package_url(
548 job_repo_url)
549
beeps0c865032013-07-30 11:37:06 -0700550 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700551
552 logging.info('Staging autotest artifacts for %s on devserver %s',
553 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700554
555 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700556 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700557 stage_time = time.time() - start_time
558
559 # Record how much of the verification time comes from a devserver
560 # restage. If we're doing things right we should not see multiple
561 # devservers for a given board/build/branch path.
562 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800563 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700564 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800565 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700566 pass
567 else:
beeps0c865032013-07-30 11:37:06 -0700568 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700569 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700570 stats_key = {
571 'board': board,
572 'build_type': build_type,
573 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700574 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700575 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800576
577 monarch_fields = {
578 'board': board,
579 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800580 'branch': branch,
581 'dev_server': devserver,
582 }
583 metrics.Counter(
584 'chromeos/autotest/provision/verify_url'
585 ).increment(fields=monarch_fields)
586 metrics.SecondsDistribution(
587 'chromeos/autotest/provision/verify_url_duration'
588 ).add(stage_time, fields=monarch_fields)
589
590
Congbin Guoe4f492d2021-09-17 14:20:46 -0700591 def stage_server_side_package(self, image=None, image_storage_server=''):
Dan Shicf4d2032015-03-12 15:04:21 -0700592 """Stage autotest server-side package on devserver.
593
594 @param image: Full path of an OS image to install or a build name.
Congbin Guoe4f492d2021-09-17 14:20:46 -0700595 @param image_storage_server: The GS bucket we download the image from.
Dan Shicf4d2032015-03-12 15:04:21 -0700596
597 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700598
599 @raise: error.AutoservError if fail to locate the build to test with, or
600 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700601 """
Congbin Guo21600592022-07-12 16:44:06 -0700602 # It's drone to download the SSP package, so we use the drone name to
603 # resolve the cache server.
604 # Note: Don't try to get IP of the host because the IP based cache
605 # server selection logic are deprecating.
606 # Both lab drones and satlab drones have the env var below.
607 drone_name = os.environ.get('DOCKER_DRONE_SERVER_NAME')
608 if not drone_name:
609 raise error.AutoservError('Cannot get drone name from environment')
Dan Shicf4d2032015-03-12 15:04:21 -0700610 if image:
611 image_name = tools.get_build_from_image(image)
612 if not image_name:
613 raise error.AutoservError(
614 'Failed to parse build name from %s' % image)
Congbin Guo21600592022-07-12 16:44:06 -0700615 ds = dev_server.ImageServer.resolve(image_name, drone_name)
Dan Shicf4d2032015-03-12 15:04:21 -0700616 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700617 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800618 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700619 if job_repo_url:
Congbin Guo21600592022-07-12 16:44:06 -0700620 _, image_name = (tools.get_devserver_build_from_package_url(
621 job_repo_url))
622 # Don't use cache server URL in job_repo_url since we are
623 # resolving a cache server for the *drone*, not the DUT.
624 ds = dev_server.ImageServer.resolve(image_name, drone_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800625 elif info.build is not None:
Congbin Guo21600592022-07-12 16:44:06 -0700626 ds = dev_server.ImageServer.resolve(info.build, drone_name)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700627 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700628 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800629 raise error.AutoservError(
630 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700631 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700632
633 # Get the OS version of the build, for any build older than
634 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
635 match = re.match('.*/R\d+-(\d+)\.', image_name)
636 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700637 raise error.AutoservError(
638 'Build %s is older than %s. Server side packaging is '
639 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700640
Congbin Guoe4f492d2021-09-17 14:20:46 -0700641 if not image_storage_server:
642 image_storage_server = CONFIG.get_config_value(
643 'CROS', 'image_storage_server', type=str)
Congbin Guoe0fbbe02021-09-17 14:49:18 -0700644 gs_bucket = image_storage_server.lstrip('gs://').rstrip('/')
Congbin Guo5a086512022-05-31 16:37:11 -0700645 return '%s/download/%s/%s/%s' % (ds.url(), gs_bucket, image_name,
646 'autotest_server_package.tar.bz2')
Dan Shicf4d2032015-03-12 15:04:21 -0700647
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700648 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700649 """Stage a build on a devserver and return the update_url.
650
651 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700652 @param artifact: a string like 'test_image'. Requests
653 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700654 @returns a tuple of (image_name, URL) like
655 (lumpy-release/R27-3837.0.0,
656 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700657 """
658 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000659 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700660 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800661 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700662 devserver.stage_artifacts(image_name, [artifact])
663 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700664 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700665 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700666 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700667 else:
668 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700669
670
beepse539be02013-07-31 21:57:39 -0700671 def stage_factory_image_for_servo(self, image_name):
672 """Stage a build on a devserver and return the update_url.
673
674 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700675
beepse539be02013-07-31 21:57:39 -0700676 @return: An update URL, eg:
677 http://<devserver>/static/canary-channel/\
678 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700679
680 @raises: ValueError if the factory artifact name is missing from
681 the config.
682
beepse539be02013-07-31 21:57:39 -0700683 """
684 if not image_name:
685 logging.error('Need an image_name to stage a factory image.')
686 return
687
Dan Shib8540a52015-07-16 14:18:23 -0700688 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700689 'CROS', 'factory_artifact', type=str, default='')
690 if not factory_artifact:
691 raise ValueError('Cannot retrieve the factory artifact name from '
692 'autotest config, and hence cannot stage factory '
693 'artifacts.')
694
beepse539be02013-07-31 21:57:39 -0700695 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800696 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700697 devserver.stage_artifacts(
698 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700699 [factory_artifact],
700 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700701
702 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
703
704
Laurence Goodby778c9a42017-05-24 19:24:07 -0700705 def prepare_for_update(self):
706 """Prepares the DUT for an update.
707
708 Subclasses may override this to perform any special actions
709 required before updating.
710 """
Laurence Goodby468de252017-06-08 17:22:53 -0700711 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700712
713
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800714 def _clear_fw_version_labels(self, rw_only):
715 """Clear firmware version labels from the machine.
716
717 @param rw_only: True to only clear fwrw_version; otherewise, clear
718 both fwro_version and fwrw_version.
719 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700720 info = self.host_info_store.get()
721 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800722 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700723 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
724 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700725
726
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800727 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700728 """Add firmware version label to the machine.
729
730 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800731 @param rw_only: True to only add fwrw_version; otherwise, add both
732 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700733
734 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700735 info = self.host_info_store.get()
736 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800737 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700738 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
739 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700740
741
Namyoon Woo33f38852020-04-13 17:26:58 -0700742 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800743 """Search for the latest package release version from the image archive,
744 and return it.
745
Namyoon Woo33f38852020-04-13 17:26:58 -0700746 @param platform: platform name, a.k.a. board or model
747 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800748
Namyoon Woo33f38852020-04-13 17:26:58 -0700749 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
750 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800751 or None if LATEST release file does not exist.
752 """
753
Namyoon Woo33f38852020-04-13 17:26:58 -0700754 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800755
Namyoon Woo33f38852020-04-13 17:26:58 -0700756 # Search the image path in reference board archive as well.
757 # For example, bob has its binary image under its reference board (gru)
758 # image archive.
759 if ref_board:
760 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800761
Namyoon Woo33f38852020-04-13 17:26:58 -0700762 for board in platforms:
763 # Read 'LATEST-1.0.0' file
764 branch_dir = provision.FW_BRANCH_GLOB % board
765 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
766 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800767
Namyoon Woo33f38852020-04-13 17:26:58 -0700768 try:
769 # The result could be one or more.
770 result = utils.system_output('gsutil ls -d ' + latest_file)
771
772 candidates = re.findall('gs://.*', result)
773
774 # Found the directory candidates. No need to check the other
775 # board name cadidates. Let's break the loop.
776 break
777 except error.CmdError:
778 # It doesn't exist. Let's move on to the next item.
779 pass
780 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800781 logging.error('No LATEST release info is available.')
782 return None
783
Namyoon Woo406c7d42020-01-24 15:57:11 -0800784 for cand_dir in candidates:
785 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800786
Namyoon Woo406c7d42020-01-24 15:57:11 -0800787 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700788 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800789 try:
790 # Check if release_path does exist.
791 release = utils.system_output('gsutil ls -d ' + release_path)
792 # Now 'release' has a full directory path: e.g.
793 # gs://chromeos-image-archive/firmware-octopus-11297.B-
794 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
795
796 # Remove "gs://chromeos-image-archive".
797 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
798
799 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
800 return release.strip('/')
801 except error.CmdError:
802 # The directory might not exist. Let's try next candidate.
803 pass
804 else:
805 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800806
Brent Peterson1cb623a2020-01-09 13:14:28 -0800807 @staticmethod
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600808 def get_version_from_image(host, bios_image, ec_image):
Brent Peterson8039b472020-02-14 10:51:23 -0800809 """Get version string from binary image using regular expression.
810
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600811 @param host: An instance of hosts.Host.
812 @param bios_image: Filename of AP BIOS image on the DUT/labstation.
813 @param ec_image: Filename of EC image on the DUT/labstation.
Brent Peterson8039b472020-02-14 10:51:23 -0800814
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600815 @return Tuple of bios version and ec version
Brent Peterson8039b472020-02-14 10:51:23 -0800816 """
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600817 if not host:
818 return None
819 cmd_args = ['futility', 'update', '--manifest']
820 if bios_image:
821 cmd_args.append('-i')
822 cmd_args.append(bios_image)
823 if ec_image:
824 cmd_args.append('-e')
825 cmd_args.append(ec_image)
826 cmd = ' '.join([utils.sh_quote_word(arg) for arg in cmd_args])
827 stdout = host.run(cmd).stdout
Jeremy Bettisf243afe2022-04-13 10:45:13 -0600828 if not isinstance(stdout, six.text_type):
Jeremy Bettise50d16e2022-04-05 12:32:04 -0600829 stdout = stdout.decode('utf-8')
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600830 io = StringIO(stdout)
831 data = json.load(io)
832 return (
833 data.get('default', {}).get('host', {}).get('versions',
834 {}).get('rw'),
835 data.get('default', {}).get('ec', {}).get('versions',
836 {}).get('rw'),
837 )
Brent Peterson1cb623a2020-01-09 13:14:28 -0800838
839
Garry Wangad2a1712020-03-26 15:06:43 -0700840 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800841 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700842 try_scp=False, install_ec=True, install_bios=True,
843 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700844 """Install firmware to the DUT.
845
846 Use stateful update if the DUT is already running the same build.
847 Stateful update does not update kernel and tends to run much faster
848 than a full reimage. If the DUT is running a different build, or it
849 failed to do a stateful update, full update, including kernel update,
850 will be applied to the DUT.
851
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800852 Once a host enters firmware_install its fw[ro|rw]_version label will
853 be removed. After the firmware is updated successfully, a new
854 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700855
856 @param build: The build version to which we want to provision the
857 firmware of the machine,
858 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800859 @param rw_only: True to only install firmware to its RW portions. Keep
860 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700861 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800862 @param local_tarball: Path to local firmware image for installing
863 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800864 @param verify_version: True to verify EC and BIOS versions after
865 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800866 @param try_scp: False to always program using servo, true to try copying
867 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700868 @param install_ec: True to install EC FW, and False to skip it.
869 @param install_bios: True to install BIOS, and False to skip it.
870 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700871
872 TODO(dshi): After bug 381718 is fixed, update here with corresponding
873 exceptions that could be raised.
874
875 """
876 if not self.servo:
877 raise error.TestError('Host %s does not have servo.' %
878 self.hostname)
879
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700880 # Get the DUT board name from AFE.
881 info = self.host_info_store.get()
882 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700883 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800884
885 if board is None or board == '':
886 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700887
Namyoon Woo382e5892020-05-20 16:48:40 -0700888 # if board_as argument is passed, then use it instead of the original
889 # board name.
890 if board_as:
891 board = board_as
892
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700893 if model is None or model == '':
Namyoon Woofb16eae2020-08-14 10:02:39 -0700894 try:
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700895 model = self.get_platform()
Namyoon Woofb16eae2020-08-14 10:02:39 -0700896 except Exception as e:
897 logging.warn('Dut is unresponsive: %s', str(e))
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700898
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800899 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700900 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800901 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700902 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800903
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700904 try:
905 ds = dev_server.ImageServer.resolve(build, self.hostname)
906 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800907
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700908 if not dest:
909 tmpd = autotemp.tempdir(unique_id='fwimage')
910 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800911
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700912 # Download firmware image
913 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
914 local_tarball = os.path.join(dest, os.path.basename(fwurl))
Jeremy Bettis62942362022-03-14 15:23:48 -0600915 logging.info('Downloading file from %s to %s.', fwurl,
916 local_tarball)
917 ds.download_file(fwurl,
918 local_tarball,
919 timeout=self.DEVSERVER_DOWNLOAD_TIMEOUT)
920 logging.info('Done downloading')
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700921 except Exception as e:
922 raise error.TestError('Failed to download firmware package: %s'
923 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700924
Namyoon Woo382e5892020-05-20 16:48:40 -0700925 ec_image = None
926 if install_ec:
927 # Extract EC image from tarball
928 logging.info('Extracting EC image.')
929 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700930 logging.info('Extracted: %s', ec_image)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800931
Namyoon Woo382e5892020-05-20 16:48:40 -0700932 bios_image = None
933 if install_bios:
934 # Extract BIOS image from tarball
935 logging.info('Extracting BIOS image.')
936 bios_image = self.servo.extract_bios_image(board, model,
937 local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700938 logging.info('Extracted: %s', bios_image)
Namyoon Woo382e5892020-05-20 16:48:40 -0700939
940 if not bios_image and not ec_image:
941 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800942
Brent Petersonc70a1832020-01-24 15:54:35 -0800943 # Clear firmware version labels
944 self._clear_fw_version_labels(rw_only)
945
946 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800947 try:
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600948 image_ec_version = None
949 image_bios_version = None
950
Garry Wang50e4a492020-08-05 12:29:57 -0700951 # Check if copying to DUT is enabled and DUT is available
952 if try_scp and self.is_up():
Brent Petersonc70a1832020-01-24 15:54:35 -0800953 # DUT is available, make temp firmware directory to store images
954 logging.info('Making temp folder.')
955 dest_folder = '/tmp/firmware'
956 self.run('mkdir -p ' + dest_folder)
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600957 dest_bios_path = None
958 dest_ec_path = None
Brent Petersonc70a1832020-01-24 15:54:35 -0800959
Namyoon Woo68b68082020-06-02 13:13:14 -0700960 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800961
Namyoon Woo382e5892020-05-20 16:48:40 -0700962 if bios_image:
963 # Send BIOS firmware image to DUT
964 logging.info('Sending BIOS firmware.')
965 dest_bios_path = os.path.join(dest_folder,
966 os.path.basename(bios_image))
967 self.send_file(bios_image, dest_bios_path)
968
969 # Initialize firmware update command for BIOS image
970 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800971
972 # Send EC firmware image to DUT when EC image was found
973 if ec_image:
974 logging.info('Sending EC firmware.')
975 dest_ec_path = os.path.join(dest_folder,
976 os.path.basename(ec_image))
977 self.send_file(ec_image, dest_ec_path)
978
979 # Add EC image to firmware update command
980 fw_cmd += ' -e %s' % dest_ec_path
981
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700982 # Make sure command is allowed to finish even if ssh fails.
983 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
984
Brent Peterson669edf42020-02-07 15:07:54 -0800985 # Update firmware on DUT
986 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700987 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700988 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700989 except error.AutoservRunError as e:
990 if e.result_obj.exit_status != 255:
991 raise
992 elif ec_image:
993 logging.warn("DUT network dropped during update"
994 " (often caused by EC resetting USB)")
995 else:
996 logging.error("DUT network dropped during update"
997 " (unexpected, since no EC image)")
998 raise
Jeremy Bettis067eb4a2022-03-18 11:15:21 -0600999 image_bios_version, image_ec_version = self.get_version_from_image(
1000 self, dest_bios_path, dest_ec_path)
Brent Petersonc70a1832020-01-24 15:54:35 -08001001 else:
1002 # Host is not available, program firmware using servo
Jeremy Bettis067eb4a2022-03-18 11:15:21 -06001003 dest_bios_path = None
1004 dest_ec_path = None
Brent Peterson669edf42020-02-07 15:07:54 -08001005 if ec_image:
Jeremy Bettis067eb4a2022-03-18 11:15:21 -06001006 dest_ec_path = self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -07001007 if bios_image:
Jeremy Bettis067eb4a2022-03-18 11:15:21 -06001008 dest_bios_path = self.servo.program_bios(
1009 bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -08001010 if utils.host_is_in_lab_zone(self.hostname):
1011 self._add_fw_version_label(build, rw_only)
Jeremy Bettis067eb4a2022-03-18 11:15:21 -06001012 image_bios_version, image_ec_version = self.get_version_from_image(
1013 self._servo_host, dest_bios_path, dest_ec_path)
Brent Peterson1cb623a2020-01-09 13:14:28 -08001014
1015 # Reboot and wait for DUT after installing firmware
1016 logging.info('Rebooting DUT.')
1017 self.servo.get_power_state_controller().reset()
1018 time.sleep(self.servo.BOOT_DELAY)
1019 self.test_wait_for_boot()
1020
1021 # When enabled verify EC and BIOS firmware version after programming
1022 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -08001023 # Check programmed EC firmware when EC image was found
1024 if ec_image:
1025 logging.info('Checking EC firmware version.')
Jeremy Bettis067eb4a2022-03-18 11:15:21 -06001026 if image_ec_version is None:
1027 raise error.TestFail(
1028 'Could not find EC version in %s' % ec_image)
Brent Peterson669edf42020-02-07 15:07:54 -08001029 dest_ec_version = self.get_ec_version()
Brent Peterson669edf42020-02-07 15:07:54 -08001030 if dest_ec_version != image_ec_version:
1031 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -07001032 'Failed to update EC firmware, version %s '
1033 '(expected %s)' % (dest_ec_version,
1034 image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -08001035
Namyoon Woo382e5892020-05-20 16:48:40 -07001036 if bios_image:
1037 # Check programmed BIOS firmware against expected version
1038 logging.info('Checking BIOS firmware version.')
Jeremy Bettis067eb4a2022-03-18 11:15:21 -06001039 if image_bios_version is None:
1040 raise error.TestFail(
1041 'Could not find BIOS version in %s' %
1042 bios_image)
Namyoon Woo382e5892020-05-20 16:48:40 -07001043 dest_bios_version = self.get_firmware_version()
Namyoon Woo382e5892020-05-20 16:48:40 -07001044 if dest_bios_version != image_bios_version:
1045 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -07001046 'Failed to update BIOS, version %s '
Namyoon Woo382e5892020-05-20 16:48:40 -07001047 '(expected %s)' % (dest_bios_version,
1048 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -07001049 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -07001050 if tmpd:
1051 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -07001052
1053
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001054 def install_image_to_servo_usb(self, image_url=None):
1055 """Installing a test image on a USB storage device.
J. Richard Barnette31b2e312013-04-04 16:05:22 -07001056
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001057 Download image to USB-storage attached to the Servo board.
Richard Barnette03a0c132012-11-05 12:40:35 -08001058
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001059 @param image_url: If specified use as the url to download to
1060 USB-storage.
1061
1062 @raises AutoservError if the image fails to download.
beepsf079cfb2013-09-18 17:49:51 -07001063
J. Richard Barnette0199cc82014-12-05 17:08:40 -08001064 """
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001065 if not image_url:
1066 logging.debug('Skip download as image_url not provided!')
1067 return
Garry Wang7b0e1b72020-03-25 19:08:59 -07001068
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001069 logging.info('Downloading image to USB')
Garry Wang7b0e1b72020-03-25 19:08:59 -07001070 metrics_field = {'download': bool(image_url)}
1071 metrics.Counter(
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001072 'chromeos/autotest/provision/servo_install/download_image'
1073 ).increment(fields=metrics_field)
Allen Li48a13fe2016-11-22 14:10:40 -08001074 with metrics.SecondsTimer(
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001075 'chromeos/autotest/servo_install/download_image_time'):
Otabek Kasimovb7cb8422020-12-23 02:38:32 -08001076 try:
1077 self.servo.image_to_servo_usb(image_path=image_url,
1078 power_off_dut=False)
1079 except error.AutotestError as e:
1080 metrics.Counter('chromeos/autotest/repair/image_to_usb_error'
1081 ).increment(
1082 fields={'host': self.hostname or ''})
1083 six.reraise(error.AutotestError, str(e), sys.exc_info()[2])
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001084
1085 def boot_in_recovery_mode(self,
1086 usb_boot_timeout=USB_BOOT_TIMEOUT,
1087 need_snk=False):
1088 """Booting host in recovery mode.
1089
1090 Boot device in recovery mode and verify that device booted from
1091 external storage as expected.
1092
1093 @param usb_boot_timeout: The usb_boot_timeout to use wait the host
1094 to boot. Factory images need a longer
1095 usb_boot_timeout than regular cros images.
1096 @param snk_mode: If True, switch servo_v4 role to 'snk'
1097 mode before boot DUT into recovery mode.
1098
1099 @raises AutoservError if the image fails to boot.
1100
1101 """
1102 logging.info('Booting from USB directly. Usb boot timeout: %s',
1103 usb_boot_timeout)
1104 with metrics.SecondsTimer(
1105 'chromeos/autotest/provision/servo_install/boot_duration'):
Otabek Kasimovb7cb8422020-12-23 02:38:32 -08001106 self.servo.boot_in_recovery_mode(snk_mode=need_snk)
Allen Li48a13fe2016-11-22 14:10:40 -08001107 if not self.wait_up(timeout=usb_boot_timeout):
Garry Wang53fc8f32020-09-18 13:30:08 -07001108 if need_snk:
1109 # Attempt to restore servo_v4 role to 'src' mode.
1110 self.servo.set_servo_v4_role('src')
Allen Li48a13fe2016-11-22 14:10:40 -08001111 raise hosts.AutoservRepairError(
1112 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001113 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001114
Garry Wang2e347df2020-10-30 14:04:26 -07001115 # Make sure the DUT is boot from an external device.
1116 if not self.is_boot_from_external_device():
1117 raise hosts.AutoservRepairError(
1118 'DUT is expected to boot from an external device(e.g. '
1119 'a usb stick), however it seems still boot from an'
1120 ' internal storage.', 'boot_from_internal_storage')
1121
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001122 def run_install_image(self,
1123 install_timeout=INSTALL_TIMEOUT,
1124 need_snk=False,
1125 is_repair=False):
1126 """Installing the image with chromeos-install.
1127
1128 Steps included:
1129 1) Recover TPM on the device
1130 2) Run chromeos-install
1131 2.a) if success: power off/on the device
1132 2.b) if fail:
1133 2.b.1) Mark for replacement if fail with hardware issue
1134 2.b.2) Run internal storage check. (Only if is_repair=True)
1135 3) Wait the device to boot as verifier of success install
1136
1137 Device has to booted from external storage.
1138
1139 @param install_timeout: The timeout to use when installing the
1140 chromeos image. Factory images need a
1141 longer install_timeout.
1142 @param snk_mode: If True, switch servo_v4 role to 'snk'
1143 mode before boot DUT into recovery mode.
1144 @param is_repair: Indicates if the method is called from a
1145 repair task.
1146
1147 @raises AutoservError if the fail in process of install image.
1148 @raises AutoservRepairError if fail to boot after install image.
1149
1150 """
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001151 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1152 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001153 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001154 try:
1155 self.run('chromeos-tpm-recovery')
1156 except error.AutoservRunError:
1157 logging.warn('chromeos-tpm-recovery is too old.')
1158
Allen Li48a13fe2016-11-22 14:10:40 -08001159 with metrics.SecondsTimer(
1160 'chromeos/autotest/provision/servo_install/install_duration'):
1161 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001162 try:
1163 self.run('chromeos-install --yes',timeout=install_timeout)
1164 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001165 except Exception as e:
1166 storage_errors = [
1167 'No space left on device',
1168 'I/O error when trying to write primary GPT',
1169 'Input/output error while writing out',
1170 'cannot read GPT header',
Otabek Kasimov2b7e8302020-08-21 09:23:31 -07001171 'can not determine destination device',
1172 'wrong fs type',
1173 'bad superblock on',
Otabek Kasimov808cd832020-05-28 18:27:46 -07001174 ]
1175 has_error = [msg for msg in storage_errors if(msg in str(e))]
1176 if has_error:
1177 info = self.host_info_store.get()
1178 info.set_version_label(
1179 audit_const.DUT_STORAGE_STATE_PREFIX,
1180 audit_const.HW_STATE_NEED_REPLACEMENT)
1181 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001182 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001183 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001184 logging.debug(
1185 'Fail install image from USB; Storage error; %s', e)
1186 raise error.AutoservError(
1187 'Failed to install image from USB due to a suspect '
1188 'disk failure, DUT storage state changed to '
1189 'need_replacement, please check debug log '
1190 'for details.')
1191 else:
Garry Wang790953f2020-10-29 21:11:57 -07001192 if is_repair:
1193 # DUT will be marked for replacement if storage is bad.
1194 audit_verify.VerifyDutStorage(self).verify()
Otabek Kasimov27bb2862020-08-10 14:40:45 -07001195
Otabek Kasimov808cd832020-05-28 18:27:46 -07001196 logging.debug('Fail install image from USB; %s', e)
1197 raise error.AutoservError(
1198 'Failed to install image from USB due to unexpected '
1199 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001200 finally:
1201 # We need reset the DUT no matter re-install success or not,
1202 # as we don't want leave the DUT in boot from usb state.
1203 logging.info('Power cycling DUT through servo.')
1204 self.servo.get_power_state_controller().power_off()
1205 self.servo.switch_usbkey('off')
Garry Wang53fc8f32020-09-18 13:30:08 -07001206 if need_snk:
1207 # Attempt to restore servo_v4 role to 'src' mode.
1208 self.servo.set_servo_v4_role('src')
Garry Wang033a31e2020-04-10 17:20:49 -07001209 # N.B. The Servo API requires that we use power_on() here
1210 # for two reasons:
1211 # 1) After turning on a DUT in recovery mode, you must turn
1212 # it off and then on with power_on() once more to
1213 # disable recovery mode (this is a Parrot specific
1214 # requirement).
1215 # 2) After power_off(), the only way to turn on is with
1216 # power_on() (this is a Storm specific requirement).
1217 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001218
1219 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001220 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001221 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1222 'test image after %d seconds' %
1223 self.BOOT_TIMEOUT,
1224 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001225
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001226 def servo_install(self,
1227 image_url=None,
1228 usb_boot_timeout=USB_BOOT_TIMEOUT,
1229 install_timeout=INSTALL_TIMEOUT,
1230 is_repair=False):
1231 """Re-install the OS on the DUT by:
1232
1233 Steps:
1234 1) Power off the host
1235 2) Installing an image on a USB-storage attached to the Servo board
1236 3) Booting that image in recovery mode
1237 4) Installing the image with chromeos-install.
1238
1239 @param image_url: If specified use as the url to install on
1240 the DUT otherwise boot the currently
1241 staged image on the USB stick.
1242 @param usb_boot_timeout: The usb_boot_timeout to use during
1243 re-image. Factory images need a longer
1244 usb_boot_timeout than regular cros images.
1245 @param install_timeout: The timeout to use when installing the
1246 chromeos image. Factory images need a
1247 longer install_timeout.
1248 @param is_repair: Indicates if the method is called from a
1249 repair task.
1250
1251 @raises AutoservError if the image fails to boot.
1252
1253 """
1254 self.servo.get_power_state_controller().power_off()
1255 if image_url:
1256 self.install_image_to_servo_usb(image_url=image_url)
1257 else:
1258 # Give the DUT some time to power_off if we skip
1259 # download image to usb. (crbug.com/982993)
1260 time.sleep(10)
1261
1262 need_snk = self.require_snk_mode_in_recovery()
1263
1264 self.boot_in_recovery_mode(usb_boot_timeout=usb_boot_timeout,
1265 need_snk=need_snk)
1266
1267 self.run_install_image(install_timeout=install_timeout,
1268 need_snk=need_snk,
1269 is_repair=is_repair)
Scott Zawalski62bacae2013-03-05 10:40:32 -05001270
Garry Wanga2e78172020-09-09 23:49:07 -07001271 def set_servo_host(self, host, servo_state=None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001272 """Set our servo host member, and associated servo.
1273
1274 @param host Our new `ServoHost`.
1275 """
1276 self._servo_host = host
Derek Beckettb66e5c82020-08-12 15:31:02 -07001277 self.servo_pwr_supported = None
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001278 if self._servo_host is not None:
1279 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001280 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001281 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Derek Beckettb66e5c82020-08-12 15:31:02 -07001282 try:
1283 self.servo_pwr_supported = self.servo.has_control('power_state')
1284 except Exception as e:
1285 logging.debug(
1286 "Could not get servo power state due to {}".format(e))
Gregory Nisbet93b23e22020-10-02 20:42:16 +00001287 else:
1288 self.servo = None
Derek Beckettb66e5c82020-08-12 15:31:02 -07001289 self.servo_pwr_supported = False
Otabek Kasimov41301a22020-05-10 15:28:21 -07001290 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001291 self.set_servo_state(servo_state)
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07001292 self._set_servo_topology()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001293
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001294
Richard Barnette9a26ad62016-06-10 12:03:08 -07001295 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001296 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001297 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001298
Richard Barnette9a26ad62016-06-10 12:03:08 -07001299 If the servo object is missing, attempt to repair the servo
1300 host. Repair failures are passed back to the caller.
1301
1302 @raise AutoservError: If there is no servo host for this CrOS
1303 host.
1304 """
1305 if self.servo:
1306 return
1307 if not self._servo_host:
1308 raise error.AutoservError('No servo host for %s.' %
1309 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001310 try:
1311 self._servo_host.repair()
1312 except:
1313 raise
1314 finally:
1315 self.set_servo_host(self._servo_host)
1316
1317
Otabek Kasimov41301a22020-05-10 15:28:21 -07001318 def set_servo_type(self):
1319 """Set servo info labels to dut host_info"""
1320 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001321 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001322 return
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -08001323 if not self.is_servo_in_working_state():
1324 logging.debug('Servo is not good, skip update servo_type.')
1325 return
Otabek Kasimov41301a22020-05-10 15:28:21 -07001326 servo_type = self.servo.get_servo_type()
1327 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001328 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001329 ' by `dut-control servo_type`! Please file a bug'
1330 ' and inform infra team as we are not expected '
1331 ' to reach this point.')
1332 return
1333 host_info = self.host_info_store.get()
1334 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1335 old_type = host_info.get_label_value(prefix)
1336 if old_type == servo_type:
1337 # do not need update
1338 return
1339 host_info.set_version_label(prefix, servo_type)
1340 self.host_info_store.commit(host_info)
1341 logging.info('ServoHost: servo_type updated to %s '
1342 '(previous: %s)', servo_type, old_type)
1343
1344
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001345 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001346 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001347 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001348 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001349 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001350 old_state = host_info.get_label_value(servo_state_prefix)
1351 if old_state == servo_state:
1352 # do not need update
1353 return
1354 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001355 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001356 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1357 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001358
1359
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001360 def get_servo_state(self):
1361 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001362 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001363 return host_info.get_label_value(servo_state_prefix)
1364
Otabek Kasimov5f039202020-10-28 15:45:29 -07001365 def is_servo_in_working_state(self):
1366 """Validate servo is in WORKING state."""
1367 servo_state = self.get_servo_state()
1368 return servo_state == servo_constants.SERVO_STATE_WORKING
1369
Dana Goyette655af512020-09-03 10:48:23 -07001370 def get_servo_usb_state(self):
1371 """Get the label value indicating the health of the USB drive.
1372
1373 @return: The label value if defined, otherwise '' (empty string).
1374 @rtype: str
1375 """
1376 host_info = self.host_info_store.get()
1377 servo_usb_state_prefix = audit_const.SERVO_USB_STATE_PREFIX
1378 return host_info.get_label_value(servo_usb_state_prefix)
1379
1380 def is_servo_usb_usable(self):
1381 """Check if the servo USB storage device is usable for FAFT.
1382
1383 @return: False if the label indicates a state that will break FAFT.
1384 True if state is okay, or if state is not defined.
1385 @rtype: bool
1386 """
1387 usb_state = self.get_servo_usb_state()
1388 return usb_state in ('', audit_const.HW_STATE_ACCEPTABLE,
1389 audit_const.HW_STATE_NORMAL,
1390 audit_const.HW_STATE_UNKNOWN)
Otabek Kasimov41301a22020-05-10 15:28:21 -07001391
Garry Wang000c6c02020-05-11 21:27:23 -07001392 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1393 if smart_usbhub_detected is None:
1394 # skip the label update here as this indicate we wasn't able
1395 # to confirm usbhub type.
1396 return
1397 host_info = self.host_info_store.get()
1398 if (smart_usbhub_detected ==
1399 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1400 # skip label update if current label match the truth.
1401 return
1402 if smart_usbhub_detected:
1403 logging.info('Adding %s label to host %s',
1404 servo_constants.SMART_USBHUB_LABEL,
1405 self.hostname)
1406 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1407 else:
1408 logging.info('Removing %s label from host %s',
1409 servo_constants.SMART_USBHUB_LABEL,
1410 self.hostname)
1411 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1412 self.host_info_store.commit(host_info)
1413
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001414 def repair(self):
1415 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001416
1417 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001418 not call back to the parent class, but instead relies on
1419 `self._repair_strategy` to coordinate the verification and
1420 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001421 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001422 message = 'Beginning repair for host %s board %s model %s'
1423 info = self.host_info_store.get()
1424 message %= (self.hostname, info.board, info.model)
1425 self.record('INFO', None, None, message)
Garry Wanga2e78172020-09-09 23:49:07 -07001426 profile_state = profile_constants.DUT_STATE_READY
Shijin Abraham78ce4402020-09-08 22:04:27 -07001427 # Initialize bluetooth peers
1428 self.initialize_btpeer()
Garry Wang87af1d02020-05-26 17:55:54 -07001429 try:
1430 self._repair_strategy.repair(self)
1431 except hosts.AutoservVerifyDependencyError as e:
Otabek Kasimovd48389b2020-12-07 02:38:34 -08001432 # TODO(otabek): remove when finish b/174191325
1433 self._stat_if_pingable_but_not_sshable()
Garry Wang87af1d02020-05-26 17:55:54 -07001434 # We don't want flag a DUT as failed if only non-critical
1435 # verifier(s) failed during the repair.
1436 if e.is_critical():
Garry Wanga2e78172020-09-09 23:49:07 -07001437 profile_state = profile_constants.DUT_STATE_REPAIR_FAILED
Otabek Kasimov86062d02020-11-17 13:30:22 -08001438 self._reboot_labstation_if_needed()
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001439 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001440 raise
Garry Wanga2e78172020-09-09 23:49:07 -07001441 finally:
1442 self.set_health_profile_dut_state(profile_state)
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001443
Otabek Kasimov8bb09912020-10-01 14:44:57 -07001444 def get_verifier_state(self, tag):
Otabek Kasimov44273d22021-02-26 17:13:24 -08001445 """Return the state of host verifier by tag.
Otabek Kasimov8bb09912020-10-01 14:44:57 -07001446
1447 @returns: bool or None
1448 """
1449 return self._repair_strategy.verifier_is_good(tag)
Richard Barnette82c35912012-11-20 10:09:10 -08001450
Otabek Kasimov44273d22021-02-26 17:13:24 -08001451 def get_repair_strategy_node(self, tag):
1452 """Return the instance of verifier/repair node for host by tag.
1453
1454 @returns: _DependencyNode or None
1455 """
1456 return self._repair_strategy.node_by_tag(tag)
1457
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001458 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001459 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001460 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001461
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001462 if self._chameleon_host:
1463 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001464
Garry Wang1a493d82020-08-31 21:01:19 -07001465 if self.health_profile:
1466 try:
1467 self.health_profile.close()
1468 except Exception as e:
1469 logging.warning(
1470 'Failed to finalize device health profile; %s', e)
1471
xixuand6011f12016-12-08 15:01:58 -08001472 if self._servo_host:
1473 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001474
Dan Shi49ca0932014-11-14 11:22:27 -08001475 def get_power_supply_info(self):
1476 """Get the output of power_supply_info.
1477
1478 power_supply_info outputs the info of each power supply, e.g.,
1479 Device: Line Power
1480 online: no
1481 type: Mains
1482 voltage (V): 0
1483 current (A): 0
1484 Device: Battery
1485 state: Discharging
1486 percentage: 95.9276
1487 technology: Li-ion
1488
1489 Above output shows two devices, Line Power and Battery, with details of
1490 each device listed. This function parses the output into a dictionary,
1491 with key being the device name, and value being a dictionary of details
1492 of the device info.
1493
1494 @return: The dictionary of power_supply_info, e.g.,
1495 {'Line Power': {'online': 'yes', 'type': 'main'},
1496 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001497 @raise error.AutoservRunError if power_supply_info tool is not found in
1498 the DUT. Caller should handle this error to avoid false failure
1499 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001500 """
1501 result = self.run('power_supply_info').stdout.strip()
1502 info = {}
1503 device_name = None
1504 device_info = {}
1505 for line in result.split('\n'):
1506 pair = [v.strip() for v in line.split(':')]
1507 if len(pair) != 2:
1508 continue
1509 if pair[0] == 'Device':
1510 if device_name:
1511 info[device_name] = device_info
1512 device_name = pair[1]
1513 device_info = {}
1514 else:
1515 device_info[pair[0]] = pair[1]
1516 if device_name and not device_name in info:
1517 info[device_name] = device_info
1518 return info
1519
1520
1521 def get_battery_percentage(self):
1522 """Get the battery percentage.
1523
1524 @return: The percentage of battery level, value range from 0-100. Return
1525 None if the battery info cannot be retrieved.
1526 """
1527 try:
1528 info = self.get_power_supply_info()
1529 logging.info(info)
1530 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001531 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001532 return None
1533
1534
Philip Chenaf69ead2020-03-27 13:06:42 -07001535 def get_battery_state(self):
1536 """Get the battery charging state.
1537
1538 @return: A string representing the battery charging state. It can be
1539 'Charging', 'Fully charged', or 'Discharging'.
1540 """
1541 try:
1542 info = self.get_power_supply_info()
1543 logging.info(info)
1544 return info['Battery']['state']
1545 except (KeyError, ValueError, error.AutoservRunError):
1546 return None
1547
1548
Daniel Campello8ca25c22019-12-13 16:48:26 -07001549 def get_battery_display_percentage(self):
1550 """Get the battery display percentage.
1551
1552 @return: The display percentage of battery level, value range from
1553 0-100. Return None if the battery info cannot be retrieved.
1554 """
1555 try:
1556 info = self.get_power_supply_info()
1557 logging.info(info)
1558 return float(info['Battery']['display percentage'])
1559 except (KeyError, ValueError, error.AutoservRunError):
1560 return None
1561
1562
Dan Shi49ca0932014-11-14 11:22:27 -08001563 def is_ac_connected(self):
1564 """Check if the dut has power adapter connected and charging.
1565
1566 @return: True if power adapter is connected and charging.
1567 """
1568 try:
1569 info = self.get_power_supply_info()
1570 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001571 except (KeyError, error.AutoservRunError):
1572 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001573
1574
Simran Basi5e6339a2013-03-21 11:34:32 -07001575 def _cleanup_poweron(self):
1576 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001577 info = self.host_info_store.get()
1578 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001579 return
1580 logging.debug('This host has recently interacted with the RPM'
1581 ' Infrastructure. Ensuring power is on.')
1582 try:
1583 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001584 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001585 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001586 logging.error('Failed to turn Power On for this host after '
1587 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001588
1589 battery_percentage = self.get_battery_percentage()
Otabek Kasimov58e22562020-11-03 17:17:41 -08001590 if (
1591 battery_percentage
1592 and battery_percentage < cros_constants.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001593 raise
1594 elif self.is_ac_connected():
1595 logging.info('The device has power adapter connected and '
1596 'charging. No need to try to turn RPM on '
1597 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001598 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001599 logging.info('Battery level is now at %s%%. The device may '
1600 'still have enough power to run test, so no '
1601 'exception will be raised.', battery_percentage)
1602
Simran Basi5e6339a2013-03-21 11:34:32 -07001603
Garry Wangad4d4fd2019-01-30 17:00:38 -08001604 def _remove_rpm_changed_tag(self):
1605 info = self.host_info_store.get()
1606 del info.attributes[self._RPM_OUTLET_CHANGED]
1607 self.host_info_store.commit(info)
1608
1609
1610 def _add_rpm_changed_tag(self):
1611 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001612 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001613 self.host_info_store.commit(info)
1614
1615
1616
beepsc87ff602013-07-31 21:53:00 -07001617 def _is_factory_image(self):
1618 """Checks if the image on the DUT is a factory image.
1619
1620 @return: True if the image on the DUT is a factory image.
1621 False otherwise.
1622 """
1623 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1624 return result.exit_status == 0
1625
1626
1627 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001628 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001629
1630 @raises: FactoryImageCheckerException for factory images, since
1631 we cannot attempt to restart ui on them.
1632 error.AutoservRunError for any other type of error that
1633 occurs while restarting ui.
1634 """
1635 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001636 raise FactoryImageCheckerException('Cannot restart ui on factory '
1637 'images')
beepsc87ff602013-07-31 21:53:00 -07001638
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001639 # TODO(jrbarnette): The command to stop/start the ui job
1640 # should live inside cros_ui, too. However that would seem
1641 # to imply interface changes to the existing start()/restart()
1642 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001643 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001644 self.run('stop ui; start ui')
1645 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001646
1647
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001648 def _start_powerd_if_needed(self):
1649 """Start powerd if it isn't already running."""
1650 self.run('start powerd', ignore_status=True)
1651
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001652 def _read_arc_prop_file(self, filename):
1653 for path in [
1654 '/usr/share/arcvm/properties/', '/usr/share/arc/properties/'
1655 ]:
1656 if self.path_exists(path + filename):
1657 return utils.parse_cmd_output('cat ' + path + filename,
1658 run_method=self.run)
1659 return None
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001660
Jiyoun Hac172ee72020-12-15 08:57:29 +09001661 def _get_arc_build_info(self):
1662 """Returns a dictionary mapping build properties to their values."""
1663 build_info = None
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001664 for filename in ['build.prop', 'vendor_build.prop']:
1665 properties = self._read_arc_prop_file(filename)
1666 if properties:
1667 if build_info:
1668 build_info.update(properties)
1669 else:
1670 build_info = properties
1671 else:
1672 logging.error('Failed to find %s in device.', filename)
Jiyoun Hac172ee72020-12-15 08:57:29 +09001673 return build_info
1674
Jiyoun Haba37f312021-01-13 09:44:16 +09001675 def get_arc_primary_abi(self):
Jiyoun Hac172ee72020-12-15 08:57:29 +09001676 """Returns the primary abi of the host."""
1677 return self._get_arc_build_info().get('ro.product.cpu.abi')
1678
Jiyoun Haba37f312021-01-13 09:44:16 +09001679 def get_arc_security_patch(self):
Jiyoun Hac172ee72020-12-15 08:57:29 +09001680 """Returns the security patch of the host."""
1681 return self._get_arc_build_info().get('ro.build.version.security_patch')
1682
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001683 def get_arc_first_api_level(self):
1684 """Returns the security patch of the host."""
1685 return self._get_arc_build_info().get('ro.product.first_api_level')
1686
xixuana3bbc422017-05-04 15:57:21 -07001687 def _get_lsb_release_content(self):
1688 """Return the content of lsb-release file of host."""
1689 return self.run(
1690 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1691
1692
Dan Shi549fb822015-03-24 18:01:11 -07001693 def get_release_version(self):
1694 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1695
1696 @returns The version string in lsb-release, under attribute
1697 CHROMEOS_RELEASE_VERSION.
1698 """
Dan Shi549fb822015-03-24 18:01:11 -07001699 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001700 lsb_release_content=self._get_lsb_release_content())
1701
1702
Don Garrettb9f35802018-01-22 18:25:40 -08001703 def get_release_builder_path(self):
1704 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1705
1706 @returns The version string in lsb-release, under attribute
1707 CHROMEOS_RELEASE_BUILDER_PATH.
1708 """
1709 return lsbrelease_utils.get_chromeos_release_builder_path(
1710 lsb_release_content=self._get_lsb_release_content())
1711
1712
xixuana3bbc422017-05-04 15:57:21 -07001713 def get_chromeos_release_milestone(self):
1714 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1715 from lsb-release.
1716
1717 @returns The version string in lsb-release, under attribute
1718 CHROMEOS_RELEASE_BUILD_TYPE.
1719 """
1720 return lsbrelease_utils.get_chromeos_release_milestone(
1721 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001722
1723
1724 def verify_cros_version_label(self):
Garry Wangd18e7b32020-08-07 18:31:44 -07001725 """Verify if host's cros-version label match the actual image in dut.
Dan Shi549fb822015-03-24 18:01:11 -07001726
Garry Wangd18e7b32020-08-07 18:31:44 -07001727 @returns True if the label match with image in dut, otherwise False
Dan Shi549fb822015-03-24 18:01:11 -07001728 """
Garry Wangd18e7b32020-08-07 18:31:44 -07001729 os_from_host = self.get_release_builder_path()
1730 info = self.host_info_store.get()
1731 os_from_label = info.get_label_value(self.VERSION_PREFIX)
1732 if not os_from_label:
1733 logging.debug('No existing %s label detected', self.VERSION_PREFIX)
1734 return True
1735
1736 # known cases where the version label will not match the
1737 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1738 # * Tests for the `arc-presubmit` append "-cheetsth" to the label.
1739 if os_from_label.endswith(provision.CHEETS_SUFFIX):
1740 logging.debug('%s label with %s suffix detected, this suffix will'
1741 ' be ignored when comparing label.',
1742 self.VERSION_PREFIX, provision.CHEETS_SUFFIX)
1743 os_from_label = os_from_label[:-len(provision.CHEETS_SUFFIX)]
1744 logging.debug('OS version from host: %s; OS verision cached in '
1745 'label: %s', os_from_host, os_from_label)
1746 return os_from_label == os_from_host
Dan Shi549fb822015-03-24 18:01:11 -07001747
1748
Laurence Goodby778c9a42017-05-24 19:24:07 -07001749 def cleanup_services(self):
1750 """Reinitializes the device for cleanup.
1751
1752 Subclasses may override this to customize the cleanup method.
1753
1754 To indicate failure of the reset, the implementation may raise
1755 any of:
1756 error.AutoservRunError
1757 error.AutotestRunError
1758 FactoryImageCheckerException
1759
1760 @raises error.AutoservRunError
1761 @raises error.AutotestRunError
1762 @raises error.FactoryImageCheckerException
1763 """
1764 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001765 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001766
1767
Gregory Nisbetec615d62020-12-11 17:59:20 +00001768 def cleanup(self):
1769 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001770 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001771 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001772 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001773 except (error.AutotestRunError, error.AutoservRunError,
1774 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001775 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001776
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001777 # cleanup routines, i.e. reboot the machine.
Gregory Nisbetec615d62020-12-11 17:59:20 +00001778 super(CrosHost, self).cleanup()
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001779
Simran Basi5e6339a2013-03-21 11:34:32 -07001780 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001781 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001782 self._cleanup_poweron()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001783
Gregory Nisbetec615d62020-12-11 17:59:20 +00001784
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001785 def reboot(self, **dargs):
1786 """
1787 This function reboots the site host. The more generic
1788 RemoteHost.reboot() performs sync and sleeps for 5
1789 seconds. This is not necessary for Chrome OS devices as the
1790 sync should be finished in a short time during the reboot
1791 command.
1792 """
Gregory Nisbetec615d62020-12-11 17:59:20 +00001793 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001794 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001795 dargs['reboot_cmd'] = ('sleep 1; '
1796 'reboot & sleep %d; '
1797 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001798 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001799 if 'fastsync' not in dargs:
1800 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001801
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001802 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001803 # Record who called us
1804 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001805 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001806 'dut_host_name' : self.hostname,
1807 'success' : True}
1808 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001809 'caller' : "%s:%s" % (orig.co_filename,
1810 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001811 'success' : True,
1812 'error' : ''}
1813
Vincent Palatin80780b22016-07-27 16:02:37 +02001814 t0 = time.time()
1815 try:
1816 super(CrosHost, self).reboot(**dargs)
1817 except Exception as e:
1818 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001819 metric_debug_fields['success'] = False
1820 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001821 raise
1822 finally:
1823 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001824 metrics.Counter(
1825 'chromeos/autotest/autoserv/reboot_count').increment(
1826 fields=metric_fields)
1827 metrics.Counter(
1828 'chromeos/autotest/autoserv/reboot_debug').increment(
1829 fields=metric_debug_fields)
1830 metrics.SecondsDistribution(
1831 'chromeos/autotest/autoserv/reboot_duration').add(
1832 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001833
1834
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001835 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001836 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001837 """
1838 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001839
1840 @param suspend_time: How long to suspend as integer seconds.
1841 @param suspend_cmd: Suspend command to execute.
1842 @param allow_early_resume: If False and if device resumes before
1843 |suspend_time|, throw an error.
1844
1845 @exception AutoservSuspendError Host resumed earlier than
1846 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001847 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001848
1849 if suspend_cmd is None:
1850 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001851 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001852 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001853 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001854 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1855 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001856
1857
Simran Basiec564392014-08-25 16:48:09 -07001858 def upstart_status(self, service_name):
1859 """Check the status of an upstart init script.
1860
1861 @param service_name: Service to look up.
1862
1863 @returns True if the service is running, False otherwise.
1864 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001865 return 'start/running' in self.run('status %s' % service_name,
1866 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001867
Tom Hughese9552342018-12-18 14:29:25 -08001868 def upstart_stop(self, service_name):
1869 """Stops an upstart job if it's running.
1870
1871 @param service_name: Service to stop
1872
1873 @returns True if service has been stopped or was already stopped
1874 False otherwise.
1875 """
1876 if not self.upstart_status(service_name):
1877 return True
1878
1879 result = self.run('stop %s' % service_name, ignore_status=True)
1880 if result.exit_status != 0:
1881 return False
1882 return True
1883
1884 def upstart_restart(self, service_name):
1885 """Restarts (or starts) an upstart job.
1886
1887 @param service_name: Service to start/restart
1888
1889 @returns True if service has been started/restarted, False otherwise.
1890 """
1891 cmd = 'start'
1892 if self.upstart_status(service_name):
1893 cmd = 'restart'
1894 cmd = cmd + ' %s' % service_name
1895 result = self.run(cmd)
1896 if result.exit_status != 0:
1897 return False
1898 return True
Simran Basiec564392014-08-25 16:48:09 -07001899
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001900 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001901 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001902
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001903 Tests for the following conditions:
1904 1. All conditions tested by the parent version of this
1905 function.
1906 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001907 3. Sufficient space in /mnt/stateful_partition/encrypted.
1908 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001909
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001910 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001911 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001912 default_kilo_inodes_required = CONFIG.get_config_value(
1913 'SERVER', 'kilo_inodes_required', type=int, default=100)
1914 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1915 kilo_inodes_required = CONFIG.get_config_value(
1916 'SERVER', 'kilo_inodes_required_%s' % board,
1917 type=int, default=default_kilo_inodes_required)
1918 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001919 self.check_diskspace(
1920 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001921 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001922 'SERVER', 'gb_diskspace_required', type=float,
1923 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001924 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1925 # Not all targets build with encrypted stateful support.
1926 if self.path_exists(encrypted_stateful_path):
1927 self.check_diskspace(
1928 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001929 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001930 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1931 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001932
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001933 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001934
beepsc87ff602013-07-31 21:53:00 -07001935 # Factory images don't run update engine,
1936 # goofy controls dbus on these DUTs.
1937 if not self._is_factory_image():
1938 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001939
1940
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001941 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
Kuo Jen Wei50ef6e82021-03-29 12:04:47 +08001942 def wait_for_service(self, service_name):
1943 """Wait for target status of an upstart init script.
1944
1945 @param service_name: Service to wait for.
1946 """
1947 if not self.upstart_status(service_name):
1948 raise error.AutoservError('Service %s not running.' % service_name)
1949
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001950 def wait_for_system_services(self):
1951 """Waits for system-services to be running.
1952
1953 Sometimes, update_engine will take a while to update firmware, so we
1954 should give this some time to finish. See crbug.com/765686#c38 for
1955 details.
1956 """
Kuo Jen Wei50ef6e82021-03-29 12:04:47 +08001957 self.wait_for_service('system-services')
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001958
1959
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001960 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001961 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001962 message = 'Beginning verify for host %s board %s model %s'
1963 info = self.host_info_store.get()
1964 message %= (self.hostname, info.board, info.model)
1965 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001966 try:
1967 self._repair_strategy.verify(self)
1968 except hosts.AutoservVerifyDependencyError as e:
1969 # We don't want flag a DUT as failed if only non-critical
1970 # verifier(s) failed during the repair.
1971 if e.is_critical():
1972 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001973
1974
Fang Deng96667ca2013-08-01 17:46:18 -07001975 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001976 connect_timeout=None, alive_interval=None,
1977 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001978 """Override default make_ssh_command to use options tuned for Chrome OS.
1979
1980 Tuning changes:
1981 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1982 connection failure. Consistency with remote_access.sh.
1983
Samuel Tan2ce155b2015-06-23 18:24:38 -07001984 - ServerAliveInterval=900; which causes SSH to ping connection every
1985 900 seconds. In conjunction with ServerAliveCountMax ensures
1986 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001987 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001988 the test completed successfully. Later increased from 180 seconds to
1989 900 seconds to account for tests where the DUT is suspended for
1990 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001991
1992 - ServerAliveCountMax=3; consistency with remote_access.sh.
1993
1994 - ConnectAttempts=4; reduce flakiness in connection errors;
1995 consistency with remote_access.sh.
1996
1997 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1998 Host keys change with every new installation, don't waste
1999 memory/space saving them.
2000
2001 - SSH protocol forced to 2; needed for ServerAliveInterval.
2002
2003 @param user User name to use for the ssh connection.
2004 @param port Port on the target host to use for ssh connection.
2005 @param opts Additional options to the ssh command.
2006 @param hosts_file Ignored.
2007 @param connect_timeout Ignored.
2008 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08002009 @param alive_count_max Ignored.
2010 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07002011 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08002012 options = ' '.join([opts, '-o Protocol=2'])
2013 return super(CrosHost, self).make_ssh_command(
2014 user=user, port=port, opts=options, hosts_file='/dev/null',
2015 connect_timeout=30, alive_interval=900, alive_count_max=3,
2016 connection_attempts=4)
2017
2018
Jason Abeleb6f924f2013-11-13 16:01:54 -08002019 def syslog(self, message, tag='autotest'):
2020 """Logs a message to syslog on host.
2021
2022 @param message String message to log into syslog
2023 @param tag String tag prefix for syslog
2024
2025 """
2026 self.run('logger -t "%s" "%s"' % (tag, message))
2027
2028
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08002029 def _ping_check_status(self, status):
2030 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002031
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08002032 @param status Check the ping status against this value.
2033 @return True iff `status` and the result of ping are the same
2034 (i.e. both True or both False).
2035
2036 """
Abhishek Pandit-Subedi038df162020-09-14 16:37:43 -07002037 ping_val = utils.ping(self.hostname,
2038 tries=1,
2039 deadline=1,
2040 timeout=2,
2041 ignore_timeout=True)
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08002042 return not (status ^ (ping_val == 0))
2043
2044 def _ping_wait_for_status(self, status, timeout):
2045 """Wait for the host to have a given status (UP or DOWN).
2046
2047 Status is checked by polling. Polling will not last longer
2048 than the number of seconds in `timeout`. The polling
2049 interval will be long enough that only approximately
2050 _PING_WAIT_COUNT polling cycles will be executed, subject
2051 to a maximum interval of about one minute.
2052
2053 @param status Waiting will stop immediately if `ping` of the
2054 host returns this status.
2055 @param timeout Poll for at most this many seconds.
2056 @return True iff the host status from `ping` matched the
2057 requested status at the time of return.
2058
2059 """
2060 # _ping_check_status() takes about 1 second, hence the
2061 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07002062 # FIXME: if the ping command errors then _ping_check_status()
2063 # returns instantly. If timeout is also smaller than twice
2064 # _PING_WAIT_COUNT then the while loop below forks many
2065 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
2066 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
2067 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08002068 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
2069 end_time = time.time() + timeout
2070 while time.time() <= end_time:
2071 if self._ping_check_status(status):
2072 return True
2073 if poll_interval > 0:
2074 time.sleep(poll_interval)
2075
2076 # The last thing we did was sleep(poll_interval), so it may
2077 # have been too long since the last `ping`. Check one more
2078 # time, just to be sure.
2079 return self._ping_check_status(status)
2080
2081 def ping_wait_up(self, timeout):
2082 """Wait for the host to respond to `ping`.
2083
2084 N.B. This method is not a reliable substitute for
2085 `wait_up()`, because a host that responds to ping will not
2086 necessarily respond to ssh. This method should only be used
2087 if the target DUT can be considered functional even if it
2088 can't be reached via ssh.
2089
2090 @param timeout Minimum time to allow before declaring the
2091 host to be non-responsive.
2092 @return True iff the host answered to ping before the timeout.
2093
2094 """
Andrew Luo4be621d2020-03-21 07:01:13 -07002095 if self.use_icmp:
2096 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
2097 else:
2098 logging.debug('Using SSH instead of ICMP for ping_wait_up.')
2099 return self.wait_up(timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002100
Andrew Bresticker678c0c72013-01-22 10:44:09 -08002101 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002102 """Wait until the host no longer responds to `ping`.
2103
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08002104 This function can be used as a slightly faster version of
2105 `wait_down()`, by avoiding potentially long ssh timeouts.
2106
2107 @param timeout Minimum time to allow for the host to become
2108 non-responsive.
2109 @return True iff the host quit answering ping before the
2110 timeout.
2111
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002112 """
Andrew Luo4be621d2020-03-21 07:01:13 -07002113 if self.use_icmp:
2114 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
2115 else:
2116 logging.debug('Using SSH instead of ICMP for ping_wait_down.')
2117 return self.wait_down(timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002118
Anand K Mistry50f218e2020-07-31 14:50:15 +10002119 def _is_host_port_forwarded(self):
Garry Wanga2e78172020-09-09 23:49:07 -07002120 """Checks if the dut is connected over port forwarding.
Anand K Mistry50f218e2020-07-31 14:50:15 +10002121
2122 N.B. This method does not detect all situations where port forwarding is
2123 occurring. Namely, running autotest on the dut may result in a
2124 false-positive, and port forwarding using a different machine on the
2125 same network will be a false-negative.
2126
2127 @return True if the dut is connected over port forwarding
2128 False otherwise
2129 """
Garry Wanga2e78172020-09-09 23:49:07 -07002130 is_localhost = self.hostname in ['localhost', '127.0.0.1']
2131 is_forwarded = is_localhost and not self.is_default_port
2132 if is_forwarded:
2133 logging.info('Detected DUT connected by port forwarding')
2134 return is_forwarded
Anand K Mistry50f218e2020-07-31 14:50:15 +10002135
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002136 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002137 """Wait for the client to enter low-power sleep mode.
2138
2139 The test for "is asleep" can't distinguish a system that is
2140 powered off; to confirm that the unit was asleep, it is
2141 necessary to force resume, and then call
2142 `test_wait_for_resume()`.
2143
2144 This function is expected to be called from a test as part
2145 of a sequence like the following:
2146
2147 ~~~~~~~~
2148 boot_id = host.get_boot_id()
2149 # trigger sleep on the host
2150 host.test_wait_for_sleep()
2151 # trigger resume on the host
2152 host.test_wait_for_resume(boot_id)
2153 ~~~~~~~~
2154
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002155 @param sleep_timeout time limit in seconds to allow the host sleep.
2156
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002157 @exception TestFail The host did not go to sleep within
2158 the allowed time.
2159 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002160 if sleep_timeout is None:
2161 sleep_timeout = self.SLEEP_TIMEOUT
2162
Anand K Mistry50f218e2020-07-31 14:50:15 +10002163 # If the dut is accessed over SSH port-forwarding, `ping` is not useful
2164 # for detecting the dut is down since a ping to localhost will always
2165 # succeed. In this case, fall back to wait_down() which uses SSH.
2166 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07002167 success = self.wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002168 else:
Garry Wanga2e78172020-09-09 23:49:07 -07002169 success = self.ping_wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002170
2171 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002172 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002173 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002174
2175
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002176 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002177 """Wait for the client to resume from low-power sleep mode.
2178
2179 The `old_boot_id` parameter should be the value from
2180 `get_boot_id()` obtained prior to entering sleep mode. A
2181 `TestFail` exception is raised if the boot id changes.
2182
2183 See @ref test_wait_for_sleep for more on this function's
2184 usage.
2185
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002186 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002187 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002188 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002189
2190 @exception TestFail The host did not respond within the
2191 allowed time.
2192 @exception TestFail The host responded, but the boot id test
2193 indicated a reboot rather than a sleep
2194 cycle.
2195 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002196 if resume_timeout is None:
2197 resume_timeout = self.RESUME_TIMEOUT
2198
2199 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002200 raise error.TestFail(
2201 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002202 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002203 else:
2204 new_boot_id = self.get_boot_id()
2205 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002206 logging.error('client rebooted (old boot %s, new boot %s)',
2207 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002208 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002209 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002210
2211
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002212 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002213 """Wait for the client to shut down.
2214
2215 The test for "has shut down" can't distinguish a system that
2216 is merely asleep; to confirm that the unit was down, it is
2217 necessary to force boot, and then call test_wait_for_boot().
2218
2219 This function is expected to be called from a test as part
2220 of a sequence like the following:
2221
2222 ~~~~~~~~
2223 boot_id = host.get_boot_id()
2224 # trigger shutdown on the host
2225 host.test_wait_for_shutdown()
2226 # trigger boot on the host
2227 host.test_wait_for_boot(boot_id)
2228 ~~~~~~~~
2229
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002230 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002231 @exception TestFail The host did not shut down within the
2232 allowed time.
2233 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002234 if shutdown_timeout is None:
2235 shutdown_timeout = self.SHUTDOWN_TIMEOUT
2236
Anand K Mistry50f218e2020-07-31 14:50:15 +10002237 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07002238 success = self.wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002239 else:
Garry Wanga2e78172020-09-09 23:49:07 -07002240 success = self.ping_wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002241
2242 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002243 raise error.TestFail(
2244 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002245 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002246
2247
2248 def test_wait_for_boot(self, old_boot_id=None):
2249 """Wait for the client to boot from cold power.
2250
2251 The `old_boot_id` parameter should be the value from
2252 `get_boot_id()` obtained prior to shutting down. A
2253 `TestFail` exception is raised if the boot id does not
2254 change. The boot id test is omitted if `old_boot_id` is not
2255 specified.
2256
2257 See @ref test_wait_for_shutdown for more on this function's
2258 usage.
2259
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002260 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002261 shut down.
2262
2263 @exception TestFail The host did not respond within the
2264 allowed time.
2265 @exception TestFail The host responded, but the boot id test
2266 indicated that there was no reboot.
2267 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002268 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002269 raise error.TestFail(
2270 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002271 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002272 elif old_boot_id:
2273 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002274 logging.error('client not rebooted (boot %s)',
2275 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002276 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002277 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07002278
2279
2280 @staticmethod
2281 def check_for_rpm_support(hostname):
2282 """For a given hostname, return whether or not it is powered by an RPM.
2283
Simran Basi1df55112013-09-06 11:25:09 -07002284 @param hostname: hostname to check for rpm support.
2285
Simran Basid5e5e272012-09-24 15:23:59 -07002286 @return None if this host does not follows the defined naming format
2287 for RPM powered DUT's in the lab. If it does follow the format,
2288 it returns a regular expression MatchObject instead.
2289 """
Fang Dengbaff9082015-01-06 13:46:15 -08002290 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002291
2292
2293 def has_power(self):
2294 """For this host, return whether or not it is powered by an RPM.
2295
2296 @return True if this host is in the CROS lab and follows the defined
2297 naming format.
2298 """
Fang Deng0ca40e22013-08-27 17:47:44 -07002299 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002300
2301
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002302 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07002303 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002304
2305 @param state Specifies which power state to set to DUT
2306 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002307 use. By default "RPM" or "CCD" will be used based
2308 on servo type. Valid values from
2309 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002310
2311 """
2312 ACCEPTABLE_STATES = ['ON', 'OFF']
2313
Garry Wang5e5538a2019-04-08 15:36:18 -07002314 if not power_method:
2315 power_method = self.get_default_power_method()
2316
2317 state = state.upper()
2318 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002319 raise error.TestError('State must be one of: %s.'
2320 % (ACCEPTABLE_STATES,))
2321
2322 if power_method == self.POWER_CONTROL_SERVO:
2323 logging.info('Setting servo port J10 to %s', state)
2324 self.servo.set('prtctl3_pwren', state.lower())
2325 time.sleep(self._USB_POWER_TIMEOUT)
2326 elif power_method == self.POWER_CONTROL_MANUAL:
2327 logging.info('You have %d seconds to set the AC power to %s.',
2328 self._POWER_CYCLE_TIMEOUT, state)
2329 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07002330 elif power_method == self.POWER_CONTROL_CCD:
2331 servo_role = 'src' if state == 'ON' else 'snk'
2332 logging.info('servo ccd power pass through detected,'
2333 ' changing servo_role to %s.', servo_role)
2334 self.servo.set_servo_v4_role(servo_role)
2335 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07002336 # Make sure we don't leave DUT with no power(servo_role=snk)
2337 # when DUT is not pingable, as we raise a exception here
2338 # that may break a power cycle in the middle.
2339 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07002340 raise error.AutoservError(
2341 'DUT failed to regain network connection after %d seconds.'
2342 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002343 else:
2344 if not self.has_power():
2345 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002346 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07002347 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07002348
2349
Garry Wang5e5538a2019-04-08 15:36:18 -07002350 def power_off(self, power_method=None):
2351 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002352
2353 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002354 use. By default "RPM" or "CCD" will be used based
2355 on servo type. Valid values from
2356 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002357
2358 """
Derek Beckettb66e5c82020-08-12 15:31:02 -07002359 self._sync_if_up()
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002360 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002361
Derek Beckettb66e5c82020-08-12 15:31:02 -07002362 def _check_supported(self):
2363 """Throw an error if dts mode control is not supported."""
2364 if not self.servo_pwr_supported:
2365 raise error.TestFail('power_state controls not supported')
2366
2367 def _sync_if_up(self):
2368 """Run sync on the DUT and wait for completion if the DUT is up.
2369
2370 Additionally, try to sync and ignore status if its not up.
2371
2372 Useful prior to reboots to ensure files are written to disc.
2373
2374 """
2375 if self.is_up_fast():
2376 self.run("sync")
2377 return
2378 # If it is not up, attempt to sync in the rare event the DUT is up but
2379 # doesn't respond to a ping. Ignore any errors.
2380 try:
2381 self.run("sync", ignore_status=True, timeout=1)
2382 except Exception:
2383 pass
2384
2385 def power_off_via_servo(self):
2386 """Force the DUT to power off.
2387
2388 The DUT is guaranteed to be off at the end of this call,
2389 regardless of its previous state, provided that there is
2390 working EC and boot firmware. There is no requirement for
2391 working OS software.
2392
2393 """
2394 self._check_supported()
2395 self._sync_if_up()
2396 self.servo.set_nocheck('power_state', 'off')
2397
2398 def power_on_via_servo(self, rec_mode='on'):
2399 """Force the DUT to power on.
2400
2401 Prior to calling this function, the DUT must be powered off,
2402 e.g. with a call to `power_off()`.
2403
2404 At power on, recovery mode is set as specified by the
2405 corresponding argument. When booting with recovery mode on, it
2406 is the caller's responsibility to unplug/plug in a bootable
2407 external storage device.
2408
2409 If the DUT requires a delay after powering on but before
2410 processing inputs such as USB stick insertion, the delay is
2411 handled by this method; the caller is not responsible for such
2412 delays.
2413
2414 @param rec_mode Setting of recovery mode to be applied at
2415 power on. default: REC_OFF aka 'off'
2416
2417 """
2418 self._check_supported()
2419 self.servo.set_nocheck('power_state', rec_mode)
2420
2421 def reset_via_servo(self):
2422 """Force the DUT to reset.
2423
2424 The DUT is guaranteed to be on at the end of this call,
2425 regardless of its previous state, provided that there is
2426 working OS software. This also guarantees that the EC has
2427 been restarted.
2428
2429 """
2430 self._check_supported()
2431 self._sync_if_up()
2432 self.servo.set_nocheck('power_state', 'reset')
2433
Simran Basid5e5e272012-09-24 15:23:59 -07002434
Garry Wang5e5538a2019-04-08 15:36:18 -07002435 def power_on(self, power_method=None):
2436 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002437
2438 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002439 use. By default "RPM" or "CCD" will be used based
2440 on servo type. Valid values from
2441 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002442
2443 """
2444 self._set_power('ON', power_method)
2445
2446
Garry Wang5e5538a2019-04-08 15:36:18 -07002447 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002448 """Cycle power to this host by turning it OFF, then ON.
2449
2450 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002451 use. By default "RPM" or "CCD" will be used based
2452 on servo type. Valid values from
2453 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002454
2455 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002456 if not power_method:
2457 power_method = self.get_default_power_method()
2458
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002459 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002460 self.POWER_CONTROL_MANUAL,
2461 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002462 self.power_off(power_method=power_method)
2463 time.sleep(self._POWER_CYCLE_TIMEOUT)
2464 self.power_on(power_method=power_method)
2465 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002466 self._add_rpm_changed_tag()
2467 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002468
2469
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002470 def get_platform_from_fwid(self):
2471 """Determine the platform from the crossystem fwid.
2472
2473 @returns a string representing this host's platform.
2474 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002475 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002476 crossystem = utils.Crossystem(self)
2477 crossystem.init()
2478 # Extract fwid value and use the leading part as the platform id.
2479 # fwid generally follow the format of {platform}.{firmware version}
2480 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2481 platform = crossystem.fwid().split('.')[0].lower()
2482 # Newer platforms start with 'Google_' while the older ones do not.
2483 return platform.replace('google_', '')
2484
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002485
Simran Basic6f1f7a2012-10-16 10:47:46 -07002486 def get_platform(self):
2487 """Determine the correct platform label for this host.
2488
2489 @returns a string representing this host's platform.
2490 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002491 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2492 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002493 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002494 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002495 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002496 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002497
2498
Greg Edelstona7b05d12020-04-01 16:00:51 -06002499 def get_model_from_cros_config(self):
2500 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002501
Greg Edelstona7b05d12020-04-01 16:00:51 -06002502 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002503 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002504 return cros_config.call_cros_config_get_output('/ name',
2505 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002506
2507
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002508 def get_architecture(self):
2509 """Determine the correct architecture label for this host.
2510
2511 @returns a string representing this host's architecture.
2512 """
2513 crossystem = utils.Crossystem(self)
2514 crossystem.init()
2515 return crossystem.arch()
2516
2517
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002518 def get_chrome_version(self):
2519 """Gets the Chrome version number and milestone as strings.
2520
2521 Invokes "chrome --version" to get the version number and milestone.
2522
2523 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2524 current Chrome version number as a string (in the form "W.X.Y.Z")
2525 and "milestone" is the first component of the version number
2526 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2527 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2528 of "chrome --version" and the milestone will be the empty string.
2529
2530 """
MK Ryu35d661e2014-09-25 17:44:10 -07002531 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002532 return utils.parse_chrome_version(version_string)
2533
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002534
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002535 def get_ec_version(self):
2536 """Get the ec version as strings.
2537
2538 @returns a string representing this host's ec version.
2539 """
Jack Rosenthald8f92732023-01-31 21:14:04 -07002540 result = self.run("cat /dev/cros_ec", ignore_status=True)
2541 return utils.get_ec_version_from_chardev_contents(result.stdout)
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002542
2543
2544 def get_firmware_version(self):
2545 """Get the firmware version as strings.
2546
2547 @returns a string representing this host's firmware version.
2548 """
2549 crossystem = utils.Crossystem(self)
2550 crossystem.init()
2551 return crossystem.fwid()
2552
2553
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08002554 def get_hardware_id(self):
2555 """Get hardware id as strings.
2556
2557 @returns a string representing this host's hardware id.
2558 """
2559 crossystem = utils.Crossystem(self)
2560 crossystem.init()
2561 return crossystem.hwid()
2562
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002563 def get_hardware_revision(self):
2564 """Get the hardware revision as strings.
2565
2566 @returns a string representing this host's hardware revision.
2567 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002568 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002569 result = self.run(command, ignore_status=True)
2570 if result.exit_status != 0:
2571 return ''
2572 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002573
2574
2575 def get_kernel_version(self):
2576 """Get the kernel version as strings.
2577
2578 @returns a string representing this host's kernel version.
2579 """
2580 return self.run('uname -r').stdout.strip()
2581
2582
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002583 def get_cpu_name(self):
2584 """Get the cpu name as strings.
2585
2586 @returns a string representing this host's cpu name.
2587 """
2588
2589 # Try get cpu name from device tree first
2590 if self.path_exists('/proc/device-tree/compatible'):
2591 command = ' | '.join(
2592 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2593 'tail -1'])
2594 return self.run(command).stdout.strip().replace(',', ' ')
2595
2596 # Get cpu name from uname -p
2597 command = 'uname -p'
2598 ret = self.run(command).stdout.strip()
2599
2600 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2601 # Try get cpu name from /proc/cpuinfo instead
2602 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2603 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2604 self = self.run(command).stdout.strip()
2605
2606 # Remove bloat from CPU name, for example
2607 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2608 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2609 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2610 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2611 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2612 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2613
2614
2615 def get_screen_resolution(self):
2616 """Get the screen(s) resolution as strings.
2617 In case of more than 1 monitor, return resolution for each monitor
2618 separate with plus sign.
2619
2620 @returns a string representing this host's screen(s) resolution.
2621 """
2622 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2623 ret = self.run(command, ignore_status=True)
2624 # We might have Chromebox without a screen
2625 if ret.exit_status != 0:
2626 return ''
2627 return ret.stdout.strip().replace('\n', '+')
2628
2629
2630 def get_mem_total_gb(self):
2631 """Get total memory available in the system in GiB (2^20).
2632
2633 @returns an integer representing total memory
2634 """
2635 mem_total_kb = self.read_from_meminfo('MemTotal')
2636 kb_in_gb = float(2 ** 20)
2637 return int(round(mem_total_kb / kb_in_gb))
2638
2639
2640 def get_disk_size_gb(self):
2641 """Get size of disk in GB (10^9)
2642
2643 @returns an integer representing size of disk, 0 in Error Case
2644 """
2645 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2646 result = self.run(command, ignore_status=True)
2647 if result.exit_status != 0:
2648 return 0
2649 _, _, block, _ = re.split(r' +', result.stdout.strip())
2650 byte_per_block = 1024.0
2651 disk_kb_in_gb = 1e9
2652 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2653
2654
2655 def get_battery_size(self):
2656 """Get size of battery in Watt-hour via sysfs
2657
2658 This method assumes that battery support voltage_min_design and
2659 charge_full_design sysfs.
2660
2661 @returns a float representing Battery size, 0 if error.
2662 """
2663 # sysfs report data in micro scale
2664 battery_scale = 1e6
2665
2666 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2667 result = self.run(command, ignore_status=True)
2668 if result.exit_status != 0:
2669 return 0
2670 voltage = float(result.stdout.strip()) / battery_scale
2671
2672 command = 'cat /sys/class/power_supply/*/charge_full_design'
2673 result = self.run(command, ignore_status=True)
2674 if result.exit_status != 0:
2675 return 0
2676 amphereHour = float(result.stdout.strip()) / battery_scale
2677
2678 return voltage * amphereHour
2679
2680
2681 def get_low_battery_shutdown_percent(self):
2682 """Get the percent-based low-battery shutdown threshold.
2683
2684 @returns a float representing low-battery shutdown percent, 0 if error.
2685 """
2686 ret = 0.0
2687 try:
2688 command = 'check_powerd_config --low_battery_shutdown_percent'
2689 ret = float(self.run(command).stdout)
2690 except error.CmdError:
2691 logging.debug("Can't run %s", command)
2692 except ValueError:
2693 logging.debug("Didn't get number from %s", command)
2694
2695 return ret
2696
2697
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002698 def has_hammer(self):
2699 """Check whether DUT has hammer device or not.
2700
2701 @returns boolean whether device has hammer or not
2702 """
2703 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2704 return self.run(command, ignore_status=True).exit_status == 0
2705
2706
Niranjan Kumar34618872017-05-31 12:57:09 -07002707 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002708 """Returns True if the specified switch was provided to Chrome.
2709
2710 @param switch The chrome switch to search for.
2711 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002712
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002713 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2714 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002715
2716
2717 def oobe_triggers_update(self):
2718 """Returns True if this host has an OOBE flow during which
2719 it will perform an update check and perhaps an update.
2720 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2721 As more such flows are developed, code handling them needs
2722 to be added here.
2723
2724 @return Boolean indicating whether this host's OOBE triggers an update.
2725 """
2726 return self.is_chrome_switch_present(
2727 '--enterprise-enable-zero-touch-enrollment=hands-off')
2728
2729
Kevin Chenga2619dc2016-03-28 11:42:08 -07002730 # TODO(kevcheng): change this to just return the board without the
2731 # 'board:' prefix and fix up all the callers. Also look into removing the
2732 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002733 def get_board(self):
2734 """Determine the correct board label for this host.
2735
2736 @returns a string representing this host's board.
2737 """
2738 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2739 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002740 return (ds_constants.BOARD_PREFIX +
2741 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002742
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002743 def get_channel(self):
2744 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002745
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002746 @returns: a string represeting this host's build channel.
2747 (stable, dev, beta). None on fail.
2748 """
2749 return lsbrelease_utils.get_chromeos_channel(
2750 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002751
Kevin Chenga328da62016-03-31 10:49:04 -07002752 def get_power_supply(self):
2753 """
2754 Determine what type of power supply the host has
2755
2756 @returns a string representing this host's power supply.
2757 'power:battery' when the device has a battery intended for
2758 extended use
2759 'power:AC_primary' when the device has a battery not intended
2760 for extended use (for moving the machine, etc)
2761 'power:AC_only' when the device has no battery at all.
2762 """
Jack Rosenthal01ee2cf2021-03-30 21:01:32 -06002763 psu = self.run(command='cros_config /hardware-properties psu-type',
2764 ignore_status=True)
Kevin Chenga328da62016-03-31 10:49:04 -07002765 if psu.exit_status:
Jack Rosenthal01ee2cf2021-03-30 21:01:32 -06002766 # Assume battery if unspecified in cros_config.
Kevin Chenga328da62016-03-31 10:49:04 -07002767 return 'power:battery'
2768
2769 psu_str = psu.stdout.strip()
2770 if psu_str == 'unknown':
2771 return None
2772
2773 return 'power:%s' % psu_str
2774
2775
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002776 def has_battery(self):
2777 """Determine if DUT has a battery.
2778
2779 Returns:
2780 Boolean, False if known not to have battery, True otherwise.
2781 """
Jack Rosenthal01ee2cf2021-03-30 21:01:32 -06002782 return self.get_power_supply() == 'power:battery'
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002783
2784
Kevin Chenga328da62016-03-31 10:49:04 -07002785 def get_servo(self):
2786 """Determine if the host has a servo attached.
2787
2788 If the host has a working servo attached, it should have a servo label.
2789
2790 @return: string 'servo' if the host has servo attached. Otherwise,
2791 returns None.
2792 """
2793 return 'servo' if self._servo_host else None
2794
2795
Kevin Chenga328da62016-03-31 10:49:04 -07002796 def has_internal_display(self):
2797 """Determine if the device under test is equipped with an internal
2798 display.
2799
2800 @return: 'internal_display' if one is present; None otherwise.
2801 """
2802 from autotest_lib.client.cros.graphics import graphics_utils
2803 from autotest_lib.client.common_lib import utils as common_utils
2804
2805 def __system_output(cmd):
2806 return self.run(cmd).stdout
2807
2808 def __read_file(remote_path):
2809 return self.run('cat %s' % remote_path).stdout
2810
2811 # Hijack the necessary client functions so that we can take advantage
2812 # of the client lib here.
2813 # FIXME: find a less hacky way than this
2814 original_system_output = utils.system_output
2815 original_read_file = common_utils.read_file
2816 utils.system_output = __system_output
2817 common_utils.read_file = __read_file
2818 try:
2819 return ('internal_display' if graphics_utils.has_internal_display()
2820 else None)
2821 finally:
2822 utils.system_output = original_system_output
2823 common_utils.read_file = original_read_file
2824
2825
Dan Shi85276d42014-04-08 22:11:45 -07002826 def is_boot_from_usb(self):
2827 """Check if DUT is boot from USB.
2828
2829 @return: True if DUT is boot from usb.
2830 """
2831 device = self.run('rootdev -s -d').stdout.strip()
2832 removable = int(self.run('cat /sys/block/%s/removable' %
2833 os.path.basename(device)).stdout.strip())
2834 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002835
Otabek Kasimov77a40332020-10-20 15:40:03 -07002836 def is_boot_from_external_device(self):
2837 """Check if DUT is boot from external storage.
2838
2839 @return: True if DUT is boot from external storage.
2840 """
2841 boot_device = self.run('rootdev -s -d', ignore_status=True,
2842 timeout=60).stdout.strip()
2843 if not boot_device:
2844 logging.debug('Boot storage not detected on the host.')
2845 return False
2846 main_storage_cmd = ('. /usr/sbin/write_gpt.sh;'
2847 ' . /usr/share/misc/chromeos-common.sh;'
2848 ' load_base_vars; get_fixed_dst_drive')
2849 main_storage = self.run(main_storage_cmd,
2850 ignore_status=True,
2851 timeout=60).stdout.strip()
Otabek Kasimov723e8562020-12-08 13:29:34 -08002852 if not main_storage or boot_device != main_storage:
2853 logging.debug('Device booted from external storage storage.')
2854 return True
2855 logging.debug('Device booted from main storage.')
2856 return False
Helen Zhang17dae2b2014-11-11 09:25:52 -08002857
2858 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002859 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002860
2861 @param key: meminfo requested
2862
2863 @return the memory value as a string
2864
2865 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002866 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2867 logging.debug('%s', meminfo)
2868 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002869
2870
Rohit Makasana98e696f2016-06-03 18:48:10 -07002871 def get_cpu_arch(self):
2872 """Returns CPU arch of the device.
2873
2874 @return CPU architecture of the DUT.
2875 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002876 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002877 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2878 ignore_status=True).stdout:
2879 return 'x86_64'
2880 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2881 ignore_status=True).stdout:
2882 return 'arm'
2883 return 'i386'
2884
2885
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002886 def get_board_type(self):
2887 """
Nikolai Artemiev5b5b6802021-05-12 12:56:43 +10002888 Get the DUT's device type / form factor from cros_config. It can be one
2889 of CHROMEBOX, CHROMEBASE, CHROMEBOOK, or CHROMEBIT.
Danny Chan471a8d12015-08-18 14:57:41 -07002890
Nikolai Artemiev5b5b6802021-05-12 12:56:43 +10002891 @return form factor value from cros_config.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002892 """
Nikolai Artemiev5b5b6802021-05-12 12:56:43 +10002893
2894 device_type = self.run('cros_config /hardware-properties form-factor',
2895 ignore_status=True).stdout
2896 if device_type:
2897 return device_type
2898
2899 # TODO: remove lsb-release fallback once cros_config works everywhere
Danny Chan471a8d12015-08-18 14:57:41 -07002900 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2901 ignore_status=True).stdout
2902 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002903 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002904 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002905
2906
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002907 def get_arc_version(self):
2908 """Return ARC version installed on the DUT.
2909
2910 @returns ARC version as string if the CrOS build has ARC, else None.
2911 """
2912 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2913 ignore_status=True).stdout
2914 if arc_version:
2915 return arc_version.split('=')[-1].strip()
2916 return None
2917
2918
Gilad Arnolda76bef02015-09-29 13:55:15 -07002919 def get_os_type(self):
2920 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002921
2922
Kevin Chenga2619dc2016-03-28 11:42:08 -07002923 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002924 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002925 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002926
2927
2928 def get_default_power_method(self):
2929 """
2930 Get the default power method for power_on/off/cycle() methods.
2931 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2932 """
2933 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002934 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002935 if self.servo and self.servo.supports_built_in_pd_control():
2936 self._default_power_method = self.POWER_CONTROL_CCD
2937 else:
2938 logging.debug('Either servo is unitialized or the servo '
2939 'setup does not support pd controls. Falling '
2940 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002941 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002942
2943
2944 def find_usb_devices(self, idVendor, idProduct):
2945 """
2946 Get usb device sysfs name for specific device.
2947
2948 @param idVendor Vendor ID to search in sysfs directory.
2949 @param idProduct Product ID to search in sysfs directory.
2950
2951 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2952 """
2953 # Look for matching file and cut at position 7 to get dir name.
2954 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2955
2956 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2957 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2958
2959 # Use uniq -d to print duplicate line from both command
2960 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2961
2962 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2963
2964
2965 def bind_usb_device(self, usb_node):
2966 """
2967 Bind usb device
2968
2969 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2970 """
2971 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2972 self.run(cmd, ignore_status=True)
2973
2974
2975 def unbind_usb_device(self, usb_node):
2976 """
2977 Unbind usb device
2978
2979 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2980 """
2981 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2982 self.run(cmd, ignore_status=True)
2983
2984
2985 def get_wlan_ip(self):
2986 """
2987 Get ip address of wlan interface.
2988
2989 @return ip address of wlan or empty string if wlan is not connected.
2990 """
2991 cmds = [
2992 'iw dev', # List wlan physical device
2993 'grep Interface', # Grep only interface name
2994 'cut -f 2 -d" "', # Cut the name part
2995 'xargs ifconfig', # Feed it to ifconfig to get ip
2996 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2997 'cut -f 2 -d " "' # Cut the ip part
2998 ]
2999 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08003000
3001 def connect_to_wifi(self, ssid, passphrase=None, security=None):
3002 """
3003 Connect to wifi network
3004
3005 @param ssid SSID of the wifi network.
3006 @param passphrase Passphrase of the wifi network. None if not existed.
3007 @param security Security of the wifi network. Default to "psk" if
3008 passphase is given without security. Possible values
3009 are "none", "psk", "802_1x".
3010
3011 @return True if succeed, False if not.
3012 """
3013 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
3014 if passphrase:
3015 cmd += ' ' + passphrase
3016 if security:
3017 cmd += ' ' + security
3018 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07003019
3020 def get_device_repair_state(self):
3021 """Get device repair state"""
3022 return self._device_repair_state
3023
Otabek Kasimov44273d22021-02-26 17:13:24 -08003024 def is_marked_for_replacement(self):
3025 """Verify if device was marked for replacemnet during admin task."""
3026 expected_state = cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT
3027 return self.get_device_repair_state() == expected_state
3028
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07003029 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07003030 """Set device repair state.
3031
3032 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07003033 file in result directory. The file will be read by Lucifer. The
3034 file will not be created if result directory not specified.
3035
3036 @params state: The new state for the device.
3037 @params resultdir: The path to result directory. If path not provided
3038 will be attempt to get retrieve it from job
3039 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07003040 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07003041 resultdir = resultdir or getattr(self.job, 'resultdir', '')
3042 if resultdir:
3043 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07003044 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07003045 logging.info('Set device state as %s. '
3046 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07003047 else:
3048 logging.debug('Cannot write the device state due missing info '
3049 'about result dir.')
3050 self._device_repair_state = state
3051
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07003052 def set_device_needs_replacement(self, resultdir=None):
3053 """Set device as required replacement.
3054
3055 @params resultdir: The path to result directory. If path not provided
3056 will be attempt to get retrieve it from job
3057 if present.
3058 """
3059 self.set_device_repair_state(
3060 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
3061 resultdir=resultdir)
3062
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08003063 def _dut_is_accessible_by_verifier(self):
3064 """Check if DUT accessible by SSH or PING verifier.
Otabek Kasimov86062d02020-11-17 13:30:22 -08003065
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08003066 @returns: bool, True - verifier marked as success.
3067 False - result not reachable, verifier did not success.
Otabek Kasimov86062d02020-11-17 13:30:22 -08003068 """
3069 if not self._repair_strategy:
3070 return False
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08003071 dut_ssh = self._repair_strategy.verifier_is_good('ssh')
3072 dut_ping = self._repair_strategy.verifier_is_good('ping')
3073 return dut_ssh == hosts.VERIFY_SUCCESS or dut_ssh == hosts.VERIFY_SUCCESS
Otabek Kasimov86062d02020-11-17 13:30:22 -08003074
Otabek Kasimovd48389b2020-12-07 02:38:34 -08003075 def _stat_if_pingable_but_not_sshable(self):
3076 """Check if DUT pingable but failed SSH verifier."""
3077 if not self._repair_strategy:
3078 return
3079 dut_ssh = self._repair_strategy.verifier_is_good('ssh')
3080 dut_ping = self._repair_strategy.verifier_is_good('ping')
3081 if (dut_ping == hosts.VERIFY_FAILED
3082 and dut_ssh == hosts.VERIFY_FAILED):
3083 metrics.Counter('chromeos/autotest/dut_pingable_no_ssh').increment(
3084 fields={'host': self.hostname})
3085
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07003086 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07003087 """Check if device require manual attention to be fixed.
3088
3089 The state 'needs_manual_repair' can be set when auto repair cannot
3090 fix the device due hardware or cable issues.
3091 """
3092 # ignore the logic if state present
3093 # state can be set by any cros repair actions
Otabek Kasimov86062d02020-11-17 13:30:22 -08003094 if self.get_device_repair_state():
Otabek Kasimov6825b762020-06-23 23:42:44 -07003095 return
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08003096 if self._dut_is_accessible_by_verifier():
3097 # DUT is accessible and we still have many options to repair it.
Otabek Kasimovc9812582020-10-08 18:52:52 -07003098 return
Otabek Kasimov9189ede2020-11-09 14:08:58 -08003099 needs_manual_repair = False
3100 dhp = self.health_profile
Otabek Kasimov69253822020-11-24 10:52:27 -08003101 if dhp and dhp.get_repair_fail_count() > 49:
3102 # 42 = 6 times during 7 days. (every 4 hour repair)
3103 # round up to 50 in case somebody will run some attempt on it.
Otabek Kasimovc9812582020-10-08 18:52:52 -07003104 logging.info(
Otabek Kasimov9189ede2020-11-09 14:08:58 -08003105 'DUT is not sshable and fail %s times.'
Otabek Kasimov69253822020-11-24 10:52:27 -08003106 ' Limit to try repair is 50 times',
Otabek Kasimov9189ede2020-11-09 14:08:58 -08003107 dhp.get_repair_fail_count())
3108 needs_manual_repair = True
3109
3110 if not needs_manual_repair:
3111 # We cannot ssh to the DUT and we have hardware or set-up issues
3112 # with servo then we need request manual repair for the DUT.
3113 servo_state_required_manual_fix = [
3114 servo_constants.SERVO_STATE_DUT_NOT_CONNECTED,
3115 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
3116 ]
3117 if self.get_servo_state() in servo_state_required_manual_fix:
3118 logging.info(
3119 'DUT required manual repair because it is not sshable'
3120 ' and possible have setup issue with Servo. Please'
3121 ' verify all connections and present of devices.')
3122 needs_manual_repair = True
3123
3124 if needs_manual_repair:
Otabek Kasimovc9812582020-10-08 18:52:52 -07003125 self.set_device_repair_state(
3126 cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)
Otabek Kasimov42506d02020-07-29 14:44:57 -07003127
Otabek Kasimov86062d02020-11-17 13:30:22 -08003128 def _reboot_labstation_if_needed(self):
3129 """Place request to reboot the labstation if DUT is not sshable.
3130
3131 @returns: None
3132 """
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003133 message_prefix = "Don't need to request servo-host reboot"
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08003134 if self._dut_is_accessible_by_verifier():
Otabek Kasimov86062d02020-11-17 13:30:22 -08003135 return
3136 if not self._servo_host:
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003137 logging.debug('%s as it not initialized', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003138 return
3139 if not self._servo_host.is_up_fast():
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003140 logging.debug('%s as servo-host is not sshable', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003141 return
3142 if not self._servo_host.is_labstation():
3143 logging.debug('Servo_v3 is not requested to reboot for the DUT')
3144 return
3145 usb_path = self._servo_host.get_main_servo_usb_path()
3146 if usb_path:
3147 connected_port = os.path.basename(os.path.normpath(usb_path))
3148 # Directly connected servo to the labstation looks like '1-5.3'
3149 # and when connected by hub - '1-5.2.3' or '1-5.2.1.3'. Where:
3150 # - '1-5' - port on labstation
3151 # - '2' or '2.1' - port on the hub or smart-hub
3152 # - '3' - port on servo hub
3153 if len(connected_port.split('.')) > 2:
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003154 logging.debug('%s as servo connected by hub', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003155 return
3156 self._servo_host.request_reboot()
3157 logging.info('Requested labstation reboot because DUT is not sshable')
3158
Otabek Kasimov42506d02020-07-29 14:44:57 -07003159 def is_file_system_writable(self, testdirs=None):
3160 """Check is the file systems are writable.
3161
3162 The standard linux response to certain unexpected file system errors
3163 (including hardware errors in block devices) is to change the file
3164 system status to read-only. This checks that that hasn't happened.
3165
3166 @param testdirs: List of directories to check. If no data provided
3167 then '/mnt/stateful_partition' and '/var/tmp'
3168 directories will be checked.
3169
3170 @returns boolean whether file-system writable.
3171 """
3172 def _check_dir(testdir):
3173 # check if we can create a file
3174 filename = os.path.join(testdir, 'writable_my_test_file')
3175 command = 'touch %s && rm %s' % (filename, filename)
3176 rv = self.run(command=command,
3177 timeout=30,
3178 ignore_status=True)
3179 is_writable = rv.exit_status == 0
3180 if not is_writable:
3181 logging.info('Cannot create a file in "%s"!'
3182 ' Probably the FS is read-only', testdir)
3183 logging.info("FileSystem is not writable!")
3184 return False
3185 return True
3186
3187 if not testdirs or len(testdirs) == 0:
3188 # N.B. Order matters here: Encrypted stateful is loop-mounted
3189 # from a file in unencrypted stateful, so we don't test for
3190 # errors in encrypted stateful if unencrypted fails.
3191 testdirs = ['/mnt/stateful_partition', '/var/tmp']
3192
3193 for dir in testdirs:
3194 # loop will be stopped if any directory fill fail the check
3195 try:
3196 if not _check_dir(dir):
3197 return False
3198 except Exception as e:
3199 # here expected only timeout error, all other will
3200 # be catch by 'ignore_status=True'
3201 logging.debug('Fail to check %s to write in it', dir)
3202 return False
3203 return True
Garry Wang1a493d82020-08-31 21:01:19 -07003204
Dana Goyettec172b172020-07-29 16:26:15 -07003205 def blocking_sync(self, freeze_for_reset=False):
3206 """Sync root device and internal device, via script.
3207
3208 The actual calls end up logged by the run() call, since they're printed
3209 to stdout/stderr in the script.
3210
3211 @param freeze_for_reset: if True, prepare for reset by blocking writes
3212 (only if enable_fs_sync_fsfreeze=True)
3213 """
3214
3215 if freeze_for_reset and self.USE_FSFREEZE:
3216 logging.info('Blocking sync and freeze')
3217 elif freeze_for_reset:
3218 logging.info('Blocking sync for reset')
3219 else:
3220 logging.info('Blocking sync')
3221
3222 # client/bin is installed on the DUT as /usr/local/autotest/bin
3223 sync_cmd = '/usr/local/autotest/bin/fs_sync.py'
3224 if freeze_for_reset and self.USE_FSFREEZE:
3225 sync_cmd += ' --freeze'
3226 return self.run(sync_cmd)
3227
Garry Wanga2e78172020-09-09 23:49:07 -07003228 def set_health_profile_dut_state(self, state):
3229 if not self.health_profile:
3230 logging.debug('Device health profile is not initialized, skip'
3231 ' set dut state.')
3232 return
3233 reset_counters = state in profile_constants.STATES_NEED_RESET_COUNTER
3234 self.health_profile.update_dut_state(state, reset_counters)
Garry Wang53fc8f32020-09-18 13:30:08 -07003235
3236 def require_snk_mode_in_recovery(self):
3237 """Check whether we need to switch servo_v4 role to snk when
3238 booting into recovery mode. (See crbug.com/1129165)
3239 """
Garry Wanga8739cc2020-10-30 00:49:23 -07003240 has_battery = True
3241 # Determine if the host has battery based on host_info first.
3242 power_info = self.host_info_store.get().get_label_value('power')
3243 if power_info:
3244 has_battery = power_info == 'battery'
3245 elif self.is_up_fast():
3246 # when running local tests host_info is not available, so we
3247 # need to determine whether the host has battery by checking
3248 # from host side.
3249 logging.debug('Label `power` is not found in host_info, checking'
3250 ' if the host has battery from host side.')
3251 has_battery = self.has_battery()
3252
3253 if not has_battery:
Garry Wang53fc8f32020-09-18 13:30:08 -07003254 logging.info(
3255 '%s does not has battery, snk mode is not needed'
3256 ' for recovery.', self.hostname)
3257 return False
Garry Wanga8739cc2020-10-30 00:49:23 -07003258
Garry Wang53fc8f32020-09-18 13:30:08 -07003259 if not self.servo.supports_built_in_pd_control():
3260 logging.info('Power delivery is not supported on this servo, snk'
3261 ' mode is not needed for recovery.')
3262 return False
3263 try:
Garry Wang53fc8f32020-09-18 13:30:08 -07003264 battery_percent = self.servo.get('battery_charge_percent')
Otabek Kasimov58e22562020-11-03 17:17:41 -08003265 if battery_percent < cros_constants.MIN_BATTERY_LEVEL:
Garry Wang53fc8f32020-09-18 13:30:08 -07003266 logging.info(
3267 'Current battery level %s%% below %s%% threshold, we'
3268 ' will attempt to boot host in recovery mode without'
3269 ' changing servo to snk mode. Please note the host may'
3270 ' not able to see usb drive in recovery mode later due'
3271 ' to servo not in snk mode.', battery_percent,
Otabek Kasimov58e22562020-11-03 17:17:41 -08003272 cros_constants.MIN_BATTERY_LEVEL)
Garry Wang53fc8f32020-09-18 13:30:08 -07003273 return False
3274 except Exception as e:
3275 logging.info(
3276 'Unexpected error occurred when getting'
3277 ' battery_charge_percent from servo; %s', str(e))
3278 return False
3279 return True
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003280
3281 def _set_servo_topology(self):
3282 """Set servo-topology info to the host-info."""
3283 logging.debug('Try to save servo topology to host-info.')
3284 if not self._servo_host:
Greg Edelstonff2665d2021-04-21 14:32:27 -06003285 logging.debug('Servo host is not initialized.')
Otabek Kasimovfe41e2d2021-02-14 20:48:52 -08003286 return
3287 if not self.is_servo_in_working_state():
3288 logging.debug('Is servo is not in working state then'
3289 ' update topology is not allowed.')
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003290 return
3291 if not self._servo_host.is_servo_topology_supported():
Otabek Kasimovfe41e2d2021-02-14 20:48:52 -08003292 logging.debug('Servo-topology is not supported.')
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003293 return
3294 servo_topology = self._servo_host.get_topology()
3295 if not servo_topology or servo_topology.is_empty():
Otabek Kasimovfe41e2d2021-02-14 20:48:52 -08003296 logging.debug('Servo topology is empty')
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003297 return
3298 servo_topology.save(self.host_info_store)