blob: 7fb9bb042349c054f2147dbde9043e743ee3e3aa [file] [log] [blame]
Derek Beckettf73baca2020-08-19 15:08:47 -07001# Lint as: python2, python3
J. Richard Barnette24adbf42012-04-11 15:04:53 -07002# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Dale Curtisaa5eedb2011-08-23 16:18:52 -07003# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
Derek Beckettf73baca2020-08-19 15:08:47 -07006from __future__ import absolute_import
7from __future__ import division
8from __future__ import print_function
9
J. Richard Barnette1d78b012012-05-15 13:56:30 -070010import logging
Dan Shi0f466e82013-02-22 15:44:58 -080011import os
Simran Basid5e5e272012-09-24 15:23:59 -070012import re
Vincent Palatindf2372c2016-10-07 17:03:00 +020013import sys
Otabek Kasimovb7cb8422020-12-23 02:38:32 -080014import six
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070015import time
16
mussa584b4462014-06-20 15:13:28 -070017import common
J. Richard Barnette45e93de2012-04-11 17:24:15 -070018from autotest_lib.client.bin import utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070019from autotest_lib.client.common_lib import autotemp
Richard Barnette0c73ffc2012-11-19 15:21:18 -080020from autotest_lib.client.common_lib import error
21from autotest_lib.client.common_lib import global_config
J. Richard Barnette91137f02016-03-10 16:52:26 -080022from autotest_lib.client.common_lib import hosts
Dan Shi549fb822015-03-24 18:01:11 -070023from autotest_lib.client.common_lib import lsbrelease_utils
Otabek Kasimov6825b762020-06-23 23:42:44 -070024from autotest_lib.client.common_lib import utils as common_utils
Greg Edelstona7b05d12020-04-01 16:00:51 -060025from autotest_lib.client.common_lib.cros import cros_config
Richard Barnette03a0c132012-11-05 12:40:35 -080026from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070027from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000028from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080029from autotest_lib.client.cros import cros_ui
Dan Shia1ecd5c2013-06-06 11:21:31 -070030from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070031from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050032from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Derek Beckettcc0c8302021-07-14 09:53:48 -070033from autotest_lib.server.cros.dynamic_suite import tools
Garry Wang1a493d82020-08-31 21:01:19 -070034from autotest_lib.server.cros.device_health_profile import device_health_profile
Garry Wanga2e78172020-09-09 23:49:07 -070035from autotest_lib.server.cros.device_health_profile import profile_constants
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070036from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070037from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070038from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080039from autotest_lib.server.hosts import chameleon_host
Otabek Kasimov832d9162020-07-27 19:24:57 -070040from autotest_lib.server.hosts import cros_constants
Richard Barnetted31580e2018-05-14 19:58:00 +000041from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080042from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070043from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070044from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070045from autotest_lib.server.hosts import servo_constants
Otabek Kasimov808cd832020-05-28 18:27:46 -070046from autotest_lib.site_utils.admin_audit import constants as audit_const
Otabek Kasimov27bb2862020-08-10 14:40:45 -070047from autotest_lib.site_utils.admin_audit import verifiers as audit_verify
Derek Beckettf73baca2020-08-19 15:08:47 -070048from six.moves import zip
Simran Basid5e5e272012-09-24 15:23:59 -070049
Andrew Luo4be621d2020-03-21 07:01:13 -070050
Dan Shib8540a52015-07-16 14:18:23 -070051CONFIG = global_config.global_config
52
beepsc87ff602013-07-31 21:53:00 -070053class FactoryImageCheckerException(error.AutoservError):
54 """Exception raised when an image is a factory image."""
55 pass
56
57
Fang Deng0ca40e22013-08-27 17:47:44 -070058class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070059 """Chromium OS specific subclass of Host."""
60
Simran Basi5ace6f22016-01-06 17:30:44 -080061 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
62
J. Richard Barnette45e93de2012-04-11 17:24:15 -070063
Richard Barnette03a0c132012-11-05 12:40:35 -080064 # Timeout values (in seconds) associated with various Chrome OS
65 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070066 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080067 # In general, a good rule of thumb is that the timeout can be up
68 # to twice the typical measured value on the slowest platform.
69 # The times here have not necessarily been empirically tested to
70 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070071 #
72 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080073 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
74 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080075 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070076 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080077 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070078 # screen delay, time to start the network on the DUT, and the
79 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070080 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080081 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080082 # network.
beepsf079cfb2013-09-18 17:49:51 -070083 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
Otabek Kasimovaeb47fe2021-01-26 20:53:55 -080084 # ADMIN_INSTALL_TIMEOUT: Time to allow for chromeos-install
85 # used by admin tasks.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080086 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
87 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070088
89 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080090 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080091 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070092 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070093 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070094 INSTALL_TIMEOUT = 480
Otabek Kasimovaeb47fe2021-01-26 20:53:55 -080095 ADMIN_INSTALL_TIMEOUT = 600
Dan Shi2c88eed2013-11-12 10:18:38 -080096 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070097
Dan Shica503482015-03-30 17:23:25 -070098 # Minimum OS version that supports server side packaging. Older builds may
99 # not have server side package built or with Autotest code change to support
100 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -0700101 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -0700102 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700103
Dana Goyettec172b172020-07-29 16:26:15 -0700104 USE_FSFREEZE = CONFIG.get_config_value(
Dana Goyette6242cb32020-09-23 11:02:57 -0700105 'CROS', 'enable_fs_freeze', type=bool, default=False)
Dana Goyettec172b172020-07-29 16:26:15 -0700106
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800107 # REBOOT_TIMEOUT: How long to wait for a reboot.
108 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700109 # We have a long timeout to ensure we don't flakily fail due to other
110 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700111 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
112 # return from reboot' bug is solved.
113 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700114
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800115 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
116 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700117 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
118 # since changing servo role will reset USB state
119 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800120 _USB_POWER_TIMEOUT = 5
121 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700122 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800123
Fang Dengdeba14f2014-11-14 11:54:09 -0800124 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
125 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700126
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800127 # Constants used in ping_wait_up() and ping_wait_down().
128 #
129 # _PING_WAIT_COUNT is the approximate number of polling
130 # cycles to use when waiting for a host state change.
131 #
132 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
133 # for arguments to the internal _ping_wait_for_status()
134 # method.
135 _PING_WAIT_COUNT = 40
136 _PING_STATUS_DOWN = False
137 _PING_STATUS_UP = True
138
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800139 # Allowed values for the power_method argument.
140
Garry Wang5e5538a2019-04-08 15:36:18 -0700141 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
142 # DUTs except those with servo_v4 CCD.
143 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
144 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800145 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
146 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
147 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700148 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800149 POWER_CONTROL_SERVO = 'servoj10'
150 POWER_CONTROL_MANUAL = 'manual'
151
152 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700153 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800154 POWER_CONTROL_SERVO,
155 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800156
Simran Basi5e6339a2013-03-21 11:34:32 -0700157 _RPM_OUTLET_CHANGED = 'outlet_changed'
158
Dan Shi9cb0eec2014-06-03 09:04:50 -0700159 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700160 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700161 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700162
Brent Peterson1cb623a2020-01-09 13:14:28 -0800163 # Regular expression for extracting EC version string
164 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
165
166 # Regular expression for extracting BIOS version string
167 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
168
Brent Petersonc70a1832020-01-24 15:54:35 -0800169 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700170 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800171
J. Richard Barnette964fba02012-10-24 17:34:29 -0700172 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800173 def check_host(host, timeout=10):
174 """
175 Check if the given host is a chrome-os host.
176
177 @param host: An ssh host representing a device.
178 @param timeout: The timeout for the run command.
179
180 @return: True if the host device is chromeos.
181
beeps46dadc92013-11-07 14:07:10 -0800182 """
183 try:
Allen Liad719c12017-06-27 23:48:04 +0000184 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700185 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700186 '! grep -q moblab /etc/lsb-release && '
Derek Beckett342e3e62021-01-05 17:17:23 -0800187 '! grep -q labstation /etc/lsb-release &&'
188 ' grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
189 ignore_status=True,
Laurence Goodby468de252017-06-08 17:22:53 -0700190 timeout=timeout).stdout
Derek Beckett342e3e62021-01-05 17:17:23 -0800191 if result:
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800192 return not (
193 lsbrelease_utils.is_jetstream(
Derek Beckett342e3e62021-01-05 17:17:23 -0800194 lsb_release_content=result) or
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800195 lsbrelease_utils.is_gce_board(
Derek Beckett342e3e62021-01-05 17:17:23 -0800196 lsb_release_content=result))
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800197
beeps46dadc92013-11-07 14:07:10 -0800198 except (error.AutoservRunError, error.AutoservSSHTimeout):
199 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700200
201 return False
beeps46dadc92013-11-07 14:07:10 -0800202
203
204 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800205 def get_chameleon_arguments(args_dict):
206 """Extract chameleon options from `args_dict` and return the result.
207
208 Recommended usage:
209 ~~~~~~~~
210 args_dict = utils.args_to_dict(args)
211 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
212 host = hosts.create_host(machine, chameleon_args=chameleon_args)
213 ~~~~~~~~
214
215 @param args_dict Dictionary from which to extract the chameleon
216 arguments.
217 """
Sam McNally66594ca2019-12-09 12:45:44 +1100218 chameleon_args = {key: args_dict[key]
219 for key in ('chameleon_host', 'chameleon_port')
220 if key in args_dict}
221 if 'chameleon_ssh_port' in args_dict:
222 chameleon_args['port'] = int(args_dict['chameleon_ssh_port'])
223 return chameleon_args
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800224
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800225 @staticmethod
226 def get_btpeer_arguments(args_dict):
227 """Extract btpeer options from `args_dict` and return the result.
228
229 This is used to parse details of Bluetooth peer.
230 Recommended usage:
231 ~~~~~~~~
232 args_dict = utils.args_to_dict(args)
233 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700234 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800235 ~~~~~~~~
236
237 @param args_dict: Dictionary from which to extract the btpeer
238 arguments.
239 """
240 if 'btpeer_host_list' in args_dict:
241 result = []
242 for btpeer in args_dict['btpeer_host_list'].split(','):
Claire Changd0b19842020-11-04 22:28:45 +0800243 # IPv6 addresses including a port number should be enclosed in
244 # square brackets.
245 delimiter = ']:' if re.search(r':.*:', btpeer) else ':'
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800246 result.append({key: value for key,value in
247 zip(('btpeer_host','btpeer_port'),
Claire Changd0b19842020-11-04 22:28:45 +0800248 btpeer.strip('[]').split(delimiter))})
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800249 return result
250 else:
Anand K Mistrye8933092020-08-05 14:49:41 +1000251 return {key: args_dict[key]
252 for key in ('btpeer_host', 'btpeer_port', 'btpeer_ssh_port')
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800253 if key in args_dict}
254
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800255
256 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700257 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800258 """Extract chameleon options from `args_dict` and return the result.
259
260 Recommended usage:
261 ~~~~~~~~
262 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700263 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
264 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800265 ~~~~~~~~
266
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700267 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800268 arguments.
269 """
Allen Li083866b2016-08-18 10:07:10 -0700270 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700271 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700272 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800273
274
275 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800276 def get_servo_arguments(args_dict):
277 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800278
279 Recommended usage:
280 ~~~~~~~~
281 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700282 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800283 host = hosts.create_host(machine, servo_args=servo_args)
284 ~~~~~~~~
285
286 @param args_dict Dictionary from which to extract the servo
287 arguments.
288 """
Garry Wang11b5e872020-03-11 15:14:08 -0700289 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
Andrew Luo4be621d2020-03-21 07:01:13 -0700290 servo_constants.SERVO_HOST_SSH_PORT_ATTR,
Garry Wang11b5e872020-03-11 15:14:08 -0700291 servo_constants.SERVO_PORT_ATTR,
Otabek Kasimov382c3bb2020-10-28 13:22:45 -0700292 servo_constants.SERVO_SERIAL_ATTR,
Garry Wang11b5e872020-03-11 15:14:08 -0700293 servo_constants.SERVO_BOARD_ATTR,
294 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200295 servo_args = {key: args_dict[key]
296 for key in servo_attrs
297 if key in args_dict}
298 return (
299 None
Garry Wang11b5e872020-03-11 15:14:08 -0700300 if servo_constants.SERVO_HOST_ATTR in servo_args
301 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200302 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700303
J. Richard Barnette964fba02012-10-24 17:34:29 -0700304
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800305 def _initialize(self,
306 hostname,
307 chameleon_args=None,
308 servo_args=None,
309 pdtester_args=None,
310 try_lab_servo=False,
311 try_servo_repair=False,
312 ssh_verbosity_flag='',
313 ssh_options='',
314 try_servo_recovery=False,
315 *args,
316 **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800317 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700318
Fang Denge545abb2014-12-30 18:43:47 -0800319 This method will attempt to create the test-assistant object
320 (chameleon/servo) when it is needed by the test. Check
321 the docstring of chameleon_host.create_chameleon_host and
322 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700323
Fang Denge545abb2014-12-30 18:43:47 -0800324 @param hostname: Hostname of the dut.
325 @param chameleon_args: A dictionary that contains args for creating
326 a ChameleonHost. See chameleon_host for details.
327 @param servo_args: A dictionary that contains args for creating
328 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700329 @param try_lab_servo: When true, indicates that an attempt should
330 be made to create a ServoHost for a DUT in
331 the test lab, even if not required by
332 `servo_args`. See servo_host for details.
333 @param try_servo_repair: If a servo host is created, check it
334 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800335 See servo_host for details.
336 @param ssh_verbosity_flag: String, to pass to the ssh command to control
337 verbosity.
338 @param ssh_options: String, other ssh options to pass to the ssh
339 command.
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800340 @param try_servo_recovery: When True, start servod in recovery mode.
341 See servo_host for details.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700342 """
Andrew Luo4be621d2020-03-21 07:01:13 -0700343 super(CrosHost, self)._initialize(hostname=hostname, *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800344 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700345 # hold special dut_state for repair process
346 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700347 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700348 # self.env is a dictionary of environment variable settings
349 # to be exported for commands run on the host.
350 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
351 # errors that might happen.
352 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700353 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700354 self._ssh_options = ssh_options
Garry Wang1a493d82020-08-31 21:01:19 -0700355 self.health_profile = None
Garry Wang5e5538a2019-04-08 15:36:18 -0700356 self._default_power_method = None
Otabek Kasimov39637412020-11-23 19:09:27 -0800357 dut_health_profile = device_health_profile.DeviceHealthProfile(
358 hostname=self.hostname,
359 host_info=self.host_info_store.get(),
360 result_dir=self.get_result_dir())
Otabek Kasimovb61729d2020-11-24 00:42:43 -0800361
362 # TODO(otabek@): remove when b/171414073 closed
Andrew Luo4be621d2020-03-21 07:01:13 -0700363 if self.use_icmp:
Derek Beckettc7677812021-02-12 14:41:11 -0800364 pingable_before_servo = self.is_up_fast(count=1)
Andrew Luo4be621d2020-03-21 07:01:13 -0700365 if pingable_before_servo:
366 logging.info('DUT is pingable before init Servo.')
367 else:
368 logging.info('Skipping ping to DUT before init Servo.')
Otabek Kasimov39637412020-11-23 19:09:27 -0800369 _servo_host, servo_state = servo_host.create_servo_host(
370 dut=self,
371 servo_args=servo_args,
372 try_lab_servo=try_lab_servo,
373 try_servo_repair=try_servo_repair,
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800374 try_servo_recovery=try_servo_recovery,
Otabek Kasimov39637412020-11-23 19:09:27 -0800375 dut_host_info=self.host_info_store.get(),
376 dut_health_profile=dut_health_profile)
377 if dut_health_profile.is_loaded():
378 logging.info('Device health profile loaded.')
379 # The device profile is located in the servo_host which make it
380 # dependency. If profile is not loaded yet then we do not have it
381 # TODO(otabek@) persist device provide out of servo-host.
382 self.health_profile = dut_health_profile
383 self.set_servo_host(_servo_host, servo_state)
Richard Barnettee519dcd2016-08-15 17:37:17 -0700384
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800385 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800386 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700387 dut=self.hostname,
388 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800389 if self._chameleon_host:
390 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800391 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800392 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700393
Shijin Abraham78ce4402020-09-08 22:04:27 -0700394 # Bluetooth peers will be populated by the test if needed
395 self._btpeer_host_list = []
396 self.btpeer_list = []
397 self.btpeer = None
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800398
howardchung83e55272019-08-08 14:08:05 +0800399 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700400 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700401 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800402
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700403 if self._pdtester_host:
404 self.pdtester_servo = self._pdtester_host.get_servo()
405 logging.info('pdtester_servo: %r', self.pdtester_servo)
406 # Create the pdtester object used to access the ec uart
407 self.pdtester = pdtester.PDTester(self.pdtester_servo,
408 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800409 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700410 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800411
Fang Deng5d518f42013-08-02 14:04:32 -0700412
Shijin Abraham78ce4402020-09-08 22:04:27 -0700413 def initialize_btpeer(self, btpeer_args=[]):
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800414 """ Initialize the Bluetooth peers
415
416 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
417 is chameleon host on Raspberry Pi.
418 @param btpeer_args: A dictionary that contains args for creating
419 a ChameleonHost. See chameleon_host for details.
420
421 """
Shijin Abraham78ce4402020-09-08 22:04:27 -0700422 logging.debug('Attempting to initialize bluetooth peers if available')
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700423 try:
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700424 if type(btpeer_args) is list:
425 btpeer_args_list = btpeer_args
426 else:
427 btpeer_args_list = [btpeer_args]
428
429 self._btpeer_host_list = chameleon_host.create_btpeer_host(
430 dut=self.hostname, btpeer_args_list=btpeer_args_list)
431 logging.debug('Bluetooth peer hosts are %s',
432 self._btpeer_host_list)
433 self.btpeer_list = [_host.create_chameleon_board() for _host in
434 self._btpeer_host_list if _host is not None]
435
436 if len(self.btpeer_list) > 0:
437 self.btpeer = self.btpeer_list[0]
438
439 logging.debug('After initialize_btpeer btpeer_list %s '
440 'btpeer_host_list is %s and btpeer is %s',
441 self.btpeer_list, self._btpeer_host_list,
442 self.btpeer)
443 except Exception as e:
444 logging.error('Exception %s in initialize_btpeer', str(e))
445
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800446
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700447 def host_version_prefix(self, image):
448 """Return version label prefix.
449
450 In case the CrOS provisioning version is something other than the
451 standard CrOS version e.g. CrOS TH version, this function will
452 find the prefix from provision.py.
453
454 @param image: The image name to find its version prefix.
455 @returns: A prefix string for the image type.
456 """
457 return provision.get_version_label_prefix(image)
458
Andrew Luo3332ab22020-04-28 16:42:03 -0700459 def stage_build_to_usb(self, build):
460 """Stage the current ChromeOS image on the USB stick connected to the
461 servo.
462
463 @param build: The build to download and send to USB.
464 """
465 if not self.servo:
466 raise error.TestError('Host %s does not have servo.' %
467 self.hostname)
468
469 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700470
471 try:
472 self.servo.image_to_servo_usb(update_url)
473 finally:
474 # servo.image_to_servo_usb turned the DUT off, so turn it back on
475 logging.debug('Turn DUT power back on.')
476 self.servo.get_power_state_controller().power_on()
477
Andrew Luo3332ab22020-04-28 16:42:03 -0700478 logging.debug('ChromeOS image %s is staged on the USB stick.',
479 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700480
beepsdae65fd2013-07-26 16:24:41 -0700481 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700482 """
483 Make sure job_repo_url of this host is valid.
484
joychen03eaad92013-06-26 09:55:21 -0700485 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700486 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
487 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
488 download and extract it. If the devserver embedded in the url is
489 unresponsive, update the job_repo_url of the host after staging it on
490 another devserver.
491
492 @param job_repo_url: A url pointing to the devserver where the autotest
493 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700494 @param tag: The tag from the server job, in the format
495 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700496
497 @raises DevServerException: If we could not resolve a devserver.
498 @raises AutoservError: If we're unable to save the new job_repo_url as
499 a result of choosing a new devserver because the old one failed to
500 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700501 @raises urllib2.URLError: If the devserver embedded in job_repo_url
502 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700503 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800504 info = self.host_info_store.get()
505 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700506 if not job_repo_url:
507 logging.warning('No job repo url set on host %s', self.hostname)
508 return
509
510 logging.info('Verifying job repo url %s', job_repo_url)
511 devserver_url, image_name = tools.get_devserver_build_from_package_url(
512 job_repo_url)
513
beeps0c865032013-07-30 11:37:06 -0700514 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700515
516 logging.info('Staging autotest artifacts for %s on devserver %s',
517 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700518
519 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700520 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700521 stage_time = time.time() - start_time
522
523 # Record how much of the verification time comes from a devserver
524 # restage. If we're doing things right we should not see multiple
525 # devservers for a given board/build/branch path.
526 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800527 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700528 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800529 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700530 pass
531 else:
beeps0c865032013-07-30 11:37:06 -0700532 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700533 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700534 stats_key = {
535 'board': board,
536 'build_type': build_type,
537 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700538 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700539 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800540
541 monarch_fields = {
542 'board': board,
543 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800544 'branch': branch,
545 'dev_server': devserver,
546 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800547
Dan Shicf4d2032015-03-12 15:04:21 -0700548 def stage_server_side_package(self, image=None):
549 """Stage autotest server-side package on devserver.
550
551 @param image: Full path of an OS image to install or a build name.
552
553 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700554
555 @raise: error.AutoservError if fail to locate the build to test with, or
556 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700557 """
Dan Shid37736b2016-07-06 15:10:29 -0700558 # If enable_drone_in_restricted_subnet is False, do not set hostname
559 # in devserver.resolve call, so a devserver in non-restricted subnet
560 # is picked to stage autotest server package for drone to download.
561 hostname = self.hostname
562 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
563 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700564 if image:
565 image_name = tools.get_build_from_image(image)
566 if not image_name:
567 raise error.AutoservError(
568 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700569 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700570 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700571 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800572 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700573 if job_repo_url:
574 devserver_url, image_name = (
575 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700576 # If enable_drone_in_restricted_subnet is True, use the
577 # existing devserver. Otherwise, resolve a new one in
578 # non-restricted subnet.
579 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
580 ds = dev_server.ImageServer(devserver_url)
581 else:
582 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800583 elif info.build is not None:
584 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700585 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700586 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800587 raise error.AutoservError(
588 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700589 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700590
591 # Get the OS version of the build, for any build older than
592 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
593 match = re.match('.*/R\d+-(\d+)\.', image_name)
594 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700595 raise error.AutoservError(
596 'Build %s is older than %s. Server side packaging is '
597 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700598
Dan Shicf4d2032015-03-12 15:04:21 -0700599 ds.stage_artifacts(image_name, ['autotest_server_package'])
600 return '%s/static/%s/%s' % (ds.url(), image_name,
601 'autotest_server_package.tar.bz2')
602
603
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700604 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700605 """Stage a build on a devserver and return the update_url.
606
607 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700608 @param artifact: a string like 'test_image'. Requests
609 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700610 @returns a tuple of (image_name, URL) like
611 (lumpy-release/R27-3837.0.0,
612 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700613 """
614 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000615 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700616 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800617 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700618 devserver.stage_artifacts(image_name, [artifact])
619 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700620 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700621 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700622 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700623 else:
624 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700625
626
beepse539be02013-07-31 21:57:39 -0700627 def stage_factory_image_for_servo(self, image_name):
628 """Stage a build on a devserver and return the update_url.
629
630 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700631
beepse539be02013-07-31 21:57:39 -0700632 @return: An update URL, eg:
633 http://<devserver>/static/canary-channel/\
634 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700635
636 @raises: ValueError if the factory artifact name is missing from
637 the config.
638
beepse539be02013-07-31 21:57:39 -0700639 """
640 if not image_name:
641 logging.error('Need an image_name to stage a factory image.')
642 return
643
Dan Shib8540a52015-07-16 14:18:23 -0700644 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700645 'CROS', 'factory_artifact', type=str, default='')
646 if not factory_artifact:
647 raise ValueError('Cannot retrieve the factory artifact name from '
648 'autotest config, and hence cannot stage factory '
649 'artifacts.')
650
beepse539be02013-07-31 21:57:39 -0700651 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800652 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700653 devserver.stage_artifacts(
654 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700655 [factory_artifact],
656 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700657
658 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
659
660
Laurence Goodby778c9a42017-05-24 19:24:07 -0700661 def prepare_for_update(self):
662 """Prepares the DUT for an update.
663
664 Subclasses may override this to perform any special actions
665 required before updating.
666 """
Laurence Goodby468de252017-06-08 17:22:53 -0700667 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700668
669
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800670 def _clear_fw_version_labels(self, rw_only):
671 """Clear firmware version labels from the machine.
672
673 @param rw_only: True to only clear fwrw_version; otherewise, clear
674 both fwro_version and fwrw_version.
675 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700676 info = self.host_info_store.get()
677 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800678 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700679 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
680 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700681
682
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800683 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700684 """Add firmware version label to the machine.
685
686 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800687 @param rw_only: True to only add fwrw_version; otherwise, add both
688 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700689
690 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700691 info = self.host_info_store.get()
692 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800693 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700694 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
695 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700696
697
Namyoon Woo33f38852020-04-13 17:26:58 -0700698 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800699 """Search for the latest package release version from the image archive,
700 and return it.
701
Namyoon Woo33f38852020-04-13 17:26:58 -0700702 @param platform: platform name, a.k.a. board or model
703 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800704
Namyoon Woo33f38852020-04-13 17:26:58 -0700705 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
706 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800707 or None if LATEST release file does not exist.
708 """
709
Namyoon Woo33f38852020-04-13 17:26:58 -0700710 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800711
Namyoon Woo33f38852020-04-13 17:26:58 -0700712 # Search the image path in reference board archive as well.
713 # For example, bob has its binary image under its reference board (gru)
714 # image archive.
715 if ref_board:
716 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800717
Namyoon Woo33f38852020-04-13 17:26:58 -0700718 for board in platforms:
719 # Read 'LATEST-1.0.0' file
720 branch_dir = provision.FW_BRANCH_GLOB % board
721 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
722 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800723
Namyoon Woo33f38852020-04-13 17:26:58 -0700724 try:
725 # The result could be one or more.
726 result = utils.system_output('gsutil ls -d ' + latest_file)
727
728 candidates = re.findall('gs://.*', result)
729
730 # Found the directory candidates. No need to check the other
731 # board name cadidates. Let's break the loop.
732 break
733 except error.CmdError:
734 # It doesn't exist. Let's move on to the next item.
735 pass
736 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800737 logging.error('No LATEST release info is available.')
738 return None
739
Namyoon Woo406c7d42020-01-24 15:57:11 -0800740 for cand_dir in candidates:
741 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800742
Namyoon Woo406c7d42020-01-24 15:57:11 -0800743 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700744 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800745 try:
746 # Check if release_path does exist.
747 release = utils.system_output('gsutil ls -d ' + release_path)
748 # Now 'release' has a full directory path: e.g.
749 # gs://chromeos-image-archive/firmware-octopus-11297.B-
750 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
751
752 # Remove "gs://chromeos-image-archive".
753 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
754
755 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
756 return release.strip('/')
757 except error.CmdError:
758 # The directory might not exist. Let's try next candidate.
759 pass
760 else:
761 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800762
Brent Peterson1cb623a2020-01-09 13:14:28 -0800763 @staticmethod
764 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800765 """Get version string from binary image using regular expression.
766
767 @param image: Binary image to search
768 @param version_regex: Regular expression to search for
769
770 @return Version string
771
772 @raises TestFail if no version string is found in image
773 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800774 with open(image, 'rb') as f:
775 image_data = f.read()
Derek Beckett98345552020-08-31 16:07:22 -0700776 match = re.findall(version_regex,
777 image_data.decode('ISO-8859-1', errors='ignore'))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800778 if match:
779 return match[0]
780 else:
781 raise error.TestFail('Failed to read version from %s.' % image)
782
783
Garry Wangad2a1712020-03-26 15:06:43 -0700784 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800785 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700786 try_scp=False, install_ec=True, install_bios=True,
787 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700788 """Install firmware to the DUT.
789
790 Use stateful update if the DUT is already running the same build.
791 Stateful update does not update kernel and tends to run much faster
792 than a full reimage. If the DUT is running a different build, or it
793 failed to do a stateful update, full update, including kernel update,
794 will be applied to the DUT.
795
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800796 Once a host enters firmware_install its fw[ro|rw]_version label will
797 be removed. After the firmware is updated successfully, a new
798 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700799
800 @param build: The build version to which we want to provision the
801 firmware of the machine,
802 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800803 @param rw_only: True to only install firmware to its RW portions. Keep
804 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700805 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800806 @param local_tarball: Path to local firmware image for installing
807 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800808 @param verify_version: True to verify EC and BIOS versions after
809 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800810 @param try_scp: False to always program using servo, true to try copying
811 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700812 @param install_ec: True to install EC FW, and False to skip it.
813 @param install_bios: True to install BIOS, and False to skip it.
814 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700815
816 TODO(dshi): After bug 381718 is fixed, update here with corresponding
817 exceptions that could be raised.
818
819 """
820 if not self.servo:
821 raise error.TestError('Host %s does not have servo.' %
822 self.hostname)
823
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700824 info = self.host_info_store.get()
825 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700826 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800827
828 if board is None or board == '':
829 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700830
Namyoon Woo382e5892020-05-20 16:48:40 -0700831 # if board_as argument is passed, then use it instead of the original
832 # board name.
833 if board_as:
834 board = board_as
835
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700836 if model is None or model == '':
Namyoon Woofb16eae2020-08-14 10:02:39 -0700837 try:
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700838 model = self.get_platform()
Namyoon Woofb16eae2020-08-14 10:02:39 -0700839 except Exception as e:
840 logging.warn('Dut is unresponsive: %s', str(e))
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700841
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800842 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700843 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800844 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700845 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800846
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700847 try:
848 ds = dev_server.ImageServer.resolve(build, self.hostname)
849 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800850
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700851 if not dest:
852 tmpd = autotemp.tempdir(unique_id='fwimage')
853 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800854
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700855 # Download firmware image
856 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
857 local_tarball = os.path.join(dest, os.path.basename(fwurl))
858 ds.download_file(fwurl, local_tarball)
859 except Exception as e:
860 raise error.TestError('Failed to download firmware package: %s'
861 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700862
Namyoon Woo382e5892020-05-20 16:48:40 -0700863 ec_image = None
864 if install_ec:
865 # Extract EC image from tarball
866 logging.info('Extracting EC image.')
867 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700868 logging.info('Extracted: %s', ec_image)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800869
Namyoon Woo382e5892020-05-20 16:48:40 -0700870 bios_image = None
871 if install_bios:
872 # Extract BIOS image from tarball
873 logging.info('Extracting BIOS image.')
874 bios_image = self.servo.extract_bios_image(board, model,
875 local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700876 logging.info('Extracted: %s', bios_image)
Namyoon Woo382e5892020-05-20 16:48:40 -0700877
878 if not bios_image and not ec_image:
879 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800880
Brent Petersonc70a1832020-01-24 15:54:35 -0800881 # Clear firmware version labels
882 self._clear_fw_version_labels(rw_only)
883
884 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800885 try:
Garry Wang50e4a492020-08-05 12:29:57 -0700886 # Check if copying to DUT is enabled and DUT is available
887 if try_scp and self.is_up():
Brent Petersonc70a1832020-01-24 15:54:35 -0800888 # DUT is available, make temp firmware directory to store images
889 logging.info('Making temp folder.')
890 dest_folder = '/tmp/firmware'
891 self.run('mkdir -p ' + dest_folder)
892
Namyoon Woo68b68082020-06-02 13:13:14 -0700893 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800894
Namyoon Woo382e5892020-05-20 16:48:40 -0700895 if bios_image:
896 # Send BIOS firmware image to DUT
897 logging.info('Sending BIOS firmware.')
898 dest_bios_path = os.path.join(dest_folder,
899 os.path.basename(bios_image))
900 self.send_file(bios_image, dest_bios_path)
901
902 # Initialize firmware update command for BIOS image
903 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800904
905 # Send EC firmware image to DUT when EC image was found
906 if ec_image:
907 logging.info('Sending EC firmware.')
908 dest_ec_path = os.path.join(dest_folder,
909 os.path.basename(ec_image))
910 self.send_file(ec_image, dest_ec_path)
911
912 # Add EC image to firmware update command
913 fw_cmd += ' -e %s' % dest_ec_path
914
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700915 # Make sure command is allowed to finish even if ssh fails.
916 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
917
Brent Peterson669edf42020-02-07 15:07:54 -0800918 # Update firmware on DUT
919 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700920 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700921 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700922 except error.AutoservRunError as e:
923 if e.result_obj.exit_status != 255:
924 raise
925 elif ec_image:
926 logging.warn("DUT network dropped during update"
927 " (often caused by EC resetting USB)")
928 else:
929 logging.error("DUT network dropped during update"
930 " (unexpected, since no EC image)")
931 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800932 else:
933 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800934 if ec_image:
935 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700936 if bios_image:
937 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800938 if utils.host_is_in_lab_zone(self.hostname):
939 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800940
941 # Reboot and wait for DUT after installing firmware
942 logging.info('Rebooting DUT.')
943 self.servo.get_power_state_controller().reset()
944 time.sleep(self.servo.BOOT_DELAY)
945 self.test_wait_for_boot()
946
947 # When enabled verify EC and BIOS firmware version after programming
948 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800949 # Check programmed EC firmware when EC image was found
950 if ec_image:
951 logging.info('Checking EC firmware version.')
952 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800953 ec_version_prefix = dest_ec_version.split('_', 1)[0]
954 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800955 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800956 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800957 if dest_ec_version != image_ec_version:
958 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700959 'Failed to update EC firmware, version %s '
960 '(expected %s)' % (dest_ec_version,
961 image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800962
Namyoon Woo382e5892020-05-20 16:48:40 -0700963 if bios_image:
964 # Check programmed BIOS firmware against expected version
965 logging.info('Checking BIOS firmware version.')
966 dest_bios_version = self.get_firmware_version()
967 bios_version_prefix = dest_bios_version.split('.', 1)[0]
968 bios_regex = self._BIOS_REGEX % bios_version_prefix
969 image_bios_version = self.get_version_from_image(bios_image,
970 bios_regex)
971 if dest_bios_version != image_bios_version:
972 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -0700973 'Failed to update BIOS, version %s '
Namyoon Woo382e5892020-05-20 16:48:40 -0700974 '(expected %s)' % (dest_bios_version,
975 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700976 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700977 if tmpd:
978 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700979
980
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -0800981 def install_image_to_servo_usb(self, image_url=None):
982 """Installing a test image on a USB storage device.
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700983
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -0800984 Download image to USB-storage attached to the Servo board.
Richard Barnette03a0c132012-11-05 12:40:35 -0800985
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -0800986 @param image_url: If specified use as the url to download to
987 USB-storage.
988
989 @raises AutoservError if the image fails to download.
beepsf079cfb2013-09-18 17:49:51 -0700990
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800991 """
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -0800992 if not image_url:
993 logging.debug('Skip download as image_url not provided!')
994 return
Garry Wang7b0e1b72020-03-25 19:08:59 -0700995
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -0800996 logging.info('Downloading image to USB')
Derek Beckett5f4edf02021-07-27 14:56:44 -0700997 try:
998 self.servo.image_to_servo_usb(image_path=image_url,
999 power_off_dut=False)
1000 except error.AutotestError as e:
1001 six.reraise(error.AutotestError, str(e), sys.exc_info()[2])
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001002
1003 def boot_in_recovery_mode(self,
1004 usb_boot_timeout=USB_BOOT_TIMEOUT,
1005 need_snk=False):
1006 """Booting host in recovery mode.
1007
1008 Boot device in recovery mode and verify that device booted from
1009 external storage as expected.
1010
1011 @param usb_boot_timeout: The usb_boot_timeout to use wait the host
1012 to boot. Factory images need a longer
1013 usb_boot_timeout than regular cros images.
1014 @param snk_mode: If True, switch servo_v4 role to 'snk'
1015 mode before boot DUT into recovery mode.
1016
1017 @raises AutoservError if the image fails to boot.
1018
1019 """
1020 logging.info('Booting from USB directly. Usb boot timeout: %s',
1021 usb_boot_timeout)
Derek Beckett5f4edf02021-07-27 14:56:44 -07001022 self.servo.boot_in_recovery_mode(snk_mode=need_snk)
1023 if not self.wait_up(timeout=usb_boot_timeout):
1024 if need_snk:
1025 # Attempt to restore servo_v4 role to 'src' mode.
1026 self.servo.set_servo_v4_role('src')
1027 raise hosts.AutoservRepairError(
1028 'DUT failed to boot from USB after %d seconds' %
1029 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001030
Garry Wang2e347df2020-10-30 14:04:26 -07001031 # Make sure the DUT is boot from an external device.
1032 if not self.is_boot_from_external_device():
1033 raise hosts.AutoservRepairError(
1034 'DUT is expected to boot from an external device(e.g. '
1035 'a usb stick), however it seems still boot from an'
1036 ' internal storage.', 'boot_from_internal_storage')
1037
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001038 def run_install_image(self,
1039 install_timeout=INSTALL_TIMEOUT,
1040 need_snk=False,
1041 is_repair=False):
1042 """Installing the image with chromeos-install.
1043
1044 Steps included:
1045 1) Recover TPM on the device
1046 2) Run chromeos-install
1047 2.a) if success: power off/on the device
1048 2.b) if fail:
1049 2.b.1) Mark for replacement if fail with hardware issue
1050 2.b.2) Run internal storage check. (Only if is_repair=True)
1051 3) Wait the device to boot as verifier of success install
1052
1053 Device has to booted from external storage.
1054
1055 @param install_timeout: The timeout to use when installing the
1056 chromeos image. Factory images need a
1057 longer install_timeout.
1058 @param snk_mode: If True, switch servo_v4 role to 'snk'
1059 mode before boot DUT into recovery mode.
1060 @param is_repair: Indicates if the method is called from a
1061 repair task.
1062
1063 @raises AutoservError if the fail in process of install image.
1064 @raises AutoservRepairError if fail to boot after install image.
1065
1066 """
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001067 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1068 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001069 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001070 try:
1071 self.run('chromeos-tpm-recovery')
1072 except error.AutoservRunError:
1073 logging.warn('chromeos-tpm-recovery is too old.')
1074
Derek Beckett5f4edf02021-07-27 14:56:44 -07001075 logging.info('Installing image through chromeos-install.')
1076 try:
1077 self.run('chromeos-install --yes', timeout=install_timeout)
1078 self.halt()
1079 except Exception as e:
1080 storage_errors = [
1081 'No space left on device',
1082 'I/O error when trying to write primary GPT',
1083 'Input/output error while writing out',
1084 'cannot read GPT header',
1085 'can not determine destination device',
1086 'wrong fs type',
1087 'bad superblock on',
1088 ]
1089 has_error = [msg for msg in storage_errors if (msg in str(e))]
1090 if has_error:
1091 info = self.host_info_store.get()
1092 info.set_version_label(audit_const.DUT_STORAGE_STATE_PREFIX,
1093 audit_const.HW_STATE_NEED_REPLACEMENT)
1094 self.host_info_store.commit(info)
1095 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001096 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Derek Beckett5f4edf02021-07-27 14:56:44 -07001097 logging.debug('Fail install image from USB; Storage error; %s',
1098 e)
1099 raise error.AutoservError(
Otabek Kasimov808cd832020-05-28 18:27:46 -07001100 'Failed to install image from USB due to a suspect '
1101 'disk failure, DUT storage state changed to '
1102 'need_replacement, please check debug log '
1103 'for details.')
Derek Beckett5f4edf02021-07-27 14:56:44 -07001104 else:
1105 if is_repair:
1106 # DUT will be marked for replacement if storage is bad.
1107 audit_verify.VerifyDutStorage(self).verify()
Otabek Kasimov27bb2862020-08-10 14:40:45 -07001108
Derek Beckett5f4edf02021-07-27 14:56:44 -07001109 logging.debug('Fail install image from USB; %s', e)
1110 raise error.AutoservError(
Otabek Kasimov808cd832020-05-28 18:27:46 -07001111 'Failed to install image from USB due to unexpected '
1112 'error, please check debug log for details.')
Derek Beckett5f4edf02021-07-27 14:56:44 -07001113 finally:
1114 # We need reset the DUT no matter re-install success or not,
1115 # as we don't want leave the DUT in boot from usb state.
1116 logging.info('Power cycling DUT through servo.')
1117 self.servo.get_power_state_controller().power_off()
1118 self.servo.switch_usbkey('off')
1119 if need_snk:
1120 # Attempt to restore servo_v4 role to 'src' mode.
1121 self.servo.set_servo_v4_role('src')
1122 # N.B. The Servo API requires that we use power_on() here
1123 # for two reasons:
1124 # 1) After turning on a DUT in recovery mode, you must turn
1125 # it off and then on with power_on() once more to
1126 # disable recovery mode (this is a Parrot specific
1127 # requirement).
1128 # 2) After power_off(), the only way to turn on is with
1129 # power_on() (this is a Storm specific requirement).
1130 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001131
1132 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001133 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001134 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1135 'test image after %d seconds' %
1136 self.BOOT_TIMEOUT,
1137 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001138
Otabek Kasimov2cb72cf2021-02-25 10:59:22 -08001139 def servo_install(self,
1140 image_url=None,
1141 usb_boot_timeout=USB_BOOT_TIMEOUT,
1142 install_timeout=INSTALL_TIMEOUT,
1143 is_repair=False):
1144 """Re-install the OS on the DUT by:
1145
1146 Steps:
1147 1) Power off the host
1148 2) Installing an image on a USB-storage attached to the Servo board
1149 3) Booting that image in recovery mode
1150 4) Installing the image with chromeos-install.
1151
1152 @param image_url: If specified use as the url to install on
1153 the DUT otherwise boot the currently
1154 staged image on the USB stick.
1155 @param usb_boot_timeout: The usb_boot_timeout to use during
1156 re-image. Factory images need a longer
1157 usb_boot_timeout than regular cros images.
1158 @param install_timeout: The timeout to use when installing the
1159 chromeos image. Factory images need a
1160 longer install_timeout.
1161 @param is_repair: Indicates if the method is called from a
1162 repair task.
1163
1164 @raises AutoservError if the image fails to boot.
1165
1166 """
1167 self.servo.get_power_state_controller().power_off()
1168 if image_url:
1169 self.install_image_to_servo_usb(image_url=image_url)
1170 else:
1171 # Give the DUT some time to power_off if we skip
1172 # download image to usb. (crbug.com/982993)
1173 time.sleep(10)
1174
1175 need_snk = self.require_snk_mode_in_recovery()
1176
1177 self.boot_in_recovery_mode(usb_boot_timeout=usb_boot_timeout,
1178 need_snk=need_snk)
1179
1180 self.run_install_image(install_timeout=install_timeout,
1181 need_snk=need_snk,
1182 is_repair=is_repair)
Scott Zawalski62bacae2013-03-05 10:40:32 -05001183
Garry Wanga2e78172020-09-09 23:49:07 -07001184 def set_servo_host(self, host, servo_state=None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001185 """Set our servo host member, and associated servo.
1186
1187 @param host Our new `ServoHost`.
1188 """
1189 self._servo_host = host
Derek Beckettb66e5c82020-08-12 15:31:02 -07001190 self.servo_pwr_supported = None
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001191 if self._servo_host is not None:
1192 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001193 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001194 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Derek Beckettb66e5c82020-08-12 15:31:02 -07001195 try:
1196 self.servo_pwr_supported = self.servo.has_control('power_state')
1197 except Exception as e:
1198 logging.debug(
1199 "Could not get servo power state due to {}".format(e))
Gregory Nisbet93b23e22020-10-02 20:42:16 +00001200 else:
1201 self.servo = None
Derek Beckettb66e5c82020-08-12 15:31:02 -07001202 self.servo_pwr_supported = False
Otabek Kasimov41301a22020-05-10 15:28:21 -07001203 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001204 self.set_servo_state(servo_state)
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07001205 self._set_servo_topology()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001206
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001207
Richard Barnette9a26ad62016-06-10 12:03:08 -07001208 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001209 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001210 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001211
Richard Barnette9a26ad62016-06-10 12:03:08 -07001212 If the servo object is missing, attempt to repair the servo
1213 host. Repair failures are passed back to the caller.
1214
1215 @raise AutoservError: If there is no servo host for this CrOS
1216 host.
1217 """
1218 if self.servo:
1219 return
1220 if not self._servo_host:
1221 raise error.AutoservError('No servo host for %s.' %
1222 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001223 try:
1224 self._servo_host.repair()
1225 except:
1226 raise
1227 finally:
1228 self.set_servo_host(self._servo_host)
1229
1230
Otabek Kasimov41301a22020-05-10 15:28:21 -07001231 def set_servo_type(self):
1232 """Set servo info labels to dut host_info"""
1233 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001234 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001235 return
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -08001236 if not self.is_servo_in_working_state():
1237 logging.debug('Servo is not good, skip update servo_type.')
1238 return
Otabek Kasimov41301a22020-05-10 15:28:21 -07001239 servo_type = self.servo.get_servo_type()
1240 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001241 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001242 ' by `dut-control servo_type`! Please file a bug'
1243 ' and inform infra team as we are not expected '
1244 ' to reach this point.')
1245 return
1246 host_info = self.host_info_store.get()
1247 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1248 old_type = host_info.get_label_value(prefix)
1249 if old_type == servo_type:
1250 # do not need update
1251 return
1252 host_info.set_version_label(prefix, servo_type)
1253 self.host_info_store.commit(host_info)
1254 logging.info('ServoHost: servo_type updated to %s '
1255 '(previous: %s)', servo_type, old_type)
1256
1257
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001258 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001259 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001260 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001261 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001262 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001263 old_state = host_info.get_label_value(servo_state_prefix)
1264 if old_state == servo_state:
1265 # do not need update
1266 return
1267 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001268 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001269 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1270 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001271
1272
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001273 def get_servo_state(self):
1274 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001275 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001276 return host_info.get_label_value(servo_state_prefix)
1277
Otabek Kasimov5f039202020-10-28 15:45:29 -07001278 def is_servo_in_working_state(self):
1279 """Validate servo is in WORKING state."""
1280 servo_state = self.get_servo_state()
1281 return servo_state == servo_constants.SERVO_STATE_WORKING
1282
Dana Goyette655af512020-09-03 10:48:23 -07001283 def get_servo_usb_state(self):
1284 """Get the label value indicating the health of the USB drive.
1285
1286 @return: The label value if defined, otherwise '' (empty string).
1287 @rtype: str
1288 """
1289 host_info = self.host_info_store.get()
1290 servo_usb_state_prefix = audit_const.SERVO_USB_STATE_PREFIX
1291 return host_info.get_label_value(servo_usb_state_prefix)
1292
1293 def is_servo_usb_usable(self):
1294 """Check if the servo USB storage device is usable for FAFT.
1295
1296 @return: False if the label indicates a state that will break FAFT.
1297 True if state is okay, or if state is not defined.
1298 @rtype: bool
1299 """
1300 usb_state = self.get_servo_usb_state()
1301 return usb_state in ('', audit_const.HW_STATE_ACCEPTABLE,
1302 audit_const.HW_STATE_NORMAL,
1303 audit_const.HW_STATE_UNKNOWN)
Otabek Kasimov41301a22020-05-10 15:28:21 -07001304
Garry Wang000c6c02020-05-11 21:27:23 -07001305 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1306 if smart_usbhub_detected is None:
1307 # skip the label update here as this indicate we wasn't able
1308 # to confirm usbhub type.
1309 return
1310 host_info = self.host_info_store.get()
1311 if (smart_usbhub_detected ==
1312 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1313 # skip label update if current label match the truth.
1314 return
1315 if smart_usbhub_detected:
1316 logging.info('Adding %s label to host %s',
1317 servo_constants.SMART_USBHUB_LABEL,
1318 self.hostname)
1319 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1320 else:
1321 logging.info('Removing %s label from host %s',
1322 servo_constants.SMART_USBHUB_LABEL,
1323 self.hostname)
1324 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1325 self.host_info_store.commit(host_info)
1326
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001327 def repair(self):
1328 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001329
1330 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001331 not call back to the parent class, but instead relies on
1332 `self._repair_strategy` to coordinate the verification and
1333 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001334 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001335 message = 'Beginning repair for host %s board %s model %s'
1336 info = self.host_info_store.get()
1337 message %= (self.hostname, info.board, info.model)
1338 self.record('INFO', None, None, message)
Garry Wanga2e78172020-09-09 23:49:07 -07001339 profile_state = profile_constants.DUT_STATE_READY
Shijin Abraham78ce4402020-09-08 22:04:27 -07001340 # Initialize bluetooth peers
1341 self.initialize_btpeer()
Garry Wang87af1d02020-05-26 17:55:54 -07001342 try:
1343 self._repair_strategy.repair(self)
1344 except hosts.AutoservVerifyDependencyError as e:
Otabek Kasimovd48389b2020-12-07 02:38:34 -08001345 # TODO(otabek): remove when finish b/174191325
1346 self._stat_if_pingable_but_not_sshable()
Garry Wang87af1d02020-05-26 17:55:54 -07001347 # We don't want flag a DUT as failed if only non-critical
1348 # verifier(s) failed during the repair.
1349 if e.is_critical():
Garry Wanga2e78172020-09-09 23:49:07 -07001350 profile_state = profile_constants.DUT_STATE_REPAIR_FAILED
Otabek Kasimov86062d02020-11-17 13:30:22 -08001351 self._reboot_labstation_if_needed()
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001352 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001353 raise
Garry Wanga2e78172020-09-09 23:49:07 -07001354 finally:
1355 self.set_health_profile_dut_state(profile_state)
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001356
Otabek Kasimov8bb09912020-10-01 14:44:57 -07001357 def get_verifier_state(self, tag):
Otabek Kasimov44273d22021-02-26 17:13:24 -08001358 """Return the state of host verifier by tag.
Otabek Kasimov8bb09912020-10-01 14:44:57 -07001359
1360 @returns: bool or None
1361 """
1362 return self._repair_strategy.verifier_is_good(tag)
Richard Barnette82c35912012-11-20 10:09:10 -08001363
Otabek Kasimov44273d22021-02-26 17:13:24 -08001364 def get_repair_strategy_node(self, tag):
1365 """Return the instance of verifier/repair node for host by tag.
1366
1367 @returns: _DependencyNode or None
1368 """
1369 return self._repair_strategy.node_by_tag(tag)
1370
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001371 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001372 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001373 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001374
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001375 if self._chameleon_host:
1376 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001377
Garry Wang1a493d82020-08-31 21:01:19 -07001378 if self.health_profile:
1379 try:
1380 self.health_profile.close()
1381 except Exception as e:
1382 logging.warning(
1383 'Failed to finalize device health profile; %s', e)
1384
xixuand6011f12016-12-08 15:01:58 -08001385 if self._servo_host:
1386 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001387
Dan Shi49ca0932014-11-14 11:22:27 -08001388 def get_power_supply_info(self):
1389 """Get the output of power_supply_info.
1390
1391 power_supply_info outputs the info of each power supply, e.g.,
1392 Device: Line Power
1393 online: no
1394 type: Mains
1395 voltage (V): 0
1396 current (A): 0
1397 Device: Battery
1398 state: Discharging
1399 percentage: 95.9276
1400 technology: Li-ion
1401
1402 Above output shows two devices, Line Power and Battery, with details of
1403 each device listed. This function parses the output into a dictionary,
1404 with key being the device name, and value being a dictionary of details
1405 of the device info.
1406
1407 @return: The dictionary of power_supply_info, e.g.,
1408 {'Line Power': {'online': 'yes', 'type': 'main'},
1409 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001410 @raise error.AutoservRunError if power_supply_info tool is not found in
1411 the DUT. Caller should handle this error to avoid false failure
1412 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001413 """
1414 result = self.run('power_supply_info').stdout.strip()
1415 info = {}
1416 device_name = None
1417 device_info = {}
1418 for line in result.split('\n'):
1419 pair = [v.strip() for v in line.split(':')]
1420 if len(pair) != 2:
1421 continue
1422 if pair[0] == 'Device':
1423 if device_name:
1424 info[device_name] = device_info
1425 device_name = pair[1]
1426 device_info = {}
1427 else:
1428 device_info[pair[0]] = pair[1]
1429 if device_name and not device_name in info:
1430 info[device_name] = device_info
1431 return info
1432
1433
1434 def get_battery_percentage(self):
1435 """Get the battery percentage.
1436
1437 @return: The percentage of battery level, value range from 0-100. Return
1438 None if the battery info cannot be retrieved.
1439 """
1440 try:
1441 info = self.get_power_supply_info()
1442 logging.info(info)
1443 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001444 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001445 return None
1446
1447
Philip Chenaf69ead2020-03-27 13:06:42 -07001448 def get_battery_state(self):
1449 """Get the battery charging state.
1450
1451 @return: A string representing the battery charging state. It can be
1452 'Charging', 'Fully charged', or 'Discharging'.
1453 """
1454 try:
1455 info = self.get_power_supply_info()
1456 logging.info(info)
1457 return info['Battery']['state']
1458 except (KeyError, ValueError, error.AutoservRunError):
1459 return None
1460
1461
Daniel Campello8ca25c22019-12-13 16:48:26 -07001462 def get_battery_display_percentage(self):
1463 """Get the battery display percentage.
1464
1465 @return: The display percentage of battery level, value range from
1466 0-100. Return None if the battery info cannot be retrieved.
1467 """
1468 try:
1469 info = self.get_power_supply_info()
1470 logging.info(info)
1471 return float(info['Battery']['display percentage'])
1472 except (KeyError, ValueError, error.AutoservRunError):
1473 return None
1474
1475
Dan Shi49ca0932014-11-14 11:22:27 -08001476 def is_ac_connected(self):
1477 """Check if the dut has power adapter connected and charging.
1478
1479 @return: True if power adapter is connected and charging.
1480 """
1481 try:
1482 info = self.get_power_supply_info()
1483 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001484 except (KeyError, error.AutoservRunError):
1485 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001486
1487
Simran Basi5e6339a2013-03-21 11:34:32 -07001488 def _cleanup_poweron(self):
1489 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001490 info = self.host_info_store.get()
1491 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001492 return
1493 logging.debug('This host has recently interacted with the RPM'
1494 ' Infrastructure. Ensuring power is on.')
1495 try:
1496 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001497 self._remove_rpm_changed_tag()
Derek Beckett5ea7bf32021-08-03 13:09:20 -07001498 except Exception:
1499 # TODO b/195443964: Re-wire as needed once TLW is available.
Simran Basi5e6339a2013-03-21 11:34:32 -07001500 logging.error('Failed to turn Power On for this host after '
1501 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001502
1503 battery_percentage = self.get_battery_percentage()
Otabek Kasimov58e22562020-11-03 17:17:41 -08001504 if (
1505 battery_percentage
1506 and battery_percentage < cros_constants.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001507 raise
1508 elif self.is_ac_connected():
1509 logging.info('The device has power adapter connected and '
1510 'charging. No need to try to turn RPM on '
1511 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001512 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001513 logging.info('Battery level is now at %s%%. The device may '
1514 'still have enough power to run test, so no '
1515 'exception will be raised.', battery_percentage)
1516
Simran Basi5e6339a2013-03-21 11:34:32 -07001517
Garry Wangad4d4fd2019-01-30 17:00:38 -08001518 def _remove_rpm_changed_tag(self):
Derek Beckett5ea7bf32021-08-03 13:09:20 -07001519 # TODO b/195443964: Re-wire as needed once TLW is available.
1520 pass
Garry Wangad4d4fd2019-01-30 17:00:38 -08001521
1522
1523 def _add_rpm_changed_tag(self):
Derek Beckett5ea7bf32021-08-03 13:09:20 -07001524 # TODO b/195443964: Re-wire as needed once TLW is available.
1525 pass
Garry Wangad4d4fd2019-01-30 17:00:38 -08001526
1527
beepsc87ff602013-07-31 21:53:00 -07001528 def _is_factory_image(self):
1529 """Checks if the image on the DUT is a factory image.
1530
1531 @return: True if the image on the DUT is a factory image.
1532 False otherwise.
1533 """
1534 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1535 return result.exit_status == 0
1536
1537
1538 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001539 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001540
1541 @raises: FactoryImageCheckerException for factory images, since
1542 we cannot attempt to restart ui on them.
1543 error.AutoservRunError for any other type of error that
1544 occurs while restarting ui.
1545 """
1546 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001547 raise FactoryImageCheckerException('Cannot restart ui on factory '
1548 'images')
beepsc87ff602013-07-31 21:53:00 -07001549
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001550 # TODO(jrbarnette): The command to stop/start the ui job
1551 # should live inside cros_ui, too. However that would seem
1552 # to imply interface changes to the existing start()/restart()
1553 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001554 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001555 self.run('stop ui; start ui')
1556 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001557
1558
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001559 def _start_powerd_if_needed(self):
1560 """Start powerd if it isn't already running."""
1561 self.run('start powerd', ignore_status=True)
1562
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001563 def _read_arc_prop_file(self, filename):
1564 for path in [
1565 '/usr/share/arcvm/properties/', '/usr/share/arc/properties/'
1566 ]:
1567 if self.path_exists(path + filename):
1568 return utils.parse_cmd_output('cat ' + path + filename,
1569 run_method=self.run)
1570 return None
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001571
Jiyoun Hac172ee72020-12-15 08:57:29 +09001572 def _get_arc_build_info(self):
1573 """Returns a dictionary mapping build properties to their values."""
1574 build_info = None
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001575 for filename in ['build.prop', 'vendor_build.prop']:
1576 properties = self._read_arc_prop_file(filename)
1577 if properties:
1578 if build_info:
1579 build_info.update(properties)
1580 else:
1581 build_info = properties
1582 else:
1583 logging.error('Failed to find %s in device.', filename)
Jiyoun Hac172ee72020-12-15 08:57:29 +09001584 return build_info
1585
Jiyoun Haba37f312021-01-13 09:44:16 +09001586 def get_arc_primary_abi(self):
Jiyoun Hac172ee72020-12-15 08:57:29 +09001587 """Returns the primary abi of the host."""
1588 return self._get_arc_build_info().get('ro.product.cpu.abi')
1589
Jiyoun Haba37f312021-01-13 09:44:16 +09001590 def get_arc_security_patch(self):
Jiyoun Hac172ee72020-12-15 08:57:29 +09001591 """Returns the security patch of the host."""
1592 return self._get_arc_build_info().get('ro.build.version.security_patch')
1593
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001594 def get_arc_first_api_level(self):
1595 """Returns the security patch of the host."""
1596 return self._get_arc_build_info().get('ro.product.first_api_level')
1597
xixuana3bbc422017-05-04 15:57:21 -07001598 def _get_lsb_release_content(self):
1599 """Return the content of lsb-release file of host."""
1600 return self.run(
1601 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1602
1603
Dan Shi549fb822015-03-24 18:01:11 -07001604 def get_release_version(self):
1605 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1606
1607 @returns The version string in lsb-release, under attribute
1608 CHROMEOS_RELEASE_VERSION.
1609 """
Dan Shi549fb822015-03-24 18:01:11 -07001610 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001611 lsb_release_content=self._get_lsb_release_content())
1612
1613
Don Garrettb9f35802018-01-22 18:25:40 -08001614 def get_release_builder_path(self):
1615 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1616
1617 @returns The version string in lsb-release, under attribute
1618 CHROMEOS_RELEASE_BUILDER_PATH.
1619 """
1620 return lsbrelease_utils.get_chromeos_release_builder_path(
1621 lsb_release_content=self._get_lsb_release_content())
1622
1623
xixuana3bbc422017-05-04 15:57:21 -07001624 def get_chromeos_release_milestone(self):
1625 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1626 from lsb-release.
1627
1628 @returns The version string in lsb-release, under attribute
1629 CHROMEOS_RELEASE_BUILD_TYPE.
1630 """
1631 return lsbrelease_utils.get_chromeos_release_milestone(
1632 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001633
1634
1635 def verify_cros_version_label(self):
Garry Wangd18e7b32020-08-07 18:31:44 -07001636 """Verify if host's cros-version label match the actual image in dut.
Dan Shi549fb822015-03-24 18:01:11 -07001637
Garry Wangd18e7b32020-08-07 18:31:44 -07001638 @returns True if the label match with image in dut, otherwise False
Dan Shi549fb822015-03-24 18:01:11 -07001639 """
Garry Wangd18e7b32020-08-07 18:31:44 -07001640 os_from_host = self.get_release_builder_path()
1641 info = self.host_info_store.get()
1642 os_from_label = info.get_label_value(self.VERSION_PREFIX)
1643 if not os_from_label:
1644 logging.debug('No existing %s label detected', self.VERSION_PREFIX)
1645 return True
1646
1647 # known cases where the version label will not match the
1648 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1649 # * Tests for the `arc-presubmit` append "-cheetsth" to the label.
1650 if os_from_label.endswith(provision.CHEETS_SUFFIX):
1651 logging.debug('%s label with %s suffix detected, this suffix will'
1652 ' be ignored when comparing label.',
1653 self.VERSION_PREFIX, provision.CHEETS_SUFFIX)
1654 os_from_label = os_from_label[:-len(provision.CHEETS_SUFFIX)]
1655 logging.debug('OS version from host: %s; OS verision cached in '
1656 'label: %s', os_from_host, os_from_label)
1657 return os_from_label == os_from_host
Dan Shi549fb822015-03-24 18:01:11 -07001658
1659
Laurence Goodby778c9a42017-05-24 19:24:07 -07001660 def cleanup_services(self):
1661 """Reinitializes the device for cleanup.
1662
1663 Subclasses may override this to customize the cleanup method.
1664
1665 To indicate failure of the reset, the implementation may raise
1666 any of:
1667 error.AutoservRunError
1668 error.AutotestRunError
1669 FactoryImageCheckerException
1670
1671 @raises error.AutoservRunError
1672 @raises error.AutotestRunError
1673 @raises error.FactoryImageCheckerException
1674 """
1675 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001676 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001677
1678
Gregory Nisbetec615d62020-12-11 17:59:20 +00001679 def cleanup(self):
1680 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001681 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001682 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001683 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001684 except (error.AutotestRunError, error.AutoservRunError,
1685 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001686 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001687
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001688 # cleanup routines, i.e. reboot the machine.
Gregory Nisbetec615d62020-12-11 17:59:20 +00001689 super(CrosHost, self).cleanup()
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001690
Simran Basi5e6339a2013-03-21 11:34:32 -07001691 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001692 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001693 self._cleanup_poweron()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001694
Gregory Nisbetec615d62020-12-11 17:59:20 +00001695
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001696 def reboot(self, **dargs):
1697 """
1698 This function reboots the site host. The more generic
1699 RemoteHost.reboot() performs sync and sleeps for 5
1700 seconds. This is not necessary for Chrome OS devices as the
1701 sync should be finished in a short time during the reboot
1702 command.
1703 """
Gregory Nisbetec615d62020-12-11 17:59:20 +00001704 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001705 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001706 dargs['reboot_cmd'] = ('sleep 1; '
1707 'reboot & sleep %d; '
1708 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001709 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001710 if 'fastsync' not in dargs:
1711 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001712
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001713 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001714 # Record who called us
1715 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001716 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001717 'dut_host_name' : self.hostname,
1718 'success' : True}
1719 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001720 'caller' : "%s:%s" % (orig.co_filename,
1721 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001722 'success' : True,
1723 'error' : ''}
1724
Vincent Palatin80780b22016-07-27 16:02:37 +02001725 try:
1726 super(CrosHost, self).reboot(**dargs)
1727 except Exception as e:
1728 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001729 metric_debug_fields['success'] = False
1730 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001731 raise
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001732
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001733 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001734 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001735 """
1736 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001737
1738 @param suspend_time: How long to suspend as integer seconds.
1739 @param suspend_cmd: Suspend command to execute.
1740 @param allow_early_resume: If False and if device resumes before
1741 |suspend_time|, throw an error.
1742
1743 @exception AutoservSuspendError Host resumed earlier than
1744 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001745 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001746
1747 if suspend_cmd is None:
1748 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001749 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001750 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001751 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001752 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1753 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001754
1755
Simran Basiec564392014-08-25 16:48:09 -07001756 def upstart_status(self, service_name):
1757 """Check the status of an upstart init script.
1758
1759 @param service_name: Service to look up.
1760
1761 @returns True if the service is running, False otherwise.
1762 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001763 return 'start/running' in self.run('status %s' % service_name,
1764 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001765
Tom Hughese9552342018-12-18 14:29:25 -08001766 def upstart_stop(self, service_name):
1767 """Stops an upstart job if it's running.
1768
1769 @param service_name: Service to stop
1770
1771 @returns True if service has been stopped or was already stopped
1772 False otherwise.
1773 """
1774 if not self.upstart_status(service_name):
1775 return True
1776
1777 result = self.run('stop %s' % service_name, ignore_status=True)
1778 if result.exit_status != 0:
1779 return False
1780 return True
1781
1782 def upstart_restart(self, service_name):
1783 """Restarts (or starts) an upstart job.
1784
1785 @param service_name: Service to start/restart
1786
1787 @returns True if service has been started/restarted, False otherwise.
1788 """
1789 cmd = 'start'
1790 if self.upstart_status(service_name):
1791 cmd = 'restart'
1792 cmd = cmd + ' %s' % service_name
1793 result = self.run(cmd)
1794 if result.exit_status != 0:
1795 return False
1796 return True
Simran Basiec564392014-08-25 16:48:09 -07001797
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001798 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001799 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001800
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001801 Tests for the following conditions:
1802 1. All conditions tested by the parent version of this
1803 function.
1804 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001805 3. Sufficient space in /mnt/stateful_partition/encrypted.
1806 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001807
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001808 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001809 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001810 default_kilo_inodes_required = CONFIG.get_config_value(
1811 'SERVER', 'kilo_inodes_required', type=int, default=100)
1812 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1813 kilo_inodes_required = CONFIG.get_config_value(
1814 'SERVER', 'kilo_inodes_required_%s' % board,
1815 type=int, default=default_kilo_inodes_required)
1816 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001817 self.check_diskspace(
1818 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001819 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001820 'SERVER', 'gb_diskspace_required', type=float,
1821 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001822 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1823 # Not all targets build with encrypted stateful support.
1824 if self.path_exists(encrypted_stateful_path):
1825 self.check_diskspace(
1826 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001827 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001828 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1829 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001830
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001831 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001832
beepsc87ff602013-07-31 21:53:00 -07001833 # Factory images don't run update engine,
1834 # goofy controls dbus on these DUTs.
1835 if not self._is_factory_image():
1836 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001837
1838
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001839 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
Kuo Jen Wei50ef6e82021-03-29 12:04:47 +08001840 def wait_for_service(self, service_name):
1841 """Wait for target status of an upstart init script.
1842
1843 @param service_name: Service to wait for.
1844 """
1845 if not self.upstart_status(service_name):
1846 raise error.AutoservError('Service %s not running.' % service_name)
1847
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001848 def wait_for_system_services(self):
1849 """Waits for system-services to be running.
1850
1851 Sometimes, update_engine will take a while to update firmware, so we
1852 should give this some time to finish. See crbug.com/765686#c38 for
1853 details.
1854 """
Kuo Jen Wei50ef6e82021-03-29 12:04:47 +08001855 self.wait_for_service('system-services')
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001856
1857
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001858 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001859 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001860 message = 'Beginning verify for host %s board %s model %s'
1861 info = self.host_info_store.get()
1862 message %= (self.hostname, info.board, info.model)
1863 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001864 try:
1865 self._repair_strategy.verify(self)
1866 except hosts.AutoservVerifyDependencyError as e:
1867 # We don't want flag a DUT as failed if only non-critical
1868 # verifier(s) failed during the repair.
1869 if e.is_critical():
1870 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001871
1872
Fang Deng96667ca2013-08-01 17:46:18 -07001873 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001874 connect_timeout=None, alive_interval=None,
1875 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001876 """Override default make_ssh_command to use options tuned for Chrome OS.
1877
1878 Tuning changes:
1879 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1880 connection failure. Consistency with remote_access.sh.
1881
Samuel Tan2ce155b2015-06-23 18:24:38 -07001882 - ServerAliveInterval=900; which causes SSH to ping connection every
1883 900 seconds. In conjunction with ServerAliveCountMax ensures
1884 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001885 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001886 the test completed successfully. Later increased from 180 seconds to
1887 900 seconds to account for tests where the DUT is suspended for
1888 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001889
1890 - ServerAliveCountMax=3; consistency with remote_access.sh.
1891
1892 - ConnectAttempts=4; reduce flakiness in connection errors;
1893 consistency with remote_access.sh.
1894
1895 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1896 Host keys change with every new installation, don't waste
1897 memory/space saving them.
1898
1899 - SSH protocol forced to 2; needed for ServerAliveInterval.
1900
1901 @param user User name to use for the ssh connection.
1902 @param port Port on the target host to use for ssh connection.
1903 @param opts Additional options to the ssh command.
1904 @param hosts_file Ignored.
1905 @param connect_timeout Ignored.
1906 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001907 @param alive_count_max Ignored.
1908 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001909 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001910 options = ' '.join([opts, '-o Protocol=2'])
1911 return super(CrosHost, self).make_ssh_command(
1912 user=user, port=port, opts=options, hosts_file='/dev/null',
1913 connect_timeout=30, alive_interval=900, alive_count_max=3,
1914 connection_attempts=4)
1915
1916
Jason Abeleb6f924f2013-11-13 16:01:54 -08001917 def syslog(self, message, tag='autotest'):
1918 """Logs a message to syslog on host.
1919
1920 @param message String message to log into syslog
1921 @param tag String tag prefix for syslog
1922
1923 """
1924 self.run('logger -t "%s" "%s"' % (tag, message))
1925
1926
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001927 def _ping_check_status(self, status):
1928 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001929
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001930 @param status Check the ping status against this value.
1931 @return True iff `status` and the result of ping are the same
1932 (i.e. both True or both False).
1933
1934 """
Abhishek Pandit-Subedi038df162020-09-14 16:37:43 -07001935 ping_val = utils.ping(self.hostname,
1936 tries=1,
1937 deadline=1,
1938 timeout=2,
1939 ignore_timeout=True)
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001940 return not (status ^ (ping_val == 0))
1941
1942 def _ping_wait_for_status(self, status, timeout):
1943 """Wait for the host to have a given status (UP or DOWN).
1944
1945 Status is checked by polling. Polling will not last longer
1946 than the number of seconds in `timeout`. The polling
1947 interval will be long enough that only approximately
1948 _PING_WAIT_COUNT polling cycles will be executed, subject
1949 to a maximum interval of about one minute.
1950
1951 @param status Waiting will stop immediately if `ping` of the
1952 host returns this status.
1953 @param timeout Poll for at most this many seconds.
1954 @return True iff the host status from `ping` matched the
1955 requested status at the time of return.
1956
1957 """
1958 # _ping_check_status() takes about 1 second, hence the
1959 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001960 # FIXME: if the ping command errors then _ping_check_status()
1961 # returns instantly. If timeout is also smaller than twice
1962 # _PING_WAIT_COUNT then the while loop below forks many
1963 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1964 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1965 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001966 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1967 end_time = time.time() + timeout
1968 while time.time() <= end_time:
1969 if self._ping_check_status(status):
1970 return True
1971 if poll_interval > 0:
1972 time.sleep(poll_interval)
1973
1974 # The last thing we did was sleep(poll_interval), so it may
1975 # have been too long since the last `ping`. Check one more
1976 # time, just to be sure.
1977 return self._ping_check_status(status)
1978
1979 def ping_wait_up(self, timeout):
1980 """Wait for the host to respond to `ping`.
1981
1982 N.B. This method is not a reliable substitute for
1983 `wait_up()`, because a host that responds to ping will not
1984 necessarily respond to ssh. This method should only be used
1985 if the target DUT can be considered functional even if it
1986 can't be reached via ssh.
1987
1988 @param timeout Minimum time to allow before declaring the
1989 host to be non-responsive.
1990 @return True iff the host answered to ping before the timeout.
1991
1992 """
Andrew Luo4be621d2020-03-21 07:01:13 -07001993 if self.use_icmp:
1994 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
1995 else:
1996 logging.debug('Using SSH instead of ICMP for ping_wait_up.')
1997 return self.wait_up(timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001998
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001999 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002000 """Wait until the host no longer responds to `ping`.
2001
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08002002 This function can be used as a slightly faster version of
2003 `wait_down()`, by avoiding potentially long ssh timeouts.
2004
2005 @param timeout Minimum time to allow for the host to become
2006 non-responsive.
2007 @return True iff the host quit answering ping before the
2008 timeout.
2009
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002010 """
Andrew Luo4be621d2020-03-21 07:01:13 -07002011 if self.use_icmp:
2012 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
2013 else:
2014 logging.debug('Using SSH instead of ICMP for ping_wait_down.')
2015 return self.wait_down(timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002016
Anand K Mistry50f218e2020-07-31 14:50:15 +10002017 def _is_host_port_forwarded(self):
Garry Wanga2e78172020-09-09 23:49:07 -07002018 """Checks if the dut is connected over port forwarding.
Anand K Mistry50f218e2020-07-31 14:50:15 +10002019
2020 N.B. This method does not detect all situations where port forwarding is
2021 occurring. Namely, running autotest on the dut may result in a
2022 false-positive, and port forwarding using a different machine on the
2023 same network will be a false-negative.
2024
2025 @return True if the dut is connected over port forwarding
2026 False otherwise
2027 """
Garry Wanga2e78172020-09-09 23:49:07 -07002028 is_localhost = self.hostname in ['localhost', '127.0.0.1']
2029 is_forwarded = is_localhost and not self.is_default_port
2030 if is_forwarded:
2031 logging.info('Detected DUT connected by port forwarding')
2032 return is_forwarded
Anand K Mistry50f218e2020-07-31 14:50:15 +10002033
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002034 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002035 """Wait for the client to enter low-power sleep mode.
2036
2037 The test for "is asleep" can't distinguish a system that is
2038 powered off; to confirm that the unit was asleep, it is
2039 necessary to force resume, and then call
2040 `test_wait_for_resume()`.
2041
2042 This function is expected to be called from a test as part
2043 of a sequence like the following:
2044
2045 ~~~~~~~~
2046 boot_id = host.get_boot_id()
2047 # trigger sleep on the host
2048 host.test_wait_for_sleep()
2049 # trigger resume on the host
2050 host.test_wait_for_resume(boot_id)
2051 ~~~~~~~~
2052
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002053 @param sleep_timeout time limit in seconds to allow the host sleep.
2054
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002055 @exception TestFail The host did not go to sleep within
2056 the allowed time.
2057 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002058 if sleep_timeout is None:
2059 sleep_timeout = self.SLEEP_TIMEOUT
2060
Anand K Mistry50f218e2020-07-31 14:50:15 +10002061 # If the dut is accessed over SSH port-forwarding, `ping` is not useful
2062 # for detecting the dut is down since a ping to localhost will always
2063 # succeed. In this case, fall back to wait_down() which uses SSH.
2064 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07002065 success = self.wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002066 else:
Garry Wanga2e78172020-09-09 23:49:07 -07002067 success = self.ping_wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002068
2069 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002070 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002071 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002072
2073
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002074 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002075 """Wait for the client to resume from low-power sleep mode.
2076
2077 The `old_boot_id` parameter should be the value from
2078 `get_boot_id()` obtained prior to entering sleep mode. A
2079 `TestFail` exception is raised if the boot id changes.
2080
2081 See @ref test_wait_for_sleep for more on this function's
2082 usage.
2083
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002084 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002085 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002086 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002087
2088 @exception TestFail The host did not respond within the
2089 allowed time.
2090 @exception TestFail The host responded, but the boot id test
2091 indicated a reboot rather than a sleep
2092 cycle.
2093 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002094 if resume_timeout is None:
2095 resume_timeout = self.RESUME_TIMEOUT
2096
2097 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002098 raise error.TestFail(
2099 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002100 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002101 else:
2102 new_boot_id = self.get_boot_id()
2103 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002104 logging.error('client rebooted (old boot %s, new boot %s)',
2105 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002106 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002107 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002108
2109
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002110 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002111 """Wait for the client to shut down.
2112
2113 The test for "has shut down" can't distinguish a system that
2114 is merely asleep; to confirm that the unit was down, it is
2115 necessary to force boot, and then call test_wait_for_boot().
2116
2117 This function is expected to be called from a test as part
2118 of a sequence like the following:
2119
2120 ~~~~~~~~
2121 boot_id = host.get_boot_id()
2122 # trigger shutdown on the host
2123 host.test_wait_for_shutdown()
2124 # trigger boot on the host
2125 host.test_wait_for_boot(boot_id)
2126 ~~~~~~~~
2127
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002128 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002129 @exception TestFail The host did not shut down within the
2130 allowed time.
2131 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002132 if shutdown_timeout is None:
2133 shutdown_timeout = self.SHUTDOWN_TIMEOUT
2134
Anand K Mistry50f218e2020-07-31 14:50:15 +10002135 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07002136 success = self.wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002137 else:
Garry Wanga2e78172020-09-09 23:49:07 -07002138 success = self.ping_wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002139
2140 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002141 raise error.TestFail(
2142 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002143 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002144
2145
2146 def test_wait_for_boot(self, old_boot_id=None):
2147 """Wait for the client to boot from cold power.
2148
2149 The `old_boot_id` parameter should be the value from
2150 `get_boot_id()` obtained prior to shutting down. A
2151 `TestFail` exception is raised if the boot id does not
2152 change. The boot id test is omitted if `old_boot_id` is not
2153 specified.
2154
2155 See @ref test_wait_for_shutdown for more on this function's
2156 usage.
2157
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002158 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002159 shut down.
2160
2161 @exception TestFail The host did not respond within the
2162 allowed time.
2163 @exception TestFail The host responded, but the boot id test
2164 indicated that there was no reboot.
2165 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002166 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002167 raise error.TestFail(
2168 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002169 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002170 elif old_boot_id:
2171 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002172 logging.error('client not rebooted (boot %s)',
2173 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002174 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002175 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07002176
2177
2178 @staticmethod
2179 def check_for_rpm_support(hostname):
2180 """For a given hostname, return whether or not it is powered by an RPM.
2181
Simran Basi1df55112013-09-06 11:25:09 -07002182 @param hostname: hostname to check for rpm support.
2183
Simran Basid5e5e272012-09-24 15:23:59 -07002184 @return None if this host does not follows the defined naming format
2185 for RPM powered DUT's in the lab. If it does follow the format,
2186 it returns a regular expression MatchObject instead.
2187 """
Fang Dengbaff9082015-01-06 13:46:15 -08002188 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002189
2190
2191 def has_power(self):
2192 """For this host, return whether or not it is powered by an RPM.
2193
2194 @return True if this host is in the CROS lab and follows the defined
2195 naming format.
2196 """
Fang Deng0ca40e22013-08-27 17:47:44 -07002197 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002198
2199
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002200 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07002201 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002202
2203 @param state Specifies which power state to set to DUT
2204 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002205 use. By default "RPM" or "CCD" will be used based
2206 on servo type. Valid values from
2207 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002208
2209 """
2210 ACCEPTABLE_STATES = ['ON', 'OFF']
2211
Garry Wang5e5538a2019-04-08 15:36:18 -07002212 if not power_method:
2213 power_method = self.get_default_power_method()
2214
2215 state = state.upper()
2216 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002217 raise error.TestError('State must be one of: %s.'
2218 % (ACCEPTABLE_STATES,))
2219
2220 if power_method == self.POWER_CONTROL_SERVO:
2221 logging.info('Setting servo port J10 to %s', state)
2222 self.servo.set('prtctl3_pwren', state.lower())
2223 time.sleep(self._USB_POWER_TIMEOUT)
2224 elif power_method == self.POWER_CONTROL_MANUAL:
2225 logging.info('You have %d seconds to set the AC power to %s.',
2226 self._POWER_CYCLE_TIMEOUT, state)
2227 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07002228 elif power_method == self.POWER_CONTROL_CCD:
2229 servo_role = 'src' if state == 'ON' else 'snk'
2230 logging.info('servo ccd power pass through detected,'
2231 ' changing servo_role to %s.', servo_role)
2232 self.servo.set_servo_v4_role(servo_role)
2233 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07002234 # Make sure we don't leave DUT with no power(servo_role=snk)
2235 # when DUT is not pingable, as we raise a exception here
2236 # that may break a power cycle in the middle.
2237 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07002238 raise error.AutoservError(
2239 'DUT failed to regain network connection after %d seconds.'
2240 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002241 else:
2242 if not self.has_power():
2243 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002244 self._add_rpm_changed_tag()
Derek Beckett5ea7bf32021-08-03 13:09:20 -07002245 # TODO b/195443964: Re-wire as needed once TLW is available.
Simran Basid5e5e272012-09-24 15:23:59 -07002246
2247
Garry Wang5e5538a2019-04-08 15:36:18 -07002248 def power_off(self, power_method=None):
2249 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002250
2251 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002252 use. By default "RPM" or "CCD" will be used based
2253 on servo type. Valid values from
2254 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002255
2256 """
Derek Beckettb66e5c82020-08-12 15:31:02 -07002257 self._sync_if_up()
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002258 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002259
Derek Beckettb66e5c82020-08-12 15:31:02 -07002260 def _check_supported(self):
2261 """Throw an error if dts mode control is not supported."""
2262 if not self.servo_pwr_supported:
2263 raise error.TestFail('power_state controls not supported')
2264
2265 def _sync_if_up(self):
2266 """Run sync on the DUT and wait for completion if the DUT is up.
2267
2268 Additionally, try to sync and ignore status if its not up.
2269
2270 Useful prior to reboots to ensure files are written to disc.
2271
2272 """
2273 if self.is_up_fast():
2274 self.run("sync")
2275 return
2276 # If it is not up, attempt to sync in the rare event the DUT is up but
2277 # doesn't respond to a ping. Ignore any errors.
2278 try:
2279 self.run("sync", ignore_status=True, timeout=1)
2280 except Exception:
2281 pass
2282
2283 def power_off_via_servo(self):
2284 """Force the DUT to power off.
2285
2286 The DUT is guaranteed to be off at the end of this call,
2287 regardless of its previous state, provided that there is
2288 working EC and boot firmware. There is no requirement for
2289 working OS software.
2290
2291 """
2292 self._check_supported()
2293 self._sync_if_up()
2294 self.servo.set_nocheck('power_state', 'off')
2295
2296 def power_on_via_servo(self, rec_mode='on'):
2297 """Force the DUT to power on.
2298
2299 Prior to calling this function, the DUT must be powered off,
2300 e.g. with a call to `power_off()`.
2301
2302 At power on, recovery mode is set as specified by the
2303 corresponding argument. When booting with recovery mode on, it
2304 is the caller's responsibility to unplug/plug in a bootable
2305 external storage device.
2306
2307 If the DUT requires a delay after powering on but before
2308 processing inputs such as USB stick insertion, the delay is
2309 handled by this method; the caller is not responsible for such
2310 delays.
2311
2312 @param rec_mode Setting of recovery mode to be applied at
2313 power on. default: REC_OFF aka 'off'
2314
2315 """
2316 self._check_supported()
2317 self.servo.set_nocheck('power_state', rec_mode)
2318
2319 def reset_via_servo(self):
2320 """Force the DUT to reset.
2321
2322 The DUT is guaranteed to be on at the end of this call,
2323 regardless of its previous state, provided that there is
2324 working OS software. This also guarantees that the EC has
2325 been restarted.
2326
2327 """
2328 self._check_supported()
2329 self._sync_if_up()
2330 self.servo.set_nocheck('power_state', 'reset')
2331
Simran Basid5e5e272012-09-24 15:23:59 -07002332
Garry Wang5e5538a2019-04-08 15:36:18 -07002333 def power_on(self, power_method=None):
2334 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002335
2336 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002337 use. By default "RPM" or "CCD" will be used based
2338 on servo type. Valid values from
2339 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002340
2341 """
2342 self._set_power('ON', power_method)
2343
2344
Garry Wang5e5538a2019-04-08 15:36:18 -07002345 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002346 """Cycle power to this host by turning it OFF, then ON.
2347
2348 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002349 use. By default "RPM" or "CCD" will be used based
2350 on servo type. Valid values from
2351 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002352
2353 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002354 if not power_method:
2355 power_method = self.get_default_power_method()
2356
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002357 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002358 self.POWER_CONTROL_MANUAL,
2359 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002360 self.power_off(power_method=power_method)
2361 time.sleep(self._POWER_CYCLE_TIMEOUT)
2362 self.power_on(power_method=power_method)
2363 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002364 self._add_rpm_changed_tag()
Derek Beckett5ea7bf32021-08-03 13:09:20 -07002365 # TODO b/195443964: Re-wire as needed once TLW is available.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002366
2367
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002368 def get_platform_from_fwid(self):
2369 """Determine the platform from the crossystem fwid.
2370
2371 @returns a string representing this host's platform.
2372 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002373 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002374 crossystem = utils.Crossystem(self)
2375 crossystem.init()
2376 # Extract fwid value and use the leading part as the platform id.
2377 # fwid generally follow the format of {platform}.{firmware version}
2378 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2379 platform = crossystem.fwid().split('.')[0].lower()
2380 # Newer platforms start with 'Google_' while the older ones do not.
2381 return platform.replace('google_', '')
2382
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002383
Simran Basic6f1f7a2012-10-16 10:47:46 -07002384 def get_platform(self):
2385 """Determine the correct platform label for this host.
2386
2387 @returns a string representing this host's platform.
2388 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002389 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2390 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002391 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002392 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002393 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002394 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002395
2396
Greg Edelstona7b05d12020-04-01 16:00:51 -06002397 def get_model_from_cros_config(self):
2398 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002399
Greg Edelstona7b05d12020-04-01 16:00:51 -06002400 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002401 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002402 return cros_config.call_cros_config_get_output('/ name',
2403 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002404
2405
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002406 def get_architecture(self):
2407 """Determine the correct architecture label for this host.
2408
2409 @returns a string representing this host's architecture.
2410 """
2411 crossystem = utils.Crossystem(self)
2412 crossystem.init()
2413 return crossystem.arch()
2414
2415
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002416 def get_chrome_version(self):
2417 """Gets the Chrome version number and milestone as strings.
2418
2419 Invokes "chrome --version" to get the version number and milestone.
2420
2421 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2422 current Chrome version number as a string (in the form "W.X.Y.Z")
2423 and "milestone" is the first component of the version number
2424 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2425 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2426 of "chrome --version" and the milestone will be the empty string.
2427
2428 """
MK Ryu35d661e2014-09-25 17:44:10 -07002429 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002430 return utils.parse_chrome_version(version_string)
2431
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002432
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002433 def get_ec_version(self):
2434 """Get the ec version as strings.
2435
2436 @returns a string representing this host's ec version.
2437 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002438 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002439 result = self.run(command, ignore_status=True)
2440 if result.exit_status != 0:
2441 return ''
2442 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002443
2444
2445 def get_firmware_version(self):
2446 """Get the firmware version as strings.
2447
2448 @returns a string representing this host's firmware version.
2449 """
2450 crossystem = utils.Crossystem(self)
2451 crossystem.init()
2452 return crossystem.fwid()
2453
2454
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08002455 def get_hardware_id(self):
2456 """Get hardware id as strings.
2457
2458 @returns a string representing this host's hardware id.
2459 """
2460 crossystem = utils.Crossystem(self)
2461 crossystem.init()
2462 return crossystem.hwid()
2463
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002464 def get_hardware_revision(self):
2465 """Get the hardware revision as strings.
2466
2467 @returns a string representing this host's hardware revision.
2468 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002469 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002470 result = self.run(command, ignore_status=True)
2471 if result.exit_status != 0:
2472 return ''
2473 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002474
2475
2476 def get_kernel_version(self):
2477 """Get the kernel version as strings.
2478
2479 @returns a string representing this host's kernel version.
2480 """
2481 return self.run('uname -r').stdout.strip()
2482
2483
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002484 def get_cpu_name(self):
2485 """Get the cpu name as strings.
2486
2487 @returns a string representing this host's cpu name.
2488 """
2489
2490 # Try get cpu name from device tree first
2491 if self.path_exists('/proc/device-tree/compatible'):
2492 command = ' | '.join(
2493 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2494 'tail -1'])
2495 return self.run(command).stdout.strip().replace(',', ' ')
2496
2497 # Get cpu name from uname -p
2498 command = 'uname -p'
2499 ret = self.run(command).stdout.strip()
2500
2501 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2502 # Try get cpu name from /proc/cpuinfo instead
2503 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2504 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2505 self = self.run(command).stdout.strip()
2506
2507 # Remove bloat from CPU name, for example
2508 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2509 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2510 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2511 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2512 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2513 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2514
2515
2516 def get_screen_resolution(self):
2517 """Get the screen(s) resolution as strings.
2518 In case of more than 1 monitor, return resolution for each monitor
2519 separate with plus sign.
2520
2521 @returns a string representing this host's screen(s) resolution.
2522 """
2523 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2524 ret = self.run(command, ignore_status=True)
2525 # We might have Chromebox without a screen
2526 if ret.exit_status != 0:
2527 return ''
2528 return ret.stdout.strip().replace('\n', '+')
2529
2530
2531 def get_mem_total_gb(self):
2532 """Get total memory available in the system in GiB (2^20).
2533
2534 @returns an integer representing total memory
2535 """
2536 mem_total_kb = self.read_from_meminfo('MemTotal')
2537 kb_in_gb = float(2 ** 20)
2538 return int(round(mem_total_kb / kb_in_gb))
2539
2540
2541 def get_disk_size_gb(self):
2542 """Get size of disk in GB (10^9)
2543
2544 @returns an integer representing size of disk, 0 in Error Case
2545 """
2546 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2547 result = self.run(command, ignore_status=True)
2548 if result.exit_status != 0:
2549 return 0
2550 _, _, block, _ = re.split(r' +', result.stdout.strip())
2551 byte_per_block = 1024.0
2552 disk_kb_in_gb = 1e9
2553 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2554
2555
2556 def get_battery_size(self):
2557 """Get size of battery in Watt-hour via sysfs
2558
2559 This method assumes that battery support voltage_min_design and
2560 charge_full_design sysfs.
2561
2562 @returns a float representing Battery size, 0 if error.
2563 """
2564 # sysfs report data in micro scale
2565 battery_scale = 1e6
2566
2567 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2568 result = self.run(command, ignore_status=True)
2569 if result.exit_status != 0:
2570 return 0
2571 voltage = float(result.stdout.strip()) / battery_scale
2572
2573 command = 'cat /sys/class/power_supply/*/charge_full_design'
2574 result = self.run(command, ignore_status=True)
2575 if result.exit_status != 0:
2576 return 0
2577 amphereHour = float(result.stdout.strip()) / battery_scale
2578
2579 return voltage * amphereHour
2580
2581
2582 def get_low_battery_shutdown_percent(self):
2583 """Get the percent-based low-battery shutdown threshold.
2584
2585 @returns a float representing low-battery shutdown percent, 0 if error.
2586 """
2587 ret = 0.0
2588 try:
2589 command = 'check_powerd_config --low_battery_shutdown_percent'
2590 ret = float(self.run(command).stdout)
2591 except error.CmdError:
2592 logging.debug("Can't run %s", command)
2593 except ValueError:
2594 logging.debug("Didn't get number from %s", command)
2595
2596 return ret
2597
2598
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002599 def has_hammer(self):
2600 """Check whether DUT has hammer device or not.
2601
2602 @returns boolean whether device has hammer or not
2603 """
2604 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2605 return self.run(command, ignore_status=True).exit_status == 0
2606
2607
Niranjan Kumar34618872017-05-31 12:57:09 -07002608 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002609 """Returns True if the specified switch was provided to Chrome.
2610
2611 @param switch The chrome switch to search for.
2612 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002613
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002614 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2615 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002616
2617
2618 def oobe_triggers_update(self):
2619 """Returns True if this host has an OOBE flow during which
2620 it will perform an update check and perhaps an update.
2621 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2622 As more such flows are developed, code handling them needs
2623 to be added here.
2624
2625 @return Boolean indicating whether this host's OOBE triggers an update.
2626 """
2627 return self.is_chrome_switch_present(
2628 '--enterprise-enable-zero-touch-enrollment=hands-off')
2629
2630
Kevin Chenga2619dc2016-03-28 11:42:08 -07002631 # TODO(kevcheng): change this to just return the board without the
2632 # 'board:' prefix and fix up all the callers. Also look into removing the
2633 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002634 def get_board(self):
2635 """Determine the correct board label for this host.
2636
2637 @returns a string representing this host's board.
2638 """
2639 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2640 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002641 return (ds_constants.BOARD_PREFIX +
2642 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002643
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002644 def get_channel(self):
2645 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002646
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002647 @returns: a string represeting this host's build channel.
2648 (stable, dev, beta). None on fail.
2649 """
2650 return lsbrelease_utils.get_chromeos_channel(
2651 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002652
Kevin Chenga328da62016-03-31 10:49:04 -07002653 def get_power_supply(self):
2654 """
2655 Determine what type of power supply the host has
2656
2657 @returns a string representing this host's power supply.
2658 'power:battery' when the device has a battery intended for
2659 extended use
2660 'power:AC_primary' when the device has a battery not intended
2661 for extended use (for moving the machine, etc)
2662 'power:AC_only' when the device has no battery at all.
2663 """
Jack Rosenthal01ee2cf2021-03-30 21:01:32 -06002664 psu = self.run(command='cros_config /hardware-properties psu-type',
2665 ignore_status=True)
Kevin Chenga328da62016-03-31 10:49:04 -07002666 if psu.exit_status:
Jack Rosenthal01ee2cf2021-03-30 21:01:32 -06002667 # Assume battery if unspecified in cros_config.
Kevin Chenga328da62016-03-31 10:49:04 -07002668 return 'power:battery'
2669
2670 psu_str = psu.stdout.strip()
2671 if psu_str == 'unknown':
2672 return None
2673
2674 return 'power:%s' % psu_str
2675
2676
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002677 def has_battery(self):
2678 """Determine if DUT has a battery.
2679
2680 Returns:
2681 Boolean, False if known not to have battery, True otherwise.
2682 """
Jack Rosenthal01ee2cf2021-03-30 21:01:32 -06002683 return self.get_power_supply() == 'power:battery'
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002684
2685
Kevin Chenga328da62016-03-31 10:49:04 -07002686 def get_servo(self):
2687 """Determine if the host has a servo attached.
2688
2689 If the host has a working servo attached, it should have a servo label.
2690
2691 @return: string 'servo' if the host has servo attached. Otherwise,
2692 returns None.
2693 """
2694 return 'servo' if self._servo_host else None
2695
2696
Kevin Chenga328da62016-03-31 10:49:04 -07002697 def has_internal_display(self):
2698 """Determine if the device under test is equipped with an internal
2699 display.
2700
2701 @return: 'internal_display' if one is present; None otherwise.
2702 """
2703 from autotest_lib.client.cros.graphics import graphics_utils
2704 from autotest_lib.client.common_lib import utils as common_utils
2705
2706 def __system_output(cmd):
2707 return self.run(cmd).stdout
2708
2709 def __read_file(remote_path):
2710 return self.run('cat %s' % remote_path).stdout
2711
2712 # Hijack the necessary client functions so that we can take advantage
2713 # of the client lib here.
2714 # FIXME: find a less hacky way than this
2715 original_system_output = utils.system_output
2716 original_read_file = common_utils.read_file
2717 utils.system_output = __system_output
2718 common_utils.read_file = __read_file
2719 try:
2720 return ('internal_display' if graphics_utils.has_internal_display()
2721 else None)
2722 finally:
2723 utils.system_output = original_system_output
2724 common_utils.read_file = original_read_file
2725
2726
Dan Shi85276d42014-04-08 22:11:45 -07002727 def is_boot_from_usb(self):
2728 """Check if DUT is boot from USB.
2729
2730 @return: True if DUT is boot from usb.
2731 """
2732 device = self.run('rootdev -s -d').stdout.strip()
2733 removable = int(self.run('cat /sys/block/%s/removable' %
2734 os.path.basename(device)).stdout.strip())
2735 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002736
Otabek Kasimov77a40332020-10-20 15:40:03 -07002737 def is_boot_from_external_device(self):
2738 """Check if DUT is boot from external storage.
2739
2740 @return: True if DUT is boot from external storage.
2741 """
2742 boot_device = self.run('rootdev -s -d', ignore_status=True,
2743 timeout=60).stdout.strip()
2744 if not boot_device:
2745 logging.debug('Boot storage not detected on the host.')
2746 return False
2747 main_storage_cmd = ('. /usr/sbin/write_gpt.sh;'
2748 ' . /usr/share/misc/chromeos-common.sh;'
2749 ' load_base_vars; get_fixed_dst_drive')
2750 main_storage = self.run(main_storage_cmd,
2751 ignore_status=True,
2752 timeout=60).stdout.strip()
Otabek Kasimov723e8562020-12-08 13:29:34 -08002753 if not main_storage or boot_device != main_storage:
2754 logging.debug('Device booted from external storage storage.')
2755 return True
2756 logging.debug('Device booted from main storage.')
2757 return False
Helen Zhang17dae2b2014-11-11 09:25:52 -08002758
2759 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002760 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002761
2762 @param key: meminfo requested
2763
2764 @return the memory value as a string
2765
2766 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002767 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2768 logging.debug('%s', meminfo)
2769 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002770
2771
Rohit Makasana98e696f2016-06-03 18:48:10 -07002772 def get_cpu_arch(self):
2773 """Returns CPU arch of the device.
2774
2775 @return CPU architecture of the DUT.
2776 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002777 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002778 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2779 ignore_status=True).stdout:
2780 return 'x86_64'
2781 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2782 ignore_status=True).stdout:
2783 return 'arm'
2784 return 'i386'
2785
2786
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002787 def get_board_type(self):
2788 """
Nikolai Artemiev5b5b6802021-05-12 12:56:43 +10002789 Get the DUT's device type / form factor from cros_config. It can be one
2790 of CHROMEBOX, CHROMEBASE, CHROMEBOOK, or CHROMEBIT.
Danny Chan471a8d12015-08-18 14:57:41 -07002791
Nikolai Artemiev5b5b6802021-05-12 12:56:43 +10002792 @return form factor value from cros_config.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002793 """
Nikolai Artemiev5b5b6802021-05-12 12:56:43 +10002794
2795 device_type = self.run('cros_config /hardware-properties form-factor',
2796 ignore_status=True).stdout
2797 if device_type:
2798 return device_type
2799
2800 # TODO: remove lsb-release fallback once cros_config works everywhere
Danny Chan471a8d12015-08-18 14:57:41 -07002801 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2802 ignore_status=True).stdout
2803 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002804 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002805 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002806
2807
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002808 def get_arc_version(self):
2809 """Return ARC version installed on the DUT.
2810
2811 @returns ARC version as string if the CrOS build has ARC, else None.
2812 """
2813 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2814 ignore_status=True).stdout
2815 if arc_version:
2816 return arc_version.split('=')[-1].strip()
2817 return None
2818
2819
Gilad Arnolda76bef02015-09-29 13:55:15 -07002820 def get_os_type(self):
2821 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002822
2823
Kevin Chenga2619dc2016-03-28 11:42:08 -07002824 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002825 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002826 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002827
2828
2829 def get_default_power_method(self):
2830 """
2831 Get the default power method for power_on/off/cycle() methods.
2832 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2833 """
2834 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002835 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002836 if self.servo and self.servo.supports_built_in_pd_control():
2837 self._default_power_method = self.POWER_CONTROL_CCD
2838 else:
2839 logging.debug('Either servo is unitialized or the servo '
2840 'setup does not support pd controls. Falling '
2841 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002842 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002843
2844
2845 def find_usb_devices(self, idVendor, idProduct):
2846 """
2847 Get usb device sysfs name for specific device.
2848
2849 @param idVendor Vendor ID to search in sysfs directory.
2850 @param idProduct Product ID to search in sysfs directory.
2851
2852 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2853 """
2854 # Look for matching file and cut at position 7 to get dir name.
2855 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2856
2857 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2858 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2859
2860 # Use uniq -d to print duplicate line from both command
2861 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2862
2863 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2864
2865
2866 def bind_usb_device(self, usb_node):
2867 """
2868 Bind usb device
2869
2870 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2871 """
2872 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2873 self.run(cmd, ignore_status=True)
2874
2875
2876 def unbind_usb_device(self, usb_node):
2877 """
2878 Unbind usb device
2879
2880 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2881 """
2882 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2883 self.run(cmd, ignore_status=True)
2884
2885
2886 def get_wlan_ip(self):
2887 """
2888 Get ip address of wlan interface.
2889
2890 @return ip address of wlan or empty string if wlan is not connected.
2891 """
2892 cmds = [
2893 'iw dev', # List wlan physical device
2894 'grep Interface', # Grep only interface name
2895 'cut -f 2 -d" "', # Cut the name part
2896 'xargs ifconfig', # Feed it to ifconfig to get ip
2897 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2898 'cut -f 2 -d " "' # Cut the ip part
2899 ]
2900 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002901
2902 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2903 """
2904 Connect to wifi network
2905
2906 @param ssid SSID of the wifi network.
2907 @param passphrase Passphrase of the wifi network. None if not existed.
2908 @param security Security of the wifi network. Default to "psk" if
2909 passphase is given without security. Possible values
2910 are "none", "psk", "802_1x".
2911
2912 @return True if succeed, False if not.
2913 """
2914 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2915 if passphrase:
2916 cmd += ' ' + passphrase
2917 if security:
2918 cmd += ' ' + security
2919 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002920
2921 def get_device_repair_state(self):
2922 """Get device repair state"""
2923 return self._device_repair_state
2924
Otabek Kasimov44273d22021-02-26 17:13:24 -08002925 def is_marked_for_replacement(self):
2926 """Verify if device was marked for replacemnet during admin task."""
2927 expected_state = cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT
2928 return self.get_device_repair_state() == expected_state
2929
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002930 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002931 """Set device repair state.
2932
2933 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002934 file in result directory. The file will be read by Lucifer. The
2935 file will not be created if result directory not specified.
2936
2937 @params state: The new state for the device.
2938 @params resultdir: The path to result directory. If path not provided
2939 will be attempt to get retrieve it from job
2940 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002941 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002942 resultdir = resultdir or getattr(self.job, 'resultdir', '')
2943 if resultdir:
2944 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07002945 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002946 logging.info('Set device state as %s. '
2947 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07002948 else:
2949 logging.debug('Cannot write the device state due missing info '
2950 'about result dir.')
2951 self._device_repair_state = state
2952
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002953 def set_device_needs_replacement(self, resultdir=None):
2954 """Set device as required replacement.
2955
2956 @params resultdir: The path to result directory. If path not provided
2957 will be attempt to get retrieve it from job
2958 if present.
2959 """
2960 self.set_device_repair_state(
2961 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
2962 resultdir=resultdir)
2963
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002964 def _dut_is_accessible_by_verifier(self):
2965 """Check if DUT accessible by SSH or PING verifier.
Otabek Kasimov86062d02020-11-17 13:30:22 -08002966
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002967 @returns: bool, True - verifier marked as success.
2968 False - result not reachable, verifier did not success.
Otabek Kasimov86062d02020-11-17 13:30:22 -08002969 """
2970 if not self._repair_strategy:
2971 return False
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002972 dut_ssh = self._repair_strategy.verifier_is_good('ssh')
2973 dut_ping = self._repair_strategy.verifier_is_good('ping')
2974 return dut_ssh == hosts.VERIFY_SUCCESS or dut_ssh == hosts.VERIFY_SUCCESS
Otabek Kasimov86062d02020-11-17 13:30:22 -08002975
Otabek Kasimovd48389b2020-12-07 02:38:34 -08002976 def _stat_if_pingable_but_not_sshable(self):
2977 """Check if DUT pingable but failed SSH verifier."""
2978 if not self._repair_strategy:
2979 return
Derek Beckett5f4edf02021-07-27 14:56:44 -07002980 self._repair_strategy.verifier_is_good('ssh')
2981 self._repair_strategy.verifier_is_good('ping')
Otabek Kasimovd48389b2020-12-07 02:38:34 -08002982
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002983 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002984 """Check if device require manual attention to be fixed.
2985
2986 The state 'needs_manual_repair' can be set when auto repair cannot
2987 fix the device due hardware or cable issues.
2988 """
2989 # ignore the logic if state present
2990 # state can be set by any cros repair actions
Otabek Kasimov86062d02020-11-17 13:30:22 -08002991 if self.get_device_repair_state():
Otabek Kasimov6825b762020-06-23 23:42:44 -07002992 return
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002993 if self._dut_is_accessible_by_verifier():
2994 # DUT is accessible and we still have many options to repair it.
Otabek Kasimovc9812582020-10-08 18:52:52 -07002995 return
Otabek Kasimov9189ede2020-11-09 14:08:58 -08002996 needs_manual_repair = False
2997 dhp = self.health_profile
Otabek Kasimov69253822020-11-24 10:52:27 -08002998 if dhp and dhp.get_repair_fail_count() > 49:
2999 # 42 = 6 times during 7 days. (every 4 hour repair)
3000 # round up to 50 in case somebody will run some attempt on it.
Otabek Kasimovc9812582020-10-08 18:52:52 -07003001 logging.info(
Otabek Kasimov9189ede2020-11-09 14:08:58 -08003002 'DUT is not sshable and fail %s times.'
Otabek Kasimov69253822020-11-24 10:52:27 -08003003 ' Limit to try repair is 50 times',
Otabek Kasimov9189ede2020-11-09 14:08:58 -08003004 dhp.get_repair_fail_count())
3005 needs_manual_repair = True
3006
3007 if not needs_manual_repair:
3008 # We cannot ssh to the DUT and we have hardware or set-up issues
3009 # with servo then we need request manual repair for the DUT.
3010 servo_state_required_manual_fix = [
3011 servo_constants.SERVO_STATE_DUT_NOT_CONNECTED,
3012 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
3013 ]
3014 if self.get_servo_state() in servo_state_required_manual_fix:
3015 logging.info(
3016 'DUT required manual repair because it is not sshable'
3017 ' and possible have setup issue with Servo. Please'
3018 ' verify all connections and present of devices.')
3019 needs_manual_repair = True
3020
3021 if needs_manual_repair:
Otabek Kasimovc9812582020-10-08 18:52:52 -07003022 self.set_device_repair_state(
3023 cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)
Otabek Kasimov42506d02020-07-29 14:44:57 -07003024
Otabek Kasimov86062d02020-11-17 13:30:22 -08003025 def _reboot_labstation_if_needed(self):
3026 """Place request to reboot the labstation if DUT is not sshable.
3027
3028 @returns: None
3029 """
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003030 message_prefix = "Don't need to request servo-host reboot"
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08003031 if self._dut_is_accessible_by_verifier():
Otabek Kasimov86062d02020-11-17 13:30:22 -08003032 return
3033 if not self._servo_host:
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003034 logging.debug('%s as it not initialized', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003035 return
3036 if not self._servo_host.is_up_fast():
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003037 logging.debug('%s as servo-host is not sshable', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003038 return
3039 if not self._servo_host.is_labstation():
3040 logging.debug('Servo_v3 is not requested to reboot for the DUT')
3041 return
3042 usb_path = self._servo_host.get_main_servo_usb_path()
3043 if usb_path:
3044 connected_port = os.path.basename(os.path.normpath(usb_path))
3045 # Directly connected servo to the labstation looks like '1-5.3'
3046 # and when connected by hub - '1-5.2.3' or '1-5.2.1.3'. Where:
3047 # - '1-5' - port on labstation
3048 # - '2' or '2.1' - port on the hub or smart-hub
3049 # - '3' - port on servo hub
3050 if len(connected_port.split('.')) > 2:
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003051 logging.debug('%s as servo connected by hub', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003052 return
3053 self._servo_host.request_reboot()
3054 logging.info('Requested labstation reboot because DUT is not sshable')
3055
Otabek Kasimov42506d02020-07-29 14:44:57 -07003056 def is_file_system_writable(self, testdirs=None):
3057 """Check is the file systems are writable.
3058
3059 The standard linux response to certain unexpected file system errors
3060 (including hardware errors in block devices) is to change the file
3061 system status to read-only. This checks that that hasn't happened.
3062
3063 @param testdirs: List of directories to check. If no data provided
3064 then '/mnt/stateful_partition' and '/var/tmp'
3065 directories will be checked.
3066
3067 @returns boolean whether file-system writable.
3068 """
3069 def _check_dir(testdir):
3070 # check if we can create a file
3071 filename = os.path.join(testdir, 'writable_my_test_file')
3072 command = 'touch %s && rm %s' % (filename, filename)
3073 rv = self.run(command=command,
3074 timeout=30,
3075 ignore_status=True)
3076 is_writable = rv.exit_status == 0
3077 if not is_writable:
3078 logging.info('Cannot create a file in "%s"!'
3079 ' Probably the FS is read-only', testdir)
3080 logging.info("FileSystem is not writable!")
3081 return False
3082 return True
3083
3084 if not testdirs or len(testdirs) == 0:
3085 # N.B. Order matters here: Encrypted stateful is loop-mounted
3086 # from a file in unencrypted stateful, so we don't test for
3087 # errors in encrypted stateful if unencrypted fails.
3088 testdirs = ['/mnt/stateful_partition', '/var/tmp']
3089
3090 for dir in testdirs:
3091 # loop will be stopped if any directory fill fail the check
3092 try:
3093 if not _check_dir(dir):
3094 return False
3095 except Exception as e:
3096 # here expected only timeout error, all other will
3097 # be catch by 'ignore_status=True'
3098 logging.debug('Fail to check %s to write in it', dir)
3099 return False
3100 return True
Garry Wang1a493d82020-08-31 21:01:19 -07003101
Dana Goyettec172b172020-07-29 16:26:15 -07003102 def blocking_sync(self, freeze_for_reset=False):
3103 """Sync root device and internal device, via script.
3104
3105 The actual calls end up logged by the run() call, since they're printed
3106 to stdout/stderr in the script.
3107
3108 @param freeze_for_reset: if True, prepare for reset by blocking writes
3109 (only if enable_fs_sync_fsfreeze=True)
3110 """
3111
3112 if freeze_for_reset and self.USE_FSFREEZE:
3113 logging.info('Blocking sync and freeze')
3114 elif freeze_for_reset:
3115 logging.info('Blocking sync for reset')
3116 else:
3117 logging.info('Blocking sync')
3118
3119 # client/bin is installed on the DUT as /usr/local/autotest/bin
3120 sync_cmd = '/usr/local/autotest/bin/fs_sync.py'
3121 if freeze_for_reset and self.USE_FSFREEZE:
3122 sync_cmd += ' --freeze'
3123 return self.run(sync_cmd)
3124
Garry Wanga2e78172020-09-09 23:49:07 -07003125 def set_health_profile_dut_state(self, state):
3126 if not self.health_profile:
3127 logging.debug('Device health profile is not initialized, skip'
3128 ' set dut state.')
3129 return
3130 reset_counters = state in profile_constants.STATES_NEED_RESET_COUNTER
3131 self.health_profile.update_dut_state(state, reset_counters)
Garry Wang53fc8f32020-09-18 13:30:08 -07003132
3133 def require_snk_mode_in_recovery(self):
3134 """Check whether we need to switch servo_v4 role to snk when
3135 booting into recovery mode. (See crbug.com/1129165)
3136 """
Garry Wanga8739cc2020-10-30 00:49:23 -07003137 has_battery = True
3138 # Determine if the host has battery based on host_info first.
3139 power_info = self.host_info_store.get().get_label_value('power')
3140 if power_info:
3141 has_battery = power_info == 'battery'
3142 elif self.is_up_fast():
3143 # when running local tests host_info is not available, so we
3144 # need to determine whether the host has battery by checking
3145 # from host side.
3146 logging.debug('Label `power` is not found in host_info, checking'
3147 ' if the host has battery from host side.')
3148 has_battery = self.has_battery()
3149
3150 if not has_battery:
Garry Wang53fc8f32020-09-18 13:30:08 -07003151 logging.info(
3152 '%s does not has battery, snk mode is not needed'
3153 ' for recovery.', self.hostname)
3154 return False
Garry Wanga8739cc2020-10-30 00:49:23 -07003155
Garry Wang53fc8f32020-09-18 13:30:08 -07003156 if not self.servo.supports_built_in_pd_control():
3157 logging.info('Power delivery is not supported on this servo, snk'
3158 ' mode is not needed for recovery.')
3159 return False
3160 try:
Garry Wang53fc8f32020-09-18 13:30:08 -07003161 battery_percent = self.servo.get('battery_charge_percent')
Otabek Kasimov58e22562020-11-03 17:17:41 -08003162 if battery_percent < cros_constants.MIN_BATTERY_LEVEL:
Garry Wang53fc8f32020-09-18 13:30:08 -07003163 logging.info(
3164 'Current battery level %s%% below %s%% threshold, we'
3165 ' will attempt to boot host in recovery mode without'
3166 ' changing servo to snk mode. Please note the host may'
3167 ' not able to see usb drive in recovery mode later due'
3168 ' to servo not in snk mode.', battery_percent,
Otabek Kasimov58e22562020-11-03 17:17:41 -08003169 cros_constants.MIN_BATTERY_LEVEL)
Garry Wang53fc8f32020-09-18 13:30:08 -07003170 return False
3171 except Exception as e:
3172 logging.info(
3173 'Unexpected error occurred when getting'
3174 ' battery_charge_percent from servo; %s', str(e))
3175 return False
3176 return True
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003177
3178 def _set_servo_topology(self):
3179 """Set servo-topology info to the host-info."""
3180 logging.debug('Try to save servo topology to host-info.')
3181 if not self._servo_host:
Greg Edelstonff2665d2021-04-21 14:32:27 -06003182 logging.debug('Servo host is not initialized.')
Otabek Kasimovfe41e2d2021-02-14 20:48:52 -08003183 return
3184 if not self.is_servo_in_working_state():
3185 logging.debug('Is servo is not in working state then'
3186 ' update topology is not allowed.')
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003187 return
3188 if not self._servo_host.is_servo_topology_supported():
Otabek Kasimovfe41e2d2021-02-14 20:48:52 -08003189 logging.debug('Servo-topology is not supported.')
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003190 return
3191 servo_topology = self._servo_host.get_topology()
3192 if not servo_topology or servo_topology.is_empty():
Otabek Kasimovfe41e2d2021-02-14 20:48:52 -08003193 logging.debug('Servo topology is empty')
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003194 return
3195 servo_topology.save(self.host_info_store)