blob: d416ee688dc5b3fa4c79c9e8110e0d360c48ecbc [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
Simran Basi5ace6f22016-01-06 17:30:44 -080030from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070031from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070032from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050033from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070034from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Garry Wang1a493d82020-08-31 21:01:19 -070035from autotest_lib.server.cros.device_health_profile import device_health_profile
Garry Wanga2e78172020-09-09 23:49:07 -070036from autotest_lib.server.cros.device_health_profile import profile_constants
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070037from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070038from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070039from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080040from autotest_lib.server.hosts import chameleon_host
Otabek Kasimov832d9162020-07-27 19:24:57 -070041from autotest_lib.server.hosts import cros_constants
Richard Barnetted31580e2018-05-14 19:58:00 +000042from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080043from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070044from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070045from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070046from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080047from autotest_lib.site_utils.rpm_control_system import rpm_client
Otabek Kasimov808cd832020-05-28 18:27:46 -070048from autotest_lib.site_utils.admin_audit import constants as audit_const
Otabek Kasimov27bb2862020-08-10 14:40:45 -070049from autotest_lib.site_utils.admin_audit import verifiers as audit_verify
Derek Beckettf73baca2020-08-19 15:08:47 -070050from six.moves import zip
Simran Basid5e5e272012-09-24 15:23:59 -070051
Simran Basi382506b2016-09-13 14:58:15 -070052# In case cros_host is being ran via SSP on an older Moblab version with an
53# older chromite version.
54try:
Mike Frysinger714c5b02020-09-04 23:22:54 -040055 from autotest_lib.utils.frozen_chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080056except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080057 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080058
Simran Basid5e5e272012-09-24 15:23:59 -070059
Dan Shib8540a52015-07-16 14:18:23 -070060CONFIG = global_config.global_config
61
beepsc87ff602013-07-31 21:53:00 -070062class FactoryImageCheckerException(error.AutoservError):
63 """Exception raised when an image is a factory image."""
64 pass
65
66
Fang Deng0ca40e22013-08-27 17:47:44 -070067class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070068 """Chromium OS specific subclass of Host."""
69
Simran Basi5ace6f22016-01-06 17:30:44 -080070 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
71
Scott Zawalski62bacae2013-03-05 10:40:32 -050072 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070073
Richard Barnette03a0c132012-11-05 12:40:35 -080074 # Timeout values (in seconds) associated with various Chrome OS
75 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070076 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080077 # In general, a good rule of thumb is that the timeout can be up
78 # to twice the typical measured value on the slowest platform.
79 # The times here have not necessarily been empirically tested to
80 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070081 #
82 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080083 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
84 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080085 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070086 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080087 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070088 # screen delay, time to start the network on the DUT, and the
89 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070090 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080091 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080092 # network.
beepsf079cfb2013-09-18 17:49:51 -070093 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
Otabek Kasimovaeb47fe2021-01-26 20:53:55 -080094 # ADMIN_INSTALL_TIMEOUT: Time to allow for chromeos-install
95 # used by admin tasks.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080096 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
97 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070098
99 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -0800100 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +0800101 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -0700102 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -0700103 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -0700104 INSTALL_TIMEOUT = 480
Otabek Kasimovaeb47fe2021-01-26 20:53:55 -0800105 ADMIN_INSTALL_TIMEOUT = 600
Dan Shi2c88eed2013-11-12 10:18:38 -0800106 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -0700107
Dan Shica503482015-03-30 17:23:25 -0700108 # Minimum OS version that supports server side packaging. Older builds may
109 # not have server side package built or with Autotest code change to support
110 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -0700111 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -0700112 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -0700113
Dana Goyettec172b172020-07-29 16:26:15 -0700114 USE_FSFREEZE = CONFIG.get_config_value(
Dana Goyette6242cb32020-09-23 11:02:57 -0700115 'CROS', 'enable_fs_freeze', type=bool, default=False)
Dana Goyettec172b172020-07-29 16:26:15 -0700116
J. Richard Barnette84890bd2014-02-21 11:05:47 -0800117 # REBOOT_TIMEOUT: How long to wait for a reboot.
118 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700119 # We have a long timeout to ensure we don't flakily fail due to other
120 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700121 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
122 # return from reboot' bug is solved.
123 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700124
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800125 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
126 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700127 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
128 # since changing servo role will reset USB state
129 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800130 _USB_POWER_TIMEOUT = 5
131 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700132 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800133
Fang Dengdeba14f2014-11-14 11:54:09 -0800134 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
135 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700136
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800137 # Constants used in ping_wait_up() and ping_wait_down().
138 #
139 # _PING_WAIT_COUNT is the approximate number of polling
140 # cycles to use when waiting for a host state change.
141 #
142 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
143 # for arguments to the internal _ping_wait_for_status()
144 # method.
145 _PING_WAIT_COUNT = 40
146 _PING_STATUS_DOWN = False
147 _PING_STATUS_UP = True
148
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800149 # Allowed values for the power_method argument.
150
Garry Wang5e5538a2019-04-08 15:36:18 -0700151 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
152 # DUTs except those with servo_v4 CCD.
153 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
154 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800155 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
156 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
157 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700158 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800159 POWER_CONTROL_SERVO = 'servoj10'
160 POWER_CONTROL_MANUAL = 'manual'
161
162 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700163 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800164 POWER_CONTROL_SERVO,
165 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800166
Simran Basi5e6339a2013-03-21 11:34:32 -0700167 _RPM_OUTLET_CHANGED = 'outlet_changed'
168
Dan Shi9cb0eec2014-06-03 09:04:50 -0700169 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700170 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700171 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700172
Brent Peterson1cb623a2020-01-09 13:14:28 -0800173 # Regular expression for extracting EC version string
174 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
175
176 # Regular expression for extracting BIOS version string
177 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
178
Brent Petersonc70a1832020-01-24 15:54:35 -0800179 # Command to update firmware located on DUT
Namyoon Woo382e5892020-05-20 16:48:40 -0700180 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800181
J. Richard Barnette964fba02012-10-24 17:34:29 -0700182 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800183 def check_host(host, timeout=10):
184 """
185 Check if the given host is a chrome-os host.
186
187 @param host: An ssh host representing a device.
188 @param timeout: The timeout for the run command.
189
190 @return: True if the host device is chromeos.
191
beeps46dadc92013-11-07 14:07:10 -0800192 """
193 try:
Allen Liad719c12017-06-27 23:48:04 +0000194 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700195 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700196 '! grep -q moblab /etc/lsb-release && '
Derek Beckett342e3e62021-01-05 17:17:23 -0800197 '! grep -q labstation /etc/lsb-release &&'
198 ' grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
199 ignore_status=True,
Laurence Goodby468de252017-06-08 17:22:53 -0700200 timeout=timeout).stdout
Derek Beckett342e3e62021-01-05 17:17:23 -0800201 if result:
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800202 return not (
203 lsbrelease_utils.is_jetstream(
Derek Beckett342e3e62021-01-05 17:17:23 -0800204 lsb_release_content=result) or
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800205 lsbrelease_utils.is_gce_board(
Derek Beckett342e3e62021-01-05 17:17:23 -0800206 lsb_release_content=result))
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800207
beeps46dadc92013-11-07 14:07:10 -0800208 except (error.AutoservRunError, error.AutoservSSHTimeout):
209 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700210
211 return False
beeps46dadc92013-11-07 14:07:10 -0800212
213
214 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800215 def get_chameleon_arguments(args_dict):
216 """Extract chameleon options from `args_dict` and return the result.
217
218 Recommended usage:
219 ~~~~~~~~
220 args_dict = utils.args_to_dict(args)
221 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
222 host = hosts.create_host(machine, chameleon_args=chameleon_args)
223 ~~~~~~~~
224
225 @param args_dict Dictionary from which to extract the chameleon
226 arguments.
227 """
Sam McNally66594ca2019-12-09 12:45:44 +1100228 chameleon_args = {key: args_dict[key]
229 for key in ('chameleon_host', 'chameleon_port')
230 if key in args_dict}
231 if 'chameleon_ssh_port' in args_dict:
232 chameleon_args['port'] = int(args_dict['chameleon_ssh_port'])
233 return chameleon_args
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800234
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800235 @staticmethod
236 def get_btpeer_arguments(args_dict):
237 """Extract btpeer options from `args_dict` and return the result.
238
239 This is used to parse details of Bluetooth peer.
240 Recommended usage:
241 ~~~~~~~~
242 args_dict = utils.args_to_dict(args)
243 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700244 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800245 ~~~~~~~~
246
247 @param args_dict: Dictionary from which to extract the btpeer
248 arguments.
249 """
250 if 'btpeer_host_list' in args_dict:
251 result = []
252 for btpeer in args_dict['btpeer_host_list'].split(','):
Claire Changd0b19842020-11-04 22:28:45 +0800253 # IPv6 addresses including a port number should be enclosed in
254 # square brackets.
255 delimiter = ']:' if re.search(r':.*:', btpeer) else ':'
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800256 result.append({key: value for key,value in
257 zip(('btpeer_host','btpeer_port'),
Claire Changd0b19842020-11-04 22:28:45 +0800258 btpeer.strip('[]').split(delimiter))})
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800259 return result
260 else:
Anand K Mistrye8933092020-08-05 14:49:41 +1000261 return {key: args_dict[key]
262 for key in ('btpeer_host', 'btpeer_port', 'btpeer_ssh_port')
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800263 if key in args_dict}
264
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800265
266 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700267 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800268 """Extract chameleon options from `args_dict` and return the result.
269
270 Recommended usage:
271 ~~~~~~~~
272 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700273 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
274 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800275 ~~~~~~~~
276
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700277 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800278 arguments.
279 """
Allen Li083866b2016-08-18 10:07:10 -0700280 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700281 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700282 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800283
284
285 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800286 def get_servo_arguments(args_dict):
287 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800288
289 Recommended usage:
290 ~~~~~~~~
291 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700292 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800293 host = hosts.create_host(machine, servo_args=servo_args)
294 ~~~~~~~~
295
296 @param args_dict Dictionary from which to extract the servo
297 arguments.
298 """
Garry Wang11b5e872020-03-11 15:14:08 -0700299 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
300 servo_constants.SERVO_PORT_ATTR,
Otabek Kasimov382c3bb2020-10-28 13:22:45 -0700301 servo_constants.SERVO_SERIAL_ATTR,
Garry Wang11b5e872020-03-11 15:14:08 -0700302 servo_constants.SERVO_BOARD_ATTR,
303 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200304 servo_args = {key: args_dict[key]
305 for key in servo_attrs
306 if key in args_dict}
307 return (
308 None
Garry Wang11b5e872020-03-11 15:14:08 -0700309 if servo_constants.SERVO_HOST_ATTR in servo_args
310 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200311 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700312
J. Richard Barnette964fba02012-10-24 17:34:29 -0700313
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800314 def _initialize(self,
315 hostname,
316 chameleon_args=None,
317 servo_args=None,
318 pdtester_args=None,
319 try_lab_servo=False,
320 try_servo_repair=False,
321 ssh_verbosity_flag='',
322 ssh_options='',
323 try_servo_recovery=False,
324 *args,
325 **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800326 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700327
Fang Denge545abb2014-12-30 18:43:47 -0800328 This method will attempt to create the test-assistant object
329 (chameleon/servo) when it is needed by the test. Check
330 the docstring of chameleon_host.create_chameleon_host and
331 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700332
Fang Denge545abb2014-12-30 18:43:47 -0800333 @param hostname: Hostname of the dut.
334 @param chameleon_args: A dictionary that contains args for creating
335 a ChameleonHost. See chameleon_host for details.
336 @param servo_args: A dictionary that contains args for creating
337 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700338 @param try_lab_servo: When true, indicates that an attempt should
339 be made to create a ServoHost for a DUT in
340 the test lab, even if not required by
341 `servo_args`. See servo_host for details.
342 @param try_servo_repair: If a servo host is created, check it
343 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800344 See servo_host for details.
345 @param ssh_verbosity_flag: String, to pass to the ssh command to control
346 verbosity.
347 @param ssh_options: String, other ssh options to pass to the ssh
348 command.
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800349 @param try_servo_recovery: When True, start servod in recovery mode.
350 See servo_host for details.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700351 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700352 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700353 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800354 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Otabek Kasimov6825b762020-06-23 23:42:44 -0700355 # hold special dut_state for repair process
356 self._device_repair_state = None
Kevin Chenga2619dc2016-03-28 11:42:08 -0700357 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700358 # self.env is a dictionary of environment variable settings
359 # to be exported for commands run on the host.
360 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
361 # errors that might happen.
362 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700363 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700364 self._ssh_options = ssh_options
Garry Wang1a493d82020-08-31 21:01:19 -0700365 self.health_profile = None
Garry Wang5e5538a2019-04-08 15:36:18 -0700366 self._default_power_method = None
Otabek Kasimov39637412020-11-23 19:09:27 -0800367 dut_health_profile = device_health_profile.DeviceHealthProfile(
368 hostname=self.hostname,
369 host_info=self.host_info_store.get(),
370 result_dir=self.get_result_dir())
Otabek Kasimovb61729d2020-11-24 00:42:43 -0800371
372 # TODO(otabek@): remove when b/171414073 closed
Otabek Kasimov7587b902020-12-07 23:54:33 -0800373 pingable_before_servo = self.is_up_fast(count=3)
Otabek Kasimovb61729d2020-11-24 00:42:43 -0800374 if pingable_before_servo:
375 logging.info('DUT is pingable before init Servo.')
Otabek Kasimov39637412020-11-23 19:09:27 -0800376 _servo_host, servo_state = servo_host.create_servo_host(
377 dut=self,
378 servo_args=servo_args,
379 try_lab_servo=try_lab_servo,
380 try_servo_repair=try_servo_repair,
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -0800381 try_servo_recovery=try_servo_recovery,
Otabek Kasimov39637412020-11-23 19:09:27 -0800382 dut_host_info=self.host_info_store.get(),
383 dut_health_profile=dut_health_profile)
384 if dut_health_profile.is_loaded():
385 logging.info('Device health profile loaded.')
386 # The device profile is located in the servo_host which make it
387 # dependency. If profile is not loaded yet then we do not have it
388 # TODO(otabek@) persist device provide out of servo-host.
389 self.health_profile = dut_health_profile
390 self.set_servo_host(_servo_host, servo_state)
Richard Barnettee519dcd2016-08-15 17:37:17 -0700391
Otabek Kasimovb61729d2020-11-24 00:42:43 -0800392 # TODO(otabek@): remove when b/171414073 closed
393 # Introduced to collect cases when servo made DUT not sshable
Otabek Kasimov7587b902020-12-07 23:54:33 -0800394 pingable_after_servo = self.is_up_fast(count=3)
Otabek Kasimovb61729d2020-11-24 00:42:43 -0800395 if pingable_after_servo:
396 logging.info('DUT is pingable after init Servo.')
397 elif pingable_before_servo:
398 logging.info('DUT was pingable before init Servo but not now')
Otabek Kasimov7587b902020-12-07 23:54:33 -0800399 if servo_args and self._servo_host and self._servo_host.hostname:
400 # collect stats only for tests.
401 dut_ping_servo_init_data = {
402 'host': self.hostname,
403 'servo_host': self._servo_host.hostname,
404 }
405 metrics.Counter('chromeos/autotest/dut_ping_servo_init2'
406 ).increment(fields=dut_ping_servo_init_data)
Otabek Kasimovb61729d2020-11-24 00:42:43 -0800407
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800408 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800409 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700410 dut=self.hostname,
411 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800412 if self._chameleon_host:
413 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800414 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800415 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700416
Shijin Abraham78ce4402020-09-08 22:04:27 -0700417 # Bluetooth peers will be populated by the test if needed
418 self._btpeer_host_list = []
419 self.btpeer_list = []
420 self.btpeer = None
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800421
howardchung83e55272019-08-08 14:08:05 +0800422 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700423 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700424 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800425
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700426 if self._pdtester_host:
427 self.pdtester_servo = self._pdtester_host.get_servo()
428 logging.info('pdtester_servo: %r', self.pdtester_servo)
429 # Create the pdtester object used to access the ec uart
430 self.pdtester = pdtester.PDTester(self.pdtester_servo,
431 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800432 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700433 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800434
Fang Deng5d518f42013-08-02 14:04:32 -0700435
Shijin Abraham78ce4402020-09-08 22:04:27 -0700436 def initialize_btpeer(self, btpeer_args=[]):
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800437 """ Initialize the Bluetooth peers
438
439 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
440 is chameleon host on Raspberry Pi.
441 @param btpeer_args: A dictionary that contains args for creating
442 a ChameleonHost. See chameleon_host for details.
443
444 """
Shijin Abraham78ce4402020-09-08 22:04:27 -0700445 logging.debug('Attempting to initialize bluetooth peers if available')
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700446 try:
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700447 if type(btpeer_args) is list:
448 btpeer_args_list = btpeer_args
449 else:
450 btpeer_args_list = [btpeer_args]
451
452 self._btpeer_host_list = chameleon_host.create_btpeer_host(
453 dut=self.hostname, btpeer_args_list=btpeer_args_list)
454 logging.debug('Bluetooth peer hosts are %s',
455 self._btpeer_host_list)
456 self.btpeer_list = [_host.create_chameleon_board() for _host in
457 self._btpeer_host_list if _host is not None]
458
459 if len(self.btpeer_list) > 0:
460 self.btpeer = self.btpeer_list[0]
461
462 logging.debug('After initialize_btpeer btpeer_list %s '
463 'btpeer_host_list is %s and btpeer is %s',
464 self.btpeer_list, self._btpeer_host_list,
465 self.btpeer)
466 except Exception as e:
467 logging.error('Exception %s in initialize_btpeer', str(e))
468
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800469
470
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000471 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700472 """Get latest stable cros image name from AFE.
473
474 Use the board name from the info store. Should that fail, try to
475 retrieve the board name from the host's installed image itself.
476
477 @returns: current stable cros image name for this host.
478 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700479 info = self.host_info_store.get()
480 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700481 logging.warn('No board label value found. Trying to infer '
482 'from the host itself.')
483 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700484 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700485 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
486 logging.error('Also failed to get the board name from the DUT '
487 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700488 raise error.AutoservError('Cannot determine board of the DUT'
489 ' while getting repair image name.')
490 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500491
492
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700493 def host_version_prefix(self, image):
494 """Return version label prefix.
495
496 In case the CrOS provisioning version is something other than the
497 standard CrOS version e.g. CrOS TH version, this function will
498 find the prefix from provision.py.
499
500 @param image: The image name to find its version prefix.
501 @returns: A prefix string for the image type.
502 """
503 return provision.get_version_label_prefix(image)
504
Andrew Luo3332ab22020-04-28 16:42:03 -0700505 def stage_build_to_usb(self, build):
506 """Stage the current ChromeOS image on the USB stick connected to the
507 servo.
508
509 @param build: The build to download and send to USB.
510 """
511 if not self.servo:
512 raise error.TestError('Host %s does not have servo.' %
513 self.hostname)
514
515 _, update_url = self.stage_image_for_servo(build)
Andrew Luob0355ea2020-06-24 16:12:57 -0700516
517 try:
518 self.servo.image_to_servo_usb(update_url)
519 finally:
520 # servo.image_to_servo_usb turned the DUT off, so turn it back on
521 logging.debug('Turn DUT power back on.')
522 self.servo.get_power_state_controller().power_on()
523
Andrew Luo3332ab22020-04-28 16:42:03 -0700524 logging.debug('ChromeOS image %s is staged on the USB stick.',
525 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700526
beepsdae65fd2013-07-26 16:24:41 -0700527 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700528 """
529 Make sure job_repo_url of this host is valid.
530
joychen03eaad92013-06-26 09:55:21 -0700531 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700532 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
533 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
534 download and extract it. If the devserver embedded in the url is
535 unresponsive, update the job_repo_url of the host after staging it on
536 another devserver.
537
538 @param job_repo_url: A url pointing to the devserver where the autotest
539 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700540 @param tag: The tag from the server job, in the format
541 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700542
543 @raises DevServerException: If we could not resolve a devserver.
544 @raises AutoservError: If we're unable to save the new job_repo_url as
545 a result of choosing a new devserver because the old one failed to
546 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700547 @raises urllib2.URLError: If the devserver embedded in job_repo_url
548 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700549 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800550 info = self.host_info_store.get()
551 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700552 if not job_repo_url:
553 logging.warning('No job repo url set on host %s', self.hostname)
554 return
555
556 logging.info('Verifying job repo url %s', job_repo_url)
557 devserver_url, image_name = tools.get_devserver_build_from_package_url(
558 job_repo_url)
559
beeps0c865032013-07-30 11:37:06 -0700560 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700561
562 logging.info('Staging autotest artifacts for %s on devserver %s',
563 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700564
565 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700566 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700567 stage_time = time.time() - start_time
568
569 # Record how much of the verification time comes from a devserver
570 # restage. If we're doing things right we should not see multiple
571 # devservers for a given board/build/branch path.
572 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800573 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700574 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800575 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700576 pass
577 else:
beeps0c865032013-07-30 11:37:06 -0700578 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700579 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700580 stats_key = {
581 'board': board,
582 'build_type': build_type,
583 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700584 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700585 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800586
587 monarch_fields = {
588 'board': board,
589 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800590 'branch': branch,
591 'dev_server': devserver,
592 }
593 metrics.Counter(
594 'chromeos/autotest/provision/verify_url'
595 ).increment(fields=monarch_fields)
596 metrics.SecondsDistribution(
597 'chromeos/autotest/provision/verify_url_duration'
598 ).add(stage_time, fields=monarch_fields)
599
600
Dan Shicf4d2032015-03-12 15:04:21 -0700601 def stage_server_side_package(self, image=None):
602 """Stage autotest server-side package on devserver.
603
604 @param image: Full path of an OS image to install or a build name.
605
606 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700607
608 @raise: error.AutoservError if fail to locate the build to test with, or
609 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700610 """
Dan Shid37736b2016-07-06 15:10:29 -0700611 # If enable_drone_in_restricted_subnet is False, do not set hostname
612 # in devserver.resolve call, so a devserver in non-restricted subnet
613 # is picked to stage autotest server package for drone to download.
614 hostname = self.hostname
615 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
616 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700617 if image:
618 image_name = tools.get_build_from_image(image)
619 if not image_name:
620 raise error.AutoservError(
621 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700622 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700623 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700624 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800625 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700626 if job_repo_url:
627 devserver_url, image_name = (
628 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700629 # If enable_drone_in_restricted_subnet is True, use the
630 # existing devserver. Otherwise, resolve a new one in
631 # non-restricted subnet.
632 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
633 ds = dev_server.ImageServer(devserver_url)
634 else:
635 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800636 elif info.build is not None:
637 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700638 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700639 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800640 raise error.AutoservError(
641 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700642 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700643
644 # Get the OS version of the build, for any build older than
645 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
646 match = re.match('.*/R\d+-(\d+)\.', image_name)
647 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700648 raise error.AutoservError(
649 'Build %s is older than %s. Server side packaging is '
650 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700651
Dan Shicf4d2032015-03-12 15:04:21 -0700652 ds.stage_artifacts(image_name, ['autotest_server_package'])
653 return '%s/static/%s/%s' % (ds.url(), image_name,
654 'autotest_server_package.tar.bz2')
655
656
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700657 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700658 """Stage a build on a devserver and return the update_url.
659
660 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700661 @param artifact: a string like 'test_image'. Requests
662 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700663 @returns a tuple of (image_name, URL) like
664 (lumpy-release/R27-3837.0.0,
665 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700666 """
667 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000668 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700669 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800670 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700671 devserver.stage_artifacts(image_name, [artifact])
672 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700673 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700674 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700675 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700676 else:
677 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700678
679
beepse539be02013-07-31 21:57:39 -0700680 def stage_factory_image_for_servo(self, image_name):
681 """Stage a build on a devserver and return the update_url.
682
683 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700684
beepse539be02013-07-31 21:57:39 -0700685 @return: An update URL, eg:
686 http://<devserver>/static/canary-channel/\
687 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700688
689 @raises: ValueError if the factory artifact name is missing from
690 the config.
691
beepse539be02013-07-31 21:57:39 -0700692 """
693 if not image_name:
694 logging.error('Need an image_name to stage a factory image.')
695 return
696
Dan Shib8540a52015-07-16 14:18:23 -0700697 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700698 'CROS', 'factory_artifact', type=str, default='')
699 if not factory_artifact:
700 raise ValueError('Cannot retrieve the factory artifact name from '
701 'autotest config, and hence cannot stage factory '
702 'artifacts.')
703
beepse539be02013-07-31 21:57:39 -0700704 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800705 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700706 devserver.stage_artifacts(
707 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700708 [factory_artifact],
709 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700710
711 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
712
713
Laurence Goodby778c9a42017-05-24 19:24:07 -0700714 def prepare_for_update(self):
715 """Prepares the DUT for an update.
716
717 Subclasses may override this to perform any special actions
718 required before updating.
719 """
Laurence Goodby468de252017-06-08 17:22:53 -0700720 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700721
722
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800723 def _clear_fw_version_labels(self, rw_only):
724 """Clear firmware version labels from the machine.
725
726 @param rw_only: True to only clear fwrw_version; otherewise, clear
727 both fwro_version and fwrw_version.
728 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700729 info = self.host_info_store.get()
730 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800731 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700732 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
733 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700734
735
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800736 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700737 """Add firmware version label to the machine.
738
739 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800740 @param rw_only: True to only add fwrw_version; otherwise, add both
741 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700742
743 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700744 info = self.host_info_store.get()
745 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800746 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700747 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
748 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700749
750
Namyoon Woo33f38852020-04-13 17:26:58 -0700751 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800752 """Search for the latest package release version from the image archive,
753 and return it.
754
Namyoon Woo33f38852020-04-13 17:26:58 -0700755 @param platform: platform name, a.k.a. board or model
756 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800757
Namyoon Woo33f38852020-04-13 17:26:58 -0700758 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
759 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800760 or None if LATEST release file does not exist.
761 """
762
Namyoon Woo33f38852020-04-13 17:26:58 -0700763 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800764
Namyoon Woo33f38852020-04-13 17:26:58 -0700765 # Search the image path in reference board archive as well.
766 # For example, bob has its binary image under its reference board (gru)
767 # image archive.
768 if ref_board:
769 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800770
Namyoon Woo33f38852020-04-13 17:26:58 -0700771 for board in platforms:
772 # Read 'LATEST-1.0.0' file
773 branch_dir = provision.FW_BRANCH_GLOB % board
774 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
775 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800776
Namyoon Woo33f38852020-04-13 17:26:58 -0700777 try:
778 # The result could be one or more.
779 result = utils.system_output('gsutil ls -d ' + latest_file)
780
781 candidates = re.findall('gs://.*', result)
782
783 # Found the directory candidates. No need to check the other
784 # board name cadidates. Let's break the loop.
785 break
786 except error.CmdError:
787 # It doesn't exist. Let's move on to the next item.
788 pass
789 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800790 logging.error('No LATEST release info is available.')
791 return None
792
Namyoon Woo406c7d42020-01-24 15:57:11 -0800793 for cand_dir in candidates:
794 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800795
Namyoon Woo406c7d42020-01-24 15:57:11 -0800796 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700797 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800798 try:
799 # Check if release_path does exist.
800 release = utils.system_output('gsutil ls -d ' + release_path)
801 # Now 'release' has a full directory path: e.g.
802 # gs://chromeos-image-archive/firmware-octopus-11297.B-
803 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
804
805 # Remove "gs://chromeos-image-archive".
806 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
807
808 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
809 return release.strip('/')
810 except error.CmdError:
811 # The directory might not exist. Let's try next candidate.
812 pass
813 else:
814 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800815
Brent Peterson1cb623a2020-01-09 13:14:28 -0800816 @staticmethod
817 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800818 """Get version string from binary image using regular expression.
819
820 @param image: Binary image to search
821 @param version_regex: Regular expression to search for
822
823 @return Version string
824
825 @raises TestFail if no version string is found in image
826 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800827 with open(image, 'rb') as f:
828 image_data = f.read()
Derek Beckett98345552020-08-31 16:07:22 -0700829 match = re.findall(version_regex,
830 image_data.decode('ISO-8859-1', errors='ignore'))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800831 if match:
832 return match[0]
833 else:
834 raise error.TestFail('Failed to read version from %s.' % image)
835
836
Garry Wangad2a1712020-03-26 15:06:43 -0700837 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800838 local_tarball=None, verify_version=False,
Namyoon Woo382e5892020-05-20 16:48:40 -0700839 try_scp=False, install_ec=True, install_bios=True,
840 board_as=None):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700841 """Install firmware to the DUT.
842
843 Use stateful update if the DUT is already running the same build.
844 Stateful update does not update kernel and tends to run much faster
845 than a full reimage. If the DUT is running a different build, or it
846 failed to do a stateful update, full update, including kernel update,
847 will be applied to the DUT.
848
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800849 Once a host enters firmware_install its fw[ro|rw]_version label will
850 be removed. After the firmware is updated successfully, a new
851 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700852
853 @param build: The build version to which we want to provision the
854 firmware of the machine,
855 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800856 @param rw_only: True to only install firmware to its RW portions. Keep
857 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700858 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800859 @param local_tarball: Path to local firmware image for installing
860 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800861 @param verify_version: True to verify EC and BIOS versions after
862 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800863 @param try_scp: False to always program using servo, true to try copying
864 the firmware and programming from the DUT.
Namyoon Woo382e5892020-05-20 16:48:40 -0700865 @param install_ec: True to install EC FW, and False to skip it.
866 @param install_bios: True to install BIOS, and False to skip it.
867 @param board_as: A board name to force to use.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700868
869 TODO(dshi): After bug 381718 is fixed, update here with corresponding
870 exceptions that could be raised.
871
872 """
873 if not self.servo:
874 raise error.TestError('Host %s does not have servo.' %
875 self.hostname)
876
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700877 # Get the DUT board name from AFE.
878 info = self.host_info_store.get()
879 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700880 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800881
882 if board is None or board == '':
883 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700884
Namyoon Woo382e5892020-05-20 16:48:40 -0700885 # if board_as argument is passed, then use it instead of the original
886 # board name.
887 if board_as:
888 board = board_as
889
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700890 if model is None or model == '':
Namyoon Woofb16eae2020-08-14 10:02:39 -0700891 try:
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700892 model = self.get_platform()
Namyoon Woofb16eae2020-08-14 10:02:39 -0700893 except Exception as e:
894 logging.warn('Dut is unresponsive: %s', str(e))
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700895
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800896 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700897 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800898 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700899 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800900
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700901 try:
902 ds = dev_server.ImageServer.resolve(build, self.hostname)
903 ds.stage_artifacts(build, ['firmware'])
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800904
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700905 if not dest:
906 tmpd = autotemp.tempdir(unique_id='fwimage')
907 dest = tmpd.name
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800908
Namyoon Woo43c1cb22020-05-28 18:58:34 -0700909 # Download firmware image
910 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
911 local_tarball = os.path.join(dest, os.path.basename(fwurl))
912 ds.download_file(fwurl, local_tarball)
913 except Exception as e:
914 raise error.TestError('Failed to download firmware package: %s'
915 % str(e))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700916
Namyoon Woo382e5892020-05-20 16:48:40 -0700917 ec_image = None
918 if install_ec:
919 # Extract EC image from tarball
920 logging.info('Extracting EC image.')
921 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700922 logging.info('Extracted: %s', ec_image)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800923
Namyoon Woo382e5892020-05-20 16:48:40 -0700924 bios_image = None
925 if install_bios:
926 # Extract BIOS image from tarball
927 logging.info('Extracting BIOS image.')
928 bios_image = self.servo.extract_bios_image(board, model,
929 local_tarball)
Mary Ruthven9c15a4e2020-10-23 10:10:53 -0700930 logging.info('Extracted: %s', bios_image)
Namyoon Woo382e5892020-05-20 16:48:40 -0700931
932 if not bios_image and not ec_image:
933 raise error.TestError('No firmware installation was processed.')
Brent Peterson1cb623a2020-01-09 13:14:28 -0800934
Brent Petersonc70a1832020-01-24 15:54:35 -0800935 # Clear firmware version labels
936 self._clear_fw_version_labels(rw_only)
937
938 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800939 try:
Garry Wang50e4a492020-08-05 12:29:57 -0700940 # Check if copying to DUT is enabled and DUT is available
941 if try_scp and self.is_up():
Brent Petersonc70a1832020-01-24 15:54:35 -0800942 # DUT is available, make temp firmware directory to store images
943 logging.info('Making temp folder.')
944 dest_folder = '/tmp/firmware'
945 self.run('mkdir -p ' + dest_folder)
946
Namyoon Woo68b68082020-06-02 13:13:14 -0700947 fw_cmd = self._FW_UPDATE_CMD % ('--wp=1' if rw_only else '')
Brent Petersonc70a1832020-01-24 15:54:35 -0800948
Namyoon Woo382e5892020-05-20 16:48:40 -0700949 if bios_image:
950 # Send BIOS firmware image to DUT
951 logging.info('Sending BIOS firmware.')
952 dest_bios_path = os.path.join(dest_folder,
953 os.path.basename(bios_image))
954 self.send_file(bios_image, dest_bios_path)
955
956 # Initialize firmware update command for BIOS image
957 fw_cmd += ' -i %s' % dest_bios_path
Brent Peterson669edf42020-02-07 15:07:54 -0800958
959 # Send EC firmware image to DUT when EC image was found
960 if ec_image:
961 logging.info('Sending EC firmware.')
962 dest_ec_path = os.path.join(dest_folder,
963 os.path.basename(ec_image))
964 self.send_file(ec_image, dest_ec_path)
965
966 # Add EC image to firmware update command
967 fw_cmd += ' -e %s' % dest_ec_path
968
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700969 # Make sure command is allowed to finish even if ssh fails.
970 fw_cmd = "trap '' SIGHUP; %s" % fw_cmd
971
Brent Peterson669edf42020-02-07 15:07:54 -0800972 # Update firmware on DUT
973 logging.info('Updating firmware.')
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700974 try:
Dana Goyette935b3fe2020-07-23 14:19:39 -0700975 self.run(fw_cmd, options="-o LogLevel=verbose")
Dana Goyettee84ef8d2020-07-09 11:55:09 -0700976 except error.AutoservRunError as e:
977 if e.result_obj.exit_status != 255:
978 raise
979 elif ec_image:
980 logging.warn("DUT network dropped during update"
981 " (often caused by EC resetting USB)")
982 else:
983 logging.error("DUT network dropped during update"
984 " (unexpected, since no EC image)")
985 raise
Brent Petersonc70a1832020-01-24 15:54:35 -0800986 else:
987 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800988 if ec_image:
989 self.servo.program_ec(ec_image, rw_only)
Namyoon Woo382e5892020-05-20 16:48:40 -0700990 if bios_image:
991 self.servo.program_bios(bios_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800992 if utils.host_is_in_lab_zone(self.hostname):
993 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800994
995 # Reboot and wait for DUT after installing firmware
996 logging.info('Rebooting DUT.')
997 self.servo.get_power_state_controller().reset()
998 time.sleep(self.servo.BOOT_DELAY)
999 self.test_wait_for_boot()
1000
1001 # When enabled verify EC and BIOS firmware version after programming
1002 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -08001003 # Check programmed EC firmware when EC image was found
1004 if ec_image:
1005 logging.info('Checking EC firmware version.')
1006 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -08001007 ec_version_prefix = dest_ec_version.split('_', 1)[0]
1008 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -08001009 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -08001010 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -08001011 if dest_ec_version != image_ec_version:
1012 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -07001013 'Failed to update EC firmware, version %s '
1014 '(expected %s)' % (dest_ec_version,
1015 image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -08001016
Namyoon Woo382e5892020-05-20 16:48:40 -07001017 if bios_image:
1018 # Check programmed BIOS firmware against expected version
1019 logging.info('Checking BIOS firmware version.')
1020 dest_bios_version = self.get_firmware_version()
1021 bios_version_prefix = dest_bios_version.split('.', 1)[0]
1022 bios_regex = self._BIOS_REGEX % bios_version_prefix
1023 image_bios_version = self.get_version_from_image(bios_image,
1024 bios_regex)
1025 if dest_bios_version != image_bios_version:
1026 raise error.TestFail(
Namyoon Wooe2c009a2020-07-22 10:09:03 -07001027 'Failed to update BIOS, version %s '
Namyoon Woo382e5892020-05-20 16:48:40 -07001028 '(expected %s)' % (dest_bios_version,
1029 image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -07001030 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -07001031 if tmpd:
1032 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -07001033
1034
Garry Wang790953f2020-10-29 21:11:57 -07001035 def servo_install(self,
1036 image_url=None,
1037 usb_boot_timeout=USB_BOOT_TIMEOUT,
1038 install_timeout=INSTALL_TIMEOUT,
1039 is_repair=False):
Scott Zawalski62bacae2013-03-05 10:40:32 -05001040 """
1041 Re-install the OS on the DUT by:
1042 1) installing a test image on a USB storage device attached to the Servo
1043 board,
Richard Barnette03a0c132012-11-05 12:40:35 -08001044 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -07001045 3) installing the image with chromeos-install.
1046
Scott Zawalski62bacae2013-03-05 10:40:32 -05001047 @param image_url: If specified use as the url to install on the DUT.
1048 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -07001049 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
1050 Factory images need a longer usb_boot_timeout than regular
1051 cros images.
1052 @param install_timeout: The timeout to use when installing the chromeos
1053 image. Factory images need a longer install_timeout.
Garry Wang790953f2020-10-29 21:11:57 -07001054 @param is_repair: Indicates if the method is called from a repair task.
Richard Barnette03a0c132012-11-05 12:40:35 -08001055
Scott Zawalski62bacae2013-03-05 10:40:32 -05001056 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -07001057
J. Richard Barnette0199cc82014-12-05 17:08:40 -08001058 """
Garry Wang7b0e1b72020-03-25 19:08:59 -07001059 if image_url:
1060 logging.info('Downloading image to USB, then booting from it.'
1061 ' Usb boot timeout = %s', usb_boot_timeout)
1062 else:
1063 logging.info('Booting from USB directly. Usb boot timeout = %s',
1064 usb_boot_timeout)
1065
1066 metrics_field = {'download': bool(image_url)}
1067 metrics.Counter(
1068 'chromeos/autotest/provision/servo_install/download_image'
1069 ).increment(fields=metrics_field)
1070
Allen Li48a13fe2016-11-22 14:10:40 -08001071 with metrics.SecondsTimer(
1072 'chromeos/autotest/provision/servo_install/boot_duration'):
Otabek Kasimovb887a002020-12-29 02:54:38 -08001073 self.servo.get_power_state_controller().power_off()
Otabek Kasimovb7cb8422020-12-23 02:38:32 -08001074 try:
1075 self.servo.image_to_servo_usb(image_path=image_url,
1076 power_off_dut=False)
1077 except error.AutotestError as e:
1078 metrics.Counter('chromeos/autotest/repair/image_to_usb_error'
1079 ).increment(
1080 fields={'host': self.hostname or ''})
1081 six.reraise(error.AutotestError, str(e), sys.exc_info()[2])
1082 # Give the DUT some time to power_off if we skip
1083 # download image to usb. (crbug.com/982993)
1084 if not image_url:
1085 time.sleep(10)
Garry Wang53fc8f32020-09-18 13:30:08 -07001086 need_snk = self.require_snk_mode_in_recovery()
Otabek Kasimovb7cb8422020-12-23 02:38:32 -08001087 self.servo.boot_in_recovery_mode(snk_mode=need_snk)
Allen Li48a13fe2016-11-22 14:10:40 -08001088 if not self.wait_up(timeout=usb_boot_timeout):
Garry Wang53fc8f32020-09-18 13:30:08 -07001089 if need_snk:
1090 # Attempt to restore servo_v4 role to 'src' mode.
1091 self.servo.set_servo_v4_role('src')
Allen Li48a13fe2016-11-22 14:10:40 -08001092 raise hosts.AutoservRepairError(
1093 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -07001094 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001095
Garry Wang2e347df2020-10-30 14:04:26 -07001096 # Make sure the DUT is boot from an external device.
1097 if not self.is_boot_from_external_device():
1098 raise hosts.AutoservRepairError(
1099 'DUT is expected to boot from an external device(e.g. '
1100 'a usb stick), however it seems still boot from an'
1101 ' internal storage.', 'boot_from_internal_storage')
1102
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001103 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
1104 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001105 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +08001106 try:
1107 self.run('chromeos-tpm-recovery')
1108 except error.AutoservRunError:
1109 logging.warn('chromeos-tpm-recovery is too old.')
1110
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +08001111
Allen Li48a13fe2016-11-22 14:10:40 -08001112 with metrics.SecondsTimer(
1113 'chromeos/autotest/provision/servo_install/install_duration'):
1114 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -07001115 try:
1116 self.run('chromeos-install --yes',timeout=install_timeout)
1117 self.halt()
Otabek Kasimov808cd832020-05-28 18:27:46 -07001118 except Exception as e:
1119 storage_errors = [
1120 'No space left on device',
1121 'I/O error when trying to write primary GPT',
1122 'Input/output error while writing out',
1123 'cannot read GPT header',
Otabek Kasimov2b7e8302020-08-21 09:23:31 -07001124 'can not determine destination device',
1125 'wrong fs type',
1126 'bad superblock on',
Otabek Kasimov808cd832020-05-28 18:27:46 -07001127 ]
1128 has_error = [msg for msg in storage_errors if(msg in str(e))]
1129 if has_error:
1130 info = self.host_info_store.get()
1131 info.set_version_label(
1132 audit_const.DUT_STORAGE_STATE_PREFIX,
1133 audit_const.HW_STATE_NEED_REPLACEMENT)
1134 self.host_info_store.commit(info)
Otabek Kasimov6825b762020-06-23 23:42:44 -07001135 self.set_device_repair_state(
Otabek Kasimov832d9162020-07-27 19:24:57 -07001136 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT)
Otabek Kasimov808cd832020-05-28 18:27:46 -07001137 logging.debug(
1138 'Fail install image from USB; Storage error; %s', e)
1139 raise error.AutoservError(
1140 'Failed to install image from USB due to a suspect '
1141 'disk failure, DUT storage state changed to '
1142 'need_replacement, please check debug log '
1143 'for details.')
1144 else:
Garry Wang790953f2020-10-29 21:11:57 -07001145 if is_repair:
1146 # DUT will be marked for replacement if storage is bad.
1147 audit_verify.VerifyDutStorage(self).verify()
Otabek Kasimov27bb2862020-08-10 14:40:45 -07001148
Otabek Kasimov808cd832020-05-28 18:27:46 -07001149 logging.debug('Fail install image from USB; %s', e)
1150 raise error.AutoservError(
1151 'Failed to install image from USB due to unexpected '
1152 'error, please check debug log for details.')
Garry Wang033a31e2020-04-10 17:20:49 -07001153 finally:
1154 # We need reset the DUT no matter re-install success or not,
1155 # as we don't want leave the DUT in boot from usb state.
1156 logging.info('Power cycling DUT through servo.')
1157 self.servo.get_power_state_controller().power_off()
1158 self.servo.switch_usbkey('off')
Garry Wang53fc8f32020-09-18 13:30:08 -07001159 if need_snk:
1160 # Attempt to restore servo_v4 role to 'src' mode.
1161 self.servo.set_servo_v4_role('src')
Garry Wang033a31e2020-04-10 17:20:49 -07001162 # N.B. The Servo API requires that we use power_on() here
1163 # for two reasons:
1164 # 1) After turning on a DUT in recovery mode, you must turn
1165 # it off and then on with power_on() once more to
1166 # disable recovery mode (this is a Parrot specific
1167 # requirement).
1168 # 2) After power_off(), the only way to turn on is with
1169 # power_on() (this is a Storm specific requirement).
1170 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -07001171
1172 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -08001173 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -07001174 raise hosts.AutoservRepairError('DUT failed to reboot installed '
1175 'test image after %d seconds' %
1176 self.BOOT_TIMEOUT,
1177 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -05001178
1179
Garry Wanga2e78172020-09-09 23:49:07 -07001180 def set_servo_host(self, host, servo_state=None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001181 """Set our servo host member, and associated servo.
1182
1183 @param host Our new `ServoHost`.
1184 """
1185 self._servo_host = host
Derek Beckettb66e5c82020-08-12 15:31:02 -07001186 self.servo_pwr_supported = None
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001187 if self._servo_host is not None:
1188 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001189 servo_state = self._servo_host.get_servo_state()
Garry Wang000c6c02020-05-11 21:27:23 -07001190 self._set_smart_usbhub_label(self._servo_host.smart_usbhub)
Derek Beckettb66e5c82020-08-12 15:31:02 -07001191 try:
1192 self.servo_pwr_supported = self.servo.has_control('power_state')
1193 except Exception as e:
1194 logging.debug(
1195 "Could not get servo power state due to {}".format(e))
Gregory Nisbet93b23e22020-10-02 20:42:16 +00001196 else:
1197 self.servo = None
Derek Beckettb66e5c82020-08-12 15:31:02 -07001198 self.servo_pwr_supported = False
Otabek Kasimov41301a22020-05-10 15:28:21 -07001199 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001200 self.set_servo_state(servo_state)
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07001201 self._set_servo_topology()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001202
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001203
Richard Barnette9a26ad62016-06-10 12:03:08 -07001204 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001205 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001206 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001207
Richard Barnette9a26ad62016-06-10 12:03:08 -07001208 If the servo object is missing, attempt to repair the servo
1209 host. Repair failures are passed back to the caller.
1210
1211 @raise AutoservError: If there is no servo host for this CrOS
1212 host.
1213 """
1214 if self.servo:
1215 return
1216 if not self._servo_host:
1217 raise error.AutoservError('No servo host for %s.' %
1218 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001219 try:
1220 self._servo_host.repair()
1221 except:
1222 raise
1223 finally:
1224 self.set_servo_host(self._servo_host)
1225
1226
Otabek Kasimov41301a22020-05-10 15:28:21 -07001227 def set_servo_type(self):
1228 """Set servo info labels to dut host_info"""
1229 if not self.servo:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001230 logging.debug('Servo is not initialized to get servo_type.')
Otabek Kasimov41301a22020-05-10 15:28:21 -07001231 return
Otabek Kasimov1b70e8d2020-12-30 13:51:00 -08001232 if not self.is_servo_in_working_state():
1233 logging.debug('Servo is not good, skip update servo_type.')
1234 return
Otabek Kasimov41301a22020-05-10 15:28:21 -07001235 servo_type = self.servo.get_servo_type()
1236 if not servo_type:
Otabek Kasimovbea89e52020-05-12 15:40:00 -07001237 logging.debug('Cannot collect servo_type from servo'
Otabek Kasimov41301a22020-05-10 15:28:21 -07001238 ' by `dut-control servo_type`! Please file a bug'
1239 ' and inform infra team as we are not expected '
1240 ' to reach this point.')
1241 return
1242 host_info = self.host_info_store.get()
1243 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1244 old_type = host_info.get_label_value(prefix)
1245 if old_type == servo_type:
1246 # do not need update
1247 return
1248 host_info.set_version_label(prefix, servo_type)
1249 self.host_info_store.commit(host_info)
1250 logging.info('ServoHost: servo_type updated to %s '
1251 '(previous: %s)', servo_type, old_type)
1252
1253
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001254 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001255 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001256 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001257 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001258 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001259 old_state = host_info.get_label_value(servo_state_prefix)
1260 if old_state == servo_state:
1261 # do not need update
1262 return
1263 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001264 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001265 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1266 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001267
1268
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001269 def get_servo_state(self):
1270 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001271 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001272 return host_info.get_label_value(servo_state_prefix)
1273
Otabek Kasimov5f039202020-10-28 15:45:29 -07001274 def is_servo_in_working_state(self):
1275 """Validate servo is in WORKING state."""
1276 servo_state = self.get_servo_state()
1277 return servo_state == servo_constants.SERVO_STATE_WORKING
1278
Dana Goyette655af512020-09-03 10:48:23 -07001279 def get_servo_usb_state(self):
1280 """Get the label value indicating the health of the USB drive.
1281
1282 @return: The label value if defined, otherwise '' (empty string).
1283 @rtype: str
1284 """
1285 host_info = self.host_info_store.get()
1286 servo_usb_state_prefix = audit_const.SERVO_USB_STATE_PREFIX
1287 return host_info.get_label_value(servo_usb_state_prefix)
1288
1289 def is_servo_usb_usable(self):
1290 """Check if the servo USB storage device is usable for FAFT.
1291
1292 @return: False if the label indicates a state that will break FAFT.
1293 True if state is okay, or if state is not defined.
1294 @rtype: bool
1295 """
1296 usb_state = self.get_servo_usb_state()
1297 return usb_state in ('', audit_const.HW_STATE_ACCEPTABLE,
1298 audit_const.HW_STATE_NORMAL,
1299 audit_const.HW_STATE_UNKNOWN)
Otabek Kasimov41301a22020-05-10 15:28:21 -07001300
Garry Wang000c6c02020-05-11 21:27:23 -07001301 def _set_smart_usbhub_label(self, smart_usbhub_detected):
1302 if smart_usbhub_detected is None:
1303 # skip the label update here as this indicate we wasn't able
1304 # to confirm usbhub type.
1305 return
1306 host_info = self.host_info_store.get()
1307 if (smart_usbhub_detected ==
1308 (servo_constants.SMART_USBHUB_LABEL in host_info.labels)):
1309 # skip label update if current label match the truth.
1310 return
1311 if smart_usbhub_detected:
1312 logging.info('Adding %s label to host %s',
1313 servo_constants.SMART_USBHUB_LABEL,
1314 self.hostname)
1315 host_info.labels.append(servo_constants.SMART_USBHUB_LABEL)
1316 else:
1317 logging.info('Removing %s label from host %s',
1318 servo_constants.SMART_USBHUB_LABEL,
1319 self.hostname)
1320 host_info.labels.remove(servo_constants.SMART_USBHUB_LABEL)
1321 self.host_info_store.commit(host_info)
1322
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001323 def repair(self):
1324 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001325
1326 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001327 not call back to the parent class, but instead relies on
1328 `self._repair_strategy` to coordinate the verification and
1329 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001330 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001331 message = 'Beginning repair for host %s board %s model %s'
1332 info = self.host_info_store.get()
1333 message %= (self.hostname, info.board, info.model)
1334 self.record('INFO', None, None, message)
Garry Wanga2e78172020-09-09 23:49:07 -07001335 profile_state = profile_constants.DUT_STATE_READY
Shijin Abraham78ce4402020-09-08 22:04:27 -07001336 # Initialize bluetooth peers
1337 self.initialize_btpeer()
Garry Wang87af1d02020-05-26 17:55:54 -07001338 try:
1339 self._repair_strategy.repair(self)
1340 except hosts.AutoservVerifyDependencyError as e:
Otabek Kasimovd48389b2020-12-07 02:38:34 -08001341 # TODO(otabek): remove when finish b/174191325
1342 self._stat_if_pingable_but_not_sshable()
Garry Wang87af1d02020-05-26 17:55:54 -07001343 # We don't want flag a DUT as failed if only non-critical
1344 # verifier(s) failed during the repair.
1345 if e.is_critical():
Garry Wanga2e78172020-09-09 23:49:07 -07001346 profile_state = profile_constants.DUT_STATE_REPAIR_FAILED
Otabek Kasimov86062d02020-11-17 13:30:22 -08001347 self._reboot_labstation_if_needed()
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07001348 self.try_set_device_needs_manual_repair()
Garry Wang87af1d02020-05-26 17:55:54 -07001349 raise
Garry Wanga2e78172020-09-09 23:49:07 -07001350 finally:
1351 self.set_health_profile_dut_state(profile_state)
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001352
Otabek Kasimov8bb09912020-10-01 14:44:57 -07001353 def get_verifier_state(self, tag):
1354 """Return the state of servo verifier.
1355
1356 @returns: bool or None
1357 """
1358 return self._repair_strategy.verifier_is_good(tag)
Richard Barnette82c35912012-11-20 10:09:10 -08001359
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001360 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001361 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001362 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001363
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001364 if self._chameleon_host:
1365 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001366
Garry Wang1a493d82020-08-31 21:01:19 -07001367 if self.health_profile:
1368 try:
1369 self.health_profile.close()
1370 except Exception as e:
1371 logging.warning(
1372 'Failed to finalize device health profile; %s', e)
1373
xixuand6011f12016-12-08 15:01:58 -08001374 if self._servo_host:
1375 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001376
Dan Shi49ca0932014-11-14 11:22:27 -08001377 def get_power_supply_info(self):
1378 """Get the output of power_supply_info.
1379
1380 power_supply_info outputs the info of each power supply, e.g.,
1381 Device: Line Power
1382 online: no
1383 type: Mains
1384 voltage (V): 0
1385 current (A): 0
1386 Device: Battery
1387 state: Discharging
1388 percentage: 95.9276
1389 technology: Li-ion
1390
1391 Above output shows two devices, Line Power and Battery, with details of
1392 each device listed. This function parses the output into a dictionary,
1393 with key being the device name, and value being a dictionary of details
1394 of the device info.
1395
1396 @return: The dictionary of power_supply_info, e.g.,
1397 {'Line Power': {'online': 'yes', 'type': 'main'},
1398 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001399 @raise error.AutoservRunError if power_supply_info tool is not found in
1400 the DUT. Caller should handle this error to avoid false failure
1401 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001402 """
1403 result = self.run('power_supply_info').stdout.strip()
1404 info = {}
1405 device_name = None
1406 device_info = {}
1407 for line in result.split('\n'):
1408 pair = [v.strip() for v in line.split(':')]
1409 if len(pair) != 2:
1410 continue
1411 if pair[0] == 'Device':
1412 if device_name:
1413 info[device_name] = device_info
1414 device_name = pair[1]
1415 device_info = {}
1416 else:
1417 device_info[pair[0]] = pair[1]
1418 if device_name and not device_name in info:
1419 info[device_name] = device_info
1420 return info
1421
1422
1423 def get_battery_percentage(self):
1424 """Get the battery percentage.
1425
1426 @return: The percentage of battery level, value range from 0-100. Return
1427 None if the battery info cannot be retrieved.
1428 """
1429 try:
1430 info = self.get_power_supply_info()
1431 logging.info(info)
1432 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001433 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001434 return None
1435
1436
Philip Chenaf69ead2020-03-27 13:06:42 -07001437 def get_battery_state(self):
1438 """Get the battery charging state.
1439
1440 @return: A string representing the battery charging state. It can be
1441 'Charging', 'Fully charged', or 'Discharging'.
1442 """
1443 try:
1444 info = self.get_power_supply_info()
1445 logging.info(info)
1446 return info['Battery']['state']
1447 except (KeyError, ValueError, error.AutoservRunError):
1448 return None
1449
1450
Daniel Campello8ca25c22019-12-13 16:48:26 -07001451 def get_battery_display_percentage(self):
1452 """Get the battery display percentage.
1453
1454 @return: The display percentage of battery level, value range from
1455 0-100. Return None if the battery info cannot be retrieved.
1456 """
1457 try:
1458 info = self.get_power_supply_info()
1459 logging.info(info)
1460 return float(info['Battery']['display percentage'])
1461 except (KeyError, ValueError, error.AutoservRunError):
1462 return None
1463
1464
Dan Shi49ca0932014-11-14 11:22:27 -08001465 def is_ac_connected(self):
1466 """Check if the dut has power adapter connected and charging.
1467
1468 @return: True if power adapter is connected and charging.
1469 """
1470 try:
1471 info = self.get_power_supply_info()
1472 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001473 except (KeyError, error.AutoservRunError):
1474 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001475
1476
Simran Basi5e6339a2013-03-21 11:34:32 -07001477 def _cleanup_poweron(self):
1478 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001479 info = self.host_info_store.get()
1480 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001481 return
1482 logging.debug('This host has recently interacted with the RPM'
1483 ' Infrastructure. Ensuring power is on.')
1484 try:
1485 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001486 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001487 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001488 logging.error('Failed to turn Power On for this host after '
1489 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001490
1491 battery_percentage = self.get_battery_percentage()
Otabek Kasimov58e22562020-11-03 17:17:41 -08001492 if (
1493 battery_percentage
1494 and battery_percentage < cros_constants.MIN_BATTERY_LEVEL):
Dan Shi49ca0932014-11-14 11:22:27 -08001495 raise
1496 elif self.is_ac_connected():
1497 logging.info('The device has power adapter connected and '
1498 'charging. No need to try to turn RPM on '
1499 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001500 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001501 logging.info('Battery level is now at %s%%. The device may '
1502 'still have enough power to run test, so no '
1503 'exception will be raised.', battery_percentage)
1504
Simran Basi5e6339a2013-03-21 11:34:32 -07001505
Garry Wangad4d4fd2019-01-30 17:00:38 -08001506 def _remove_rpm_changed_tag(self):
1507 info = self.host_info_store.get()
1508 del info.attributes[self._RPM_OUTLET_CHANGED]
1509 self.host_info_store.commit(info)
1510
1511
1512 def _add_rpm_changed_tag(self):
1513 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001514 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001515 self.host_info_store.commit(info)
1516
1517
1518
beepsc87ff602013-07-31 21:53:00 -07001519 def _is_factory_image(self):
1520 """Checks if the image on the DUT is a factory image.
1521
1522 @return: True if the image on the DUT is a factory image.
1523 False otherwise.
1524 """
1525 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1526 return result.exit_status == 0
1527
1528
1529 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001530 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001531
1532 @raises: FactoryImageCheckerException for factory images, since
1533 we cannot attempt to restart ui on them.
1534 error.AutoservRunError for any other type of error that
1535 occurs while restarting ui.
1536 """
1537 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001538 raise FactoryImageCheckerException('Cannot restart ui on factory '
1539 'images')
beepsc87ff602013-07-31 21:53:00 -07001540
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001541 # TODO(jrbarnette): The command to stop/start the ui job
1542 # should live inside cros_ui, too. However that would seem
1543 # to imply interface changes to the existing start()/restart()
1544 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001545 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001546 self.run('stop ui; start ui')
1547 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001548
1549
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001550 def _start_powerd_if_needed(self):
1551 """Start powerd if it isn't already running."""
1552 self.run('start powerd', ignore_status=True)
1553
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001554 def _read_arc_prop_file(self, filename):
1555 for path in [
1556 '/usr/share/arcvm/properties/', '/usr/share/arc/properties/'
1557 ]:
1558 if self.path_exists(path + filename):
1559 return utils.parse_cmd_output('cat ' + path + filename,
1560 run_method=self.run)
1561 return None
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001562
Jiyoun Hac172ee72020-12-15 08:57:29 +09001563 def _get_arc_build_info(self):
1564 """Returns a dictionary mapping build properties to their values."""
1565 build_info = None
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001566 for filename in ['build.prop', 'vendor_build.prop']:
1567 properties = self._read_arc_prop_file(filename)
1568 if properties:
1569 if build_info:
1570 build_info.update(properties)
1571 else:
1572 build_info = properties
1573 else:
1574 logging.error('Failed to find %s in device.', filename)
Jiyoun Hac172ee72020-12-15 08:57:29 +09001575 return build_info
1576
Jiyoun Haba37f312021-01-13 09:44:16 +09001577 def get_arc_primary_abi(self):
Jiyoun Hac172ee72020-12-15 08:57:29 +09001578 """Returns the primary abi of the host."""
1579 return self._get_arc_build_info().get('ro.product.cpu.abi')
1580
Jiyoun Haba37f312021-01-13 09:44:16 +09001581 def get_arc_security_patch(self):
Jiyoun Hac172ee72020-12-15 08:57:29 +09001582 """Returns the security patch of the host."""
1583 return self._get_arc_build_info().get('ro.build.version.security_patch')
1584
Kazuhiro Inabaa7a00492020-12-17 16:05:12 +09001585 def get_arc_first_api_level(self):
1586 """Returns the security patch of the host."""
1587 return self._get_arc_build_info().get('ro.product.first_api_level')
1588
xixuana3bbc422017-05-04 15:57:21 -07001589 def _get_lsb_release_content(self):
1590 """Return the content of lsb-release file of host."""
1591 return self.run(
1592 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1593
1594
Dan Shi549fb822015-03-24 18:01:11 -07001595 def get_release_version(self):
1596 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1597
1598 @returns The version string in lsb-release, under attribute
1599 CHROMEOS_RELEASE_VERSION.
1600 """
Dan Shi549fb822015-03-24 18:01:11 -07001601 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001602 lsb_release_content=self._get_lsb_release_content())
1603
1604
Don Garrettb9f35802018-01-22 18:25:40 -08001605 def get_release_builder_path(self):
1606 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1607
1608 @returns The version string in lsb-release, under attribute
1609 CHROMEOS_RELEASE_BUILDER_PATH.
1610 """
1611 return lsbrelease_utils.get_chromeos_release_builder_path(
1612 lsb_release_content=self._get_lsb_release_content())
1613
1614
xixuana3bbc422017-05-04 15:57:21 -07001615 def get_chromeos_release_milestone(self):
1616 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1617 from lsb-release.
1618
1619 @returns The version string in lsb-release, under attribute
1620 CHROMEOS_RELEASE_BUILD_TYPE.
1621 """
1622 return lsbrelease_utils.get_chromeos_release_milestone(
1623 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001624
1625
1626 def verify_cros_version_label(self):
Garry Wangd18e7b32020-08-07 18:31:44 -07001627 """Verify if host's cros-version label match the actual image in dut.
Dan Shi549fb822015-03-24 18:01:11 -07001628
Garry Wangd18e7b32020-08-07 18:31:44 -07001629 @returns True if the label match with image in dut, otherwise False
Dan Shi549fb822015-03-24 18:01:11 -07001630 """
Garry Wangd18e7b32020-08-07 18:31:44 -07001631 os_from_host = self.get_release_builder_path()
1632 info = self.host_info_store.get()
1633 os_from_label = info.get_label_value(self.VERSION_PREFIX)
1634 if not os_from_label:
1635 logging.debug('No existing %s label detected', self.VERSION_PREFIX)
1636 return True
1637
1638 # known cases where the version label will not match the
1639 # original CHROMEOS_RELEASE_BUILDER_PATH setting:
1640 # * Tests for the `arc-presubmit` append "-cheetsth" to the label.
1641 if os_from_label.endswith(provision.CHEETS_SUFFIX):
1642 logging.debug('%s label with %s suffix detected, this suffix will'
1643 ' be ignored when comparing label.',
1644 self.VERSION_PREFIX, provision.CHEETS_SUFFIX)
1645 os_from_label = os_from_label[:-len(provision.CHEETS_SUFFIX)]
1646 logging.debug('OS version from host: %s; OS verision cached in '
1647 'label: %s', os_from_host, os_from_label)
1648 return os_from_label == os_from_host
Dan Shi549fb822015-03-24 18:01:11 -07001649
1650
Laurence Goodby778c9a42017-05-24 19:24:07 -07001651 def cleanup_services(self):
1652 """Reinitializes the device for cleanup.
1653
1654 Subclasses may override this to customize the cleanup method.
1655
1656 To indicate failure of the reset, the implementation may raise
1657 any of:
1658 error.AutoservRunError
1659 error.AutotestRunError
1660 FactoryImageCheckerException
1661
1662 @raises error.AutoservRunError
1663 @raises error.AutotestRunError
1664 @raises error.FactoryImageCheckerException
1665 """
1666 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001667 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001668
1669
Gregory Nisbetec615d62020-12-11 17:59:20 +00001670 def cleanup(self):
1671 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001672 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001673 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001674 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001675 except (error.AutotestRunError, error.AutoservRunError,
1676 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001677 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001678
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001679 # cleanup routines, i.e. reboot the machine.
Gregory Nisbetec615d62020-12-11 17:59:20 +00001680 super(CrosHost, self).cleanup()
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001681
Simran Basi5e6339a2013-03-21 11:34:32 -07001682 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001683 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001684 self._cleanup_poweron()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001685
Gregory Nisbetec615d62020-12-11 17:59:20 +00001686
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001687 def reboot(self, **dargs):
1688 """
1689 This function reboots the site host. The more generic
1690 RemoteHost.reboot() performs sync and sleeps for 5
1691 seconds. This is not necessary for Chrome OS devices as the
1692 sync should be finished in a short time during the reboot
1693 command.
1694 """
Gregory Nisbetec615d62020-12-11 17:59:20 +00001695 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001696 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001697 dargs['reboot_cmd'] = ('sleep 1; '
1698 'reboot & sleep %d; '
1699 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001700 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001701 if 'fastsync' not in dargs:
1702 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001703
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001704 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001705 # Record who called us
1706 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001707 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001708 'dut_host_name' : self.hostname,
1709 'success' : True}
1710 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001711 'caller' : "%s:%s" % (orig.co_filename,
1712 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001713 'success' : True,
1714 'error' : ''}
1715
Vincent Palatin80780b22016-07-27 16:02:37 +02001716 t0 = time.time()
1717 try:
1718 super(CrosHost, self).reboot(**dargs)
1719 except Exception as e:
1720 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001721 metric_debug_fields['success'] = False
1722 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001723 raise
1724 finally:
1725 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001726 metrics.Counter(
1727 'chromeos/autotest/autoserv/reboot_count').increment(
1728 fields=metric_fields)
1729 metrics.Counter(
1730 'chromeos/autotest/autoserv/reboot_debug').increment(
1731 fields=metric_debug_fields)
1732 metrics.SecondsDistribution(
1733 'chromeos/autotest/autoserv/reboot_duration').add(
1734 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001735
1736
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001737 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001738 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001739 """
1740 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001741
1742 @param suspend_time: How long to suspend as integer seconds.
1743 @param suspend_cmd: Suspend command to execute.
1744 @param allow_early_resume: If False and if device resumes before
1745 |suspend_time|, throw an error.
1746
1747 @exception AutoservSuspendError Host resumed earlier than
1748 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001749 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001750
1751 if suspend_cmd is None:
1752 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001753 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001754 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001755 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001756 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1757 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001758
1759
Simran Basiec564392014-08-25 16:48:09 -07001760 def upstart_status(self, service_name):
1761 """Check the status of an upstart init script.
1762
1763 @param service_name: Service to look up.
1764
1765 @returns True if the service is running, False otherwise.
1766 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001767 return 'start/running' in self.run('status %s' % service_name,
1768 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001769
Tom Hughese9552342018-12-18 14:29:25 -08001770 def upstart_stop(self, service_name):
1771 """Stops an upstart job if it's running.
1772
1773 @param service_name: Service to stop
1774
1775 @returns True if service has been stopped or was already stopped
1776 False otherwise.
1777 """
1778 if not self.upstart_status(service_name):
1779 return True
1780
1781 result = self.run('stop %s' % service_name, ignore_status=True)
1782 if result.exit_status != 0:
1783 return False
1784 return True
1785
1786 def upstart_restart(self, service_name):
1787 """Restarts (or starts) an upstart job.
1788
1789 @param service_name: Service to start/restart
1790
1791 @returns True if service has been started/restarted, False otherwise.
1792 """
1793 cmd = 'start'
1794 if self.upstart_status(service_name):
1795 cmd = 'restart'
1796 cmd = cmd + ' %s' % service_name
1797 result = self.run(cmd)
1798 if result.exit_status != 0:
1799 return False
1800 return True
Simran Basiec564392014-08-25 16:48:09 -07001801
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001802 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001803 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001804
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001805 Tests for the following conditions:
1806 1. All conditions tested by the parent version of this
1807 function.
1808 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001809 3. Sufficient space in /mnt/stateful_partition/encrypted.
1810 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001811
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001812 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001813 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001814 default_kilo_inodes_required = CONFIG.get_config_value(
1815 'SERVER', 'kilo_inodes_required', type=int, default=100)
1816 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1817 kilo_inodes_required = CONFIG.get_config_value(
1818 'SERVER', 'kilo_inodes_required_%s' % board,
1819 type=int, default=default_kilo_inodes_required)
1820 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001821 self.check_diskspace(
1822 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001823 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001824 'SERVER', 'gb_diskspace_required', type=float,
1825 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001826 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1827 # Not all targets build with encrypted stateful support.
1828 if self.path_exists(encrypted_stateful_path):
1829 self.check_diskspace(
1830 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001831 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001832 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1833 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001834
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001835 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001836
beepsc87ff602013-07-31 21:53:00 -07001837 # Factory images don't run update engine,
1838 # goofy controls dbus on these DUTs.
1839 if not self._is_factory_image():
1840 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001841
1842
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001843 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1844 def wait_for_system_services(self):
1845 """Waits for system-services to be running.
1846
1847 Sometimes, update_engine will take a while to update firmware, so we
1848 should give this some time to finish. See crbug.com/765686#c38 for
1849 details.
1850 """
1851 if not self.upstart_status('system-services'):
1852 raise error.AutoservError('Chrome failed to reach login. '
1853 'System services not running.')
1854
1855
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001856 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001857 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001858 message = 'Beginning verify for host %s board %s model %s'
1859 info = self.host_info_store.get()
1860 message %= (self.hostname, info.board, info.model)
1861 self.record('INFO', None, None, message)
Garry Wang87af1d02020-05-26 17:55:54 -07001862 try:
1863 self._repair_strategy.verify(self)
1864 except hosts.AutoservVerifyDependencyError as e:
1865 # We don't want flag a DUT as failed if only non-critical
1866 # verifier(s) failed during the repair.
1867 if e.is_critical():
1868 raise
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001869
1870
Fang Deng96667ca2013-08-01 17:46:18 -07001871 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001872 connect_timeout=None, alive_interval=None,
1873 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001874 """Override default make_ssh_command to use options tuned for Chrome OS.
1875
1876 Tuning changes:
1877 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1878 connection failure. Consistency with remote_access.sh.
1879
Samuel Tan2ce155b2015-06-23 18:24:38 -07001880 - ServerAliveInterval=900; which causes SSH to ping connection every
1881 900 seconds. In conjunction with ServerAliveCountMax ensures
1882 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001883 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001884 the test completed successfully. Later increased from 180 seconds to
1885 900 seconds to account for tests where the DUT is suspended for
1886 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001887
1888 - ServerAliveCountMax=3; consistency with remote_access.sh.
1889
1890 - ConnectAttempts=4; reduce flakiness in connection errors;
1891 consistency with remote_access.sh.
1892
1893 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1894 Host keys change with every new installation, don't waste
1895 memory/space saving them.
1896
1897 - SSH protocol forced to 2; needed for ServerAliveInterval.
1898
1899 @param user User name to use for the ssh connection.
1900 @param port Port on the target host to use for ssh connection.
1901 @param opts Additional options to the ssh command.
1902 @param hosts_file Ignored.
1903 @param connect_timeout Ignored.
1904 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001905 @param alive_count_max Ignored.
1906 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001907 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001908 options = ' '.join([opts, '-o Protocol=2'])
1909 return super(CrosHost, self).make_ssh_command(
1910 user=user, port=port, opts=options, hosts_file='/dev/null',
1911 connect_timeout=30, alive_interval=900, alive_count_max=3,
1912 connection_attempts=4)
1913
1914
Jason Abeleb6f924f2013-11-13 16:01:54 -08001915 def syslog(self, message, tag='autotest'):
1916 """Logs a message to syslog on host.
1917
1918 @param message String message to log into syslog
1919 @param tag String tag prefix for syslog
1920
1921 """
1922 self.run('logger -t "%s" "%s"' % (tag, message))
1923
1924
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001925 def _ping_check_status(self, status):
1926 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001927
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001928 @param status Check the ping status against this value.
1929 @return True iff `status` and the result of ping are the same
1930 (i.e. both True or both False).
1931
1932 """
Abhishek Pandit-Subedi038df162020-09-14 16:37:43 -07001933 ping_val = utils.ping(self.hostname,
1934 tries=1,
1935 deadline=1,
1936 timeout=2,
1937 ignore_timeout=True)
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001938 return not (status ^ (ping_val == 0))
1939
1940 def _ping_wait_for_status(self, status, timeout):
1941 """Wait for the host to have a given status (UP or DOWN).
1942
1943 Status is checked by polling. Polling will not last longer
1944 than the number of seconds in `timeout`. The polling
1945 interval will be long enough that only approximately
1946 _PING_WAIT_COUNT polling cycles will be executed, subject
1947 to a maximum interval of about one minute.
1948
1949 @param status Waiting will stop immediately if `ping` of the
1950 host returns this status.
1951 @param timeout Poll for at most this many seconds.
1952 @return True iff the host status from `ping` matched the
1953 requested status at the time of return.
1954
1955 """
1956 # _ping_check_status() takes about 1 second, hence the
1957 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001958 # FIXME: if the ping command errors then _ping_check_status()
1959 # returns instantly. If timeout is also smaller than twice
1960 # _PING_WAIT_COUNT then the while loop below forks many
1961 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1962 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1963 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001964 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1965 end_time = time.time() + timeout
1966 while time.time() <= end_time:
1967 if self._ping_check_status(status):
1968 return True
1969 if poll_interval > 0:
1970 time.sleep(poll_interval)
1971
1972 # The last thing we did was sleep(poll_interval), so it may
1973 # have been too long since the last `ping`. Check one more
1974 # time, just to be sure.
1975 return self._ping_check_status(status)
1976
1977 def ping_wait_up(self, timeout):
1978 """Wait for the host to respond to `ping`.
1979
1980 N.B. This method is not a reliable substitute for
1981 `wait_up()`, because a host that responds to ping will not
1982 necessarily respond to ssh. This method should only be used
1983 if the target DUT can be considered functional even if it
1984 can't be reached via ssh.
1985
1986 @param timeout Minimum time to allow before declaring the
1987 host to be non-responsive.
1988 @return True iff the host answered to ping before the timeout.
1989
1990 """
1991 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001992
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001993 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001994 """Wait until the host no longer responds to `ping`.
1995
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001996 This function can be used as a slightly faster version of
1997 `wait_down()`, by avoiding potentially long ssh timeouts.
1998
1999 @param timeout Minimum time to allow for the host to become
2000 non-responsive.
2001 @return True iff the host quit answering ping before the
2002 timeout.
2003
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002004 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08002005 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002006
Anand K Mistry50f218e2020-07-31 14:50:15 +10002007 def _is_host_port_forwarded(self):
Garry Wanga2e78172020-09-09 23:49:07 -07002008 """Checks if the dut is connected over port forwarding.
Anand K Mistry50f218e2020-07-31 14:50:15 +10002009
2010 N.B. This method does not detect all situations where port forwarding is
2011 occurring. Namely, running autotest on the dut may result in a
2012 false-positive, and port forwarding using a different machine on the
2013 same network will be a false-negative.
2014
2015 @return True if the dut is connected over port forwarding
2016 False otherwise
2017 """
Garry Wanga2e78172020-09-09 23:49:07 -07002018 is_localhost = self.hostname in ['localhost', '127.0.0.1']
2019 is_forwarded = is_localhost and not self.is_default_port
2020 if is_forwarded:
2021 logging.info('Detected DUT connected by port forwarding')
2022 return is_forwarded
Anand K Mistry50f218e2020-07-31 14:50:15 +10002023
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002024 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002025 """Wait for the client to enter low-power sleep mode.
2026
2027 The test for "is asleep" can't distinguish a system that is
2028 powered off; to confirm that the unit was asleep, it is
2029 necessary to force resume, and then call
2030 `test_wait_for_resume()`.
2031
2032 This function is expected to be called from a test as part
2033 of a sequence like the following:
2034
2035 ~~~~~~~~
2036 boot_id = host.get_boot_id()
2037 # trigger sleep on the host
2038 host.test_wait_for_sleep()
2039 # trigger resume on the host
2040 host.test_wait_for_resume(boot_id)
2041 ~~~~~~~~
2042
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002043 @param sleep_timeout time limit in seconds to allow the host sleep.
2044
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002045 @exception TestFail The host did not go to sleep within
2046 the allowed time.
2047 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002048 if sleep_timeout is None:
2049 sleep_timeout = self.SLEEP_TIMEOUT
2050
Anand K Mistry50f218e2020-07-31 14:50:15 +10002051 # If the dut is accessed over SSH port-forwarding, `ping` is not useful
2052 # for detecting the dut is down since a ping to localhost will always
2053 # succeed. In this case, fall back to wait_down() which uses SSH.
2054 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07002055 success = self.wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002056 else:
Garry Wanga2e78172020-09-09 23:49:07 -07002057 success = self.ping_wait_down(timeout=sleep_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002058
2059 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002060 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002061 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002062
2063
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002064 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002065 """Wait for the client to resume from low-power sleep mode.
2066
2067 The `old_boot_id` parameter should be the value from
2068 `get_boot_id()` obtained prior to entering sleep mode. A
2069 `TestFail` exception is raised if the boot id changes.
2070
2071 See @ref test_wait_for_sleep for more on this function's
2072 usage.
2073
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002074 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002075 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002076 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002077
2078 @exception TestFail The host did not respond within the
2079 allowed time.
2080 @exception TestFail The host responded, but the boot id test
2081 indicated a reboot rather than a sleep
2082 cycle.
2083 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002084 if resume_timeout is None:
2085 resume_timeout = self.RESUME_TIMEOUT
2086
2087 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002088 raise error.TestFail(
2089 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08002090 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002091 else:
2092 new_boot_id = self.get_boot_id()
2093 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002094 logging.error('client rebooted (old boot %s, new boot %s)',
2095 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002096 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002097 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002098
2099
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002100 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002101 """Wait for the client to shut down.
2102
2103 The test for "has shut down" can't distinguish a system that
2104 is merely asleep; to confirm that the unit was down, it is
2105 necessary to force boot, and then call test_wait_for_boot().
2106
2107 This function is expected to be called from a test as part
2108 of a sequence like the following:
2109
2110 ~~~~~~~~
2111 boot_id = host.get_boot_id()
2112 # trigger shutdown on the host
2113 host.test_wait_for_shutdown()
2114 # trigger boot on the host
2115 host.test_wait_for_boot(boot_id)
2116 ~~~~~~~~
2117
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002118 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002119 @exception TestFail The host did not shut down within the
2120 allowed time.
2121 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002122 if shutdown_timeout is None:
2123 shutdown_timeout = self.SHUTDOWN_TIMEOUT
2124
Anand K Mistry50f218e2020-07-31 14:50:15 +10002125 if self._is_host_port_forwarded():
Garry Wanga2e78172020-09-09 23:49:07 -07002126 success = self.wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002127 else:
Garry Wanga2e78172020-09-09 23:49:07 -07002128 success = self.ping_wait_down(timeout=shutdown_timeout)
Anand K Mistry50f218e2020-07-31 14:50:15 +10002129
2130 if not success:
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002131 raise error.TestFail(
2132 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08002133 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002134
2135
2136 def test_wait_for_boot(self, old_boot_id=None):
2137 """Wait for the client to boot from cold power.
2138
2139 The `old_boot_id` parameter should be the value from
2140 `get_boot_id()` obtained prior to shutting down. A
2141 `TestFail` exception is raised if the boot id does not
2142 change. The boot id test is omitted if `old_boot_id` is not
2143 specified.
2144
2145 See @ref test_wait_for_shutdown for more on this function's
2146 usage.
2147
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08002148 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002149 shut down.
2150
2151 @exception TestFail The host did not respond within the
2152 allowed time.
2153 @exception TestFail The host responded, but the boot id test
2154 indicated that there was no reboot.
2155 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002156 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002157 raise error.TestFail(
2158 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07002159 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002160 elif old_boot_id:
2161 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002162 logging.error('client not rebooted (boot %s)',
2163 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07002164 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08002165 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07002166
2167
2168 @staticmethod
2169 def check_for_rpm_support(hostname):
2170 """For a given hostname, return whether or not it is powered by an RPM.
2171
Simran Basi1df55112013-09-06 11:25:09 -07002172 @param hostname: hostname to check for rpm support.
2173
Simran Basid5e5e272012-09-24 15:23:59 -07002174 @return None if this host does not follows the defined naming format
2175 for RPM powered DUT's in the lab. If it does follow the format,
2176 it returns a regular expression MatchObject instead.
2177 """
Fang Dengbaff9082015-01-06 13:46:15 -08002178 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002179
2180
2181 def has_power(self):
2182 """For this host, return whether or not it is powered by an RPM.
2183
2184 @return True if this host is in the CROS lab and follows the defined
2185 naming format.
2186 """
Fang Deng0ca40e22013-08-27 17:47:44 -07002187 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07002188
2189
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002190 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07002191 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002192
2193 @param state Specifies which power state to set to DUT
2194 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002195 use. By default "RPM" or "CCD" will be used based
2196 on servo type. Valid values from
2197 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002198
2199 """
2200 ACCEPTABLE_STATES = ['ON', 'OFF']
2201
Garry Wang5e5538a2019-04-08 15:36:18 -07002202 if not power_method:
2203 power_method = self.get_default_power_method()
2204
2205 state = state.upper()
2206 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002207 raise error.TestError('State must be one of: %s.'
2208 % (ACCEPTABLE_STATES,))
2209
2210 if power_method == self.POWER_CONTROL_SERVO:
2211 logging.info('Setting servo port J10 to %s', state)
2212 self.servo.set('prtctl3_pwren', state.lower())
2213 time.sleep(self._USB_POWER_TIMEOUT)
2214 elif power_method == self.POWER_CONTROL_MANUAL:
2215 logging.info('You have %d seconds to set the AC power to %s.',
2216 self._POWER_CYCLE_TIMEOUT, state)
2217 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07002218 elif power_method == self.POWER_CONTROL_CCD:
2219 servo_role = 'src' if state == 'ON' else 'snk'
2220 logging.info('servo ccd power pass through detected,'
2221 ' changing servo_role to %s.', servo_role)
2222 self.servo.set_servo_v4_role(servo_role)
2223 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07002224 # Make sure we don't leave DUT with no power(servo_role=snk)
2225 # when DUT is not pingable, as we raise a exception here
2226 # that may break a power cycle in the middle.
2227 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07002228 raise error.AutoservError(
2229 'DUT failed to regain network connection after %d seconds.'
2230 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002231 else:
2232 if not self.has_power():
2233 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08002234 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07002235 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07002236
2237
Garry Wang5e5538a2019-04-08 15:36:18 -07002238 def power_off(self, power_method=None):
2239 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002240
2241 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002242 use. By default "RPM" or "CCD" will be used based
2243 on servo type. Valid values from
2244 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002245
2246 """
Derek Beckettb66e5c82020-08-12 15:31:02 -07002247 self._sync_if_up()
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002248 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07002249
Derek Beckettb66e5c82020-08-12 15:31:02 -07002250 def _check_supported(self):
2251 """Throw an error if dts mode control is not supported."""
2252 if not self.servo_pwr_supported:
2253 raise error.TestFail('power_state controls not supported')
2254
2255 def _sync_if_up(self):
2256 """Run sync on the DUT and wait for completion if the DUT is up.
2257
2258 Additionally, try to sync and ignore status if its not up.
2259
2260 Useful prior to reboots to ensure files are written to disc.
2261
2262 """
2263 if self.is_up_fast():
2264 self.run("sync")
2265 return
2266 # If it is not up, attempt to sync in the rare event the DUT is up but
2267 # doesn't respond to a ping. Ignore any errors.
2268 try:
2269 self.run("sync", ignore_status=True, timeout=1)
2270 except Exception:
2271 pass
2272
2273 def power_off_via_servo(self):
2274 """Force the DUT to power off.
2275
2276 The DUT is guaranteed to be off at the end of this call,
2277 regardless of its previous state, provided that there is
2278 working EC and boot firmware. There is no requirement for
2279 working OS software.
2280
2281 """
2282 self._check_supported()
2283 self._sync_if_up()
2284 self.servo.set_nocheck('power_state', 'off')
2285
2286 def power_on_via_servo(self, rec_mode='on'):
2287 """Force the DUT to power on.
2288
2289 Prior to calling this function, the DUT must be powered off,
2290 e.g. with a call to `power_off()`.
2291
2292 At power on, recovery mode is set as specified by the
2293 corresponding argument. When booting with recovery mode on, it
2294 is the caller's responsibility to unplug/plug in a bootable
2295 external storage device.
2296
2297 If the DUT requires a delay after powering on but before
2298 processing inputs such as USB stick insertion, the delay is
2299 handled by this method; the caller is not responsible for such
2300 delays.
2301
2302 @param rec_mode Setting of recovery mode to be applied at
2303 power on. default: REC_OFF aka 'off'
2304
2305 """
2306 self._check_supported()
2307 self.servo.set_nocheck('power_state', rec_mode)
2308
2309 def reset_via_servo(self):
2310 """Force the DUT to reset.
2311
2312 The DUT is guaranteed to be on at the end of this call,
2313 regardless of its previous state, provided that there is
2314 working OS software. This also guarantees that the EC has
2315 been restarted.
2316
2317 """
2318 self._check_supported()
2319 self._sync_if_up()
2320 self.servo.set_nocheck('power_state', 'reset')
2321
Simran Basid5e5e272012-09-24 15:23:59 -07002322
Garry Wang5e5538a2019-04-08 15:36:18 -07002323 def power_on(self, power_method=None):
2324 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002325
2326 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002327 use. By default "RPM" or "CCD" will be used based
2328 on servo type. Valid values from
2329 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002330
2331 """
2332 self._set_power('ON', power_method)
2333
2334
Garry Wang5e5538a2019-04-08 15:36:18 -07002335 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002336 """Cycle power to this host by turning it OFF, then ON.
2337
2338 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07002339 use. By default "RPM" or "CCD" will be used based
2340 on servo type. Valid values from
2341 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002342
2343 """
Garry Wang5e5538a2019-04-08 15:36:18 -07002344 if not power_method:
2345 power_method = self.get_default_power_method()
2346
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002347 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07002348 self.POWER_CONTROL_MANUAL,
2349 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08002350 self.power_off(power_method=power_method)
2351 time.sleep(self._POWER_CYCLE_TIMEOUT)
2352 self.power_on(power_method=power_method)
2353 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08002354 self._add_rpm_changed_tag()
2355 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07002356
2357
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002358 def get_platform_from_fwid(self):
2359 """Determine the platform from the crossystem fwid.
2360
2361 @returns a string representing this host's platform.
2362 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002363 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002364 crossystem = utils.Crossystem(self)
2365 crossystem.init()
2366 # Extract fwid value and use the leading part as the platform id.
2367 # fwid generally follow the format of {platform}.{firmware version}
2368 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
2369 platform = crossystem.fwid().split('.')[0].lower()
2370 # Newer platforms start with 'Google_' while the older ones do not.
2371 return platform.replace('google_', '')
2372
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002373
Simran Basic6f1f7a2012-10-16 10:47:46 -07002374 def get_platform(self):
2375 """Determine the correct platform label for this host.
2376
2377 @returns a string representing this host's platform.
2378 """
C Shapiroed87c6f2018-04-19 09:13:58 -06002379 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2380 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06002381 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002382 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06002383 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07002384 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07002385
2386
Greg Edelstona7b05d12020-04-01 16:00:51 -06002387 def get_model_from_cros_config(self):
2388 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002389
Greg Edelstona7b05d12020-04-01 16:00:51 -06002390 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002391 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06002392 return cros_config.call_cros_config_get_output('/ name',
2393 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08002394
2395
Hung-ying Tyanb1328032014-04-01 14:18:54 +08002396 def get_architecture(self):
2397 """Determine the correct architecture label for this host.
2398
2399 @returns a string representing this host's architecture.
2400 """
2401 crossystem = utils.Crossystem(self)
2402 crossystem.init()
2403 return crossystem.arch()
2404
2405
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002406 def get_chrome_version(self):
2407 """Gets the Chrome version number and milestone as strings.
2408
2409 Invokes "chrome --version" to get the version number and milestone.
2410
2411 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
2412 current Chrome version number as a string (in the form "W.X.Y.Z")
2413 and "milestone" is the first component of the version number
2414 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
2415 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
2416 of "chrome --version" and the milestone will be the empty string.
2417
2418 """
MK Ryu35d661e2014-09-25 17:44:10 -07002419 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08002420 return utils.parse_chrome_version(version_string)
2421
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002422
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002423 def get_ec_version(self):
2424 """Get the ec version as strings.
2425
2426 @returns a string representing this host's ec version.
2427 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002428 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002429 result = self.run(command, ignore_status=True)
2430 if result.exit_status != 0:
2431 return ''
2432 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002433
2434
2435 def get_firmware_version(self):
2436 """Get the firmware version as strings.
2437
2438 @returns a string representing this host's firmware version.
2439 """
2440 crossystem = utils.Crossystem(self)
2441 crossystem.init()
2442 return crossystem.fwid()
2443
2444
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08002445 def get_hardware_id(self):
2446 """Get hardware id as strings.
2447
2448 @returns a string representing this host's hardware id.
2449 """
2450 crossystem = utils.Crossystem(self)
2451 crossystem.init()
2452 return crossystem.hwid()
2453
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002454 def get_hardware_revision(self):
2455 """Get the hardware revision as strings.
2456
2457 @returns a string representing this host's hardware revision.
2458 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002459 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002460 result = self.run(command, ignore_status=True)
2461 if result.exit_status != 0:
2462 return ''
2463 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002464
2465
2466 def get_kernel_version(self):
2467 """Get the kernel version as strings.
2468
2469 @returns a string representing this host's kernel version.
2470 """
2471 return self.run('uname -r').stdout.strip()
2472
2473
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002474 def get_cpu_name(self):
2475 """Get the cpu name as strings.
2476
2477 @returns a string representing this host's cpu name.
2478 """
2479
2480 # Try get cpu name from device tree first
2481 if self.path_exists('/proc/device-tree/compatible'):
2482 command = ' | '.join(
2483 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2484 'tail -1'])
2485 return self.run(command).stdout.strip().replace(',', ' ')
2486
2487 # Get cpu name from uname -p
2488 command = 'uname -p'
2489 ret = self.run(command).stdout.strip()
2490
2491 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2492 # Try get cpu name from /proc/cpuinfo instead
2493 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2494 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2495 self = self.run(command).stdout.strip()
2496
2497 # Remove bloat from CPU name, for example
2498 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2499 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2500 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2501 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2502 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2503 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2504
2505
2506 def get_screen_resolution(self):
2507 """Get the screen(s) resolution as strings.
2508 In case of more than 1 monitor, return resolution for each monitor
2509 separate with plus sign.
2510
2511 @returns a string representing this host's screen(s) resolution.
2512 """
2513 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2514 ret = self.run(command, ignore_status=True)
2515 # We might have Chromebox without a screen
2516 if ret.exit_status != 0:
2517 return ''
2518 return ret.stdout.strip().replace('\n', '+')
2519
2520
2521 def get_mem_total_gb(self):
2522 """Get total memory available in the system in GiB (2^20).
2523
2524 @returns an integer representing total memory
2525 """
2526 mem_total_kb = self.read_from_meminfo('MemTotal')
2527 kb_in_gb = float(2 ** 20)
2528 return int(round(mem_total_kb / kb_in_gb))
2529
2530
2531 def get_disk_size_gb(self):
2532 """Get size of disk in GB (10^9)
2533
2534 @returns an integer representing size of disk, 0 in Error Case
2535 """
2536 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2537 result = self.run(command, ignore_status=True)
2538 if result.exit_status != 0:
2539 return 0
2540 _, _, block, _ = re.split(r' +', result.stdout.strip())
2541 byte_per_block = 1024.0
2542 disk_kb_in_gb = 1e9
2543 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2544
2545
2546 def get_battery_size(self):
2547 """Get size of battery in Watt-hour via sysfs
2548
2549 This method assumes that battery support voltage_min_design and
2550 charge_full_design sysfs.
2551
2552 @returns a float representing Battery size, 0 if error.
2553 """
2554 # sysfs report data in micro scale
2555 battery_scale = 1e6
2556
2557 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2558 result = self.run(command, ignore_status=True)
2559 if result.exit_status != 0:
2560 return 0
2561 voltage = float(result.stdout.strip()) / battery_scale
2562
2563 command = 'cat /sys/class/power_supply/*/charge_full_design'
2564 result = self.run(command, ignore_status=True)
2565 if result.exit_status != 0:
2566 return 0
2567 amphereHour = float(result.stdout.strip()) / battery_scale
2568
2569 return voltage * amphereHour
2570
2571
2572 def get_low_battery_shutdown_percent(self):
2573 """Get the percent-based low-battery shutdown threshold.
2574
2575 @returns a float representing low-battery shutdown percent, 0 if error.
2576 """
2577 ret = 0.0
2578 try:
2579 command = 'check_powerd_config --low_battery_shutdown_percent'
2580 ret = float(self.run(command).stdout)
2581 except error.CmdError:
2582 logging.debug("Can't run %s", command)
2583 except ValueError:
2584 logging.debug("Didn't get number from %s", command)
2585
2586 return ret
2587
2588
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002589 def has_hammer(self):
2590 """Check whether DUT has hammer device or not.
2591
2592 @returns boolean whether device has hammer or not
2593 """
2594 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2595 return self.run(command, ignore_status=True).exit_status == 0
2596
2597
Niranjan Kumar34618872017-05-31 12:57:09 -07002598 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002599 """Returns True if the specified switch was provided to Chrome.
2600
2601 @param switch The chrome switch to search for.
2602 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002603
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002604 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2605 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002606
2607
2608 def oobe_triggers_update(self):
2609 """Returns True if this host has an OOBE flow during which
2610 it will perform an update check and perhaps an update.
2611 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2612 As more such flows are developed, code handling them needs
2613 to be added here.
2614
2615 @return Boolean indicating whether this host's OOBE triggers an update.
2616 """
2617 return self.is_chrome_switch_present(
2618 '--enterprise-enable-zero-touch-enrollment=hands-off')
2619
2620
Kevin Chenga2619dc2016-03-28 11:42:08 -07002621 # TODO(kevcheng): change this to just return the board without the
2622 # 'board:' prefix and fix up all the callers. Also look into removing the
2623 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002624 def get_board(self):
2625 """Determine the correct board label for this host.
2626
2627 @returns a string representing this host's board.
2628 """
2629 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2630 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002631 return (ds_constants.BOARD_PREFIX +
2632 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002633
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002634 def get_channel(self):
2635 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002636
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002637 @returns: a string represeting this host's build channel.
2638 (stable, dev, beta). None on fail.
2639 """
2640 return lsbrelease_utils.get_chromeos_channel(
2641 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002642
Kevin Chenga328da62016-03-31 10:49:04 -07002643 def get_power_supply(self):
2644 """
2645 Determine what type of power supply the host has
2646
2647 @returns a string representing this host's power supply.
2648 'power:battery' when the device has a battery intended for
2649 extended use
2650 'power:AC_primary' when the device has a battery not intended
2651 for extended use (for moving the machine, etc)
2652 'power:AC_only' when the device has no battery at all.
2653 """
2654 psu = self.run(command='mosys psu type', ignore_status=True)
2655 if psu.exit_status:
2656 # The psu command for mosys is not included for all platforms. The
2657 # assumption is that the device will have a battery if the command
2658 # is not found.
2659 return 'power:battery'
2660
2661 psu_str = psu.stdout.strip()
2662 if psu_str == 'unknown':
2663 return None
2664
2665 return 'power:%s' % psu_str
2666
2667
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002668 def has_battery(self):
2669 """Determine if DUT has a battery.
2670
2671 Returns:
2672 Boolean, False if known not to have battery, True otherwise.
2673 """
2674 rv = True
2675 power_supply = self.get_power_supply()
2676 if power_supply == 'power:battery':
2677 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2678 board_type = self.get_board_type()
2679 if board_type in _NO_BATTERY_BOARD_TYPE:
2680 logging.warn('Do NOT believe type %s has battery. '
2681 'See debug for mosys details', board_type)
Sam Hurst57fa60a2020-05-08 08:55:47 -07002682 psu = utils.system_output('mosys -vvvv psu type',
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002683 ignore_status=True)
2684 logging.debug(psu)
2685 rv = False
2686 elif power_supply == 'power:AC_only':
2687 rv = False
2688
2689 return rv
2690
2691
Kevin Chenga328da62016-03-31 10:49:04 -07002692 def get_servo(self):
2693 """Determine if the host has a servo attached.
2694
2695 If the host has a working servo attached, it should have a servo label.
2696
2697 @return: string 'servo' if the host has servo attached. Otherwise,
2698 returns None.
2699 """
2700 return 'servo' if self._servo_host else None
2701
2702
Kevin Chenga328da62016-03-31 10:49:04 -07002703 def has_internal_display(self):
2704 """Determine if the device under test is equipped with an internal
2705 display.
2706
2707 @return: 'internal_display' if one is present; None otherwise.
2708 """
2709 from autotest_lib.client.cros.graphics import graphics_utils
2710 from autotest_lib.client.common_lib import utils as common_utils
2711
2712 def __system_output(cmd):
2713 return self.run(cmd).stdout
2714
2715 def __read_file(remote_path):
2716 return self.run('cat %s' % remote_path).stdout
2717
2718 # Hijack the necessary client functions so that we can take advantage
2719 # of the client lib here.
2720 # FIXME: find a less hacky way than this
2721 original_system_output = utils.system_output
2722 original_read_file = common_utils.read_file
2723 utils.system_output = __system_output
2724 common_utils.read_file = __read_file
2725 try:
2726 return ('internal_display' if graphics_utils.has_internal_display()
2727 else None)
2728 finally:
2729 utils.system_output = original_system_output
2730 common_utils.read_file = original_read_file
2731
2732
Dan Shi85276d42014-04-08 22:11:45 -07002733 def is_boot_from_usb(self):
2734 """Check if DUT is boot from USB.
2735
2736 @return: True if DUT is boot from usb.
2737 """
2738 device = self.run('rootdev -s -d').stdout.strip()
2739 removable = int(self.run('cat /sys/block/%s/removable' %
2740 os.path.basename(device)).stdout.strip())
2741 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002742
Otabek Kasimov77a40332020-10-20 15:40:03 -07002743 def is_boot_from_external_device(self):
2744 """Check if DUT is boot from external storage.
2745
2746 @return: True if DUT is boot from external storage.
2747 """
2748 boot_device = self.run('rootdev -s -d', ignore_status=True,
2749 timeout=60).stdout.strip()
2750 if not boot_device:
2751 logging.debug('Boot storage not detected on the host.')
2752 return False
2753 main_storage_cmd = ('. /usr/sbin/write_gpt.sh;'
2754 ' . /usr/share/misc/chromeos-common.sh;'
2755 ' load_base_vars; get_fixed_dst_drive')
2756 main_storage = self.run(main_storage_cmd,
2757 ignore_status=True,
2758 timeout=60).stdout.strip()
Otabek Kasimov723e8562020-12-08 13:29:34 -08002759 if not main_storage or boot_device != main_storage:
2760 logging.debug('Device booted from external storage storage.')
2761 return True
2762 logging.debug('Device booted from main storage.')
2763 return False
Helen Zhang17dae2b2014-11-11 09:25:52 -08002764
2765 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002766 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002767
2768 @param key: meminfo requested
2769
2770 @return the memory value as a string
2771
2772 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002773 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2774 logging.debug('%s', meminfo)
2775 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002776
2777
Rohit Makasana98e696f2016-06-03 18:48:10 -07002778 def get_cpu_arch(self):
2779 """Returns CPU arch of the device.
2780
2781 @return CPU architecture of the DUT.
2782 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002783 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002784 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2785 ignore_status=True).stdout:
2786 return 'x86_64'
2787 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2788 ignore_status=True).stdout:
2789 return 'arm'
2790 return 'i386'
2791
2792
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002793 def get_board_type(self):
2794 """
2795 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002796 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2797
2798 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002799 """
Danny Chan471a8d12015-08-18 14:57:41 -07002800 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2801 ignore_status=True).stdout
2802 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002803 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002804 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002805
2806
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002807 def get_arc_version(self):
2808 """Return ARC version installed on the DUT.
2809
2810 @returns ARC version as string if the CrOS build has ARC, else None.
2811 """
2812 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2813 ignore_status=True).stdout
2814 if arc_version:
2815 return arc_version.split('=')[-1].strip()
2816 return None
2817
2818
Gilad Arnolda76bef02015-09-29 13:55:15 -07002819 def get_os_type(self):
2820 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002821
2822
Kevin Chenga2619dc2016-03-28 11:42:08 -07002823 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002824 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002825 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002826
2827
2828 def get_default_power_method(self):
2829 """
2830 Get the default power method for power_on/off/cycle() methods.
2831 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2832 """
2833 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002834 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002835 if self.servo and self.servo.supports_built_in_pd_control():
2836 self._default_power_method = self.POWER_CONTROL_CCD
2837 else:
2838 logging.debug('Either servo is unitialized or the servo '
2839 'setup does not support pd controls. Falling '
2840 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002841 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002842
2843
2844 def find_usb_devices(self, idVendor, idProduct):
2845 """
2846 Get usb device sysfs name for specific device.
2847
2848 @param idVendor Vendor ID to search in sysfs directory.
2849 @param idProduct Product ID to search in sysfs directory.
2850
2851 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2852 """
2853 # Look for matching file and cut at position 7 to get dir name.
2854 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2855
2856 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2857 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2858
2859 # Use uniq -d to print duplicate line from both command
2860 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2861
2862 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2863
2864
2865 def bind_usb_device(self, usb_node):
2866 """
2867 Bind usb device
2868
2869 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2870 """
2871 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2872 self.run(cmd, ignore_status=True)
2873
2874
2875 def unbind_usb_device(self, usb_node):
2876 """
2877 Unbind usb device
2878
2879 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2880 """
2881 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2882 self.run(cmd, ignore_status=True)
2883
2884
2885 def get_wlan_ip(self):
2886 """
2887 Get ip address of wlan interface.
2888
2889 @return ip address of wlan or empty string if wlan is not connected.
2890 """
2891 cmds = [
2892 'iw dev', # List wlan physical device
2893 'grep Interface', # Grep only interface name
2894 'cut -f 2 -d" "', # Cut the name part
2895 'xargs ifconfig', # Feed it to ifconfig to get ip
2896 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2897 'cut -f 2 -d " "' # Cut the ip part
2898 ]
2899 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002900
2901 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2902 """
2903 Connect to wifi network
2904
2905 @param ssid SSID of the wifi network.
2906 @param passphrase Passphrase of the wifi network. None if not existed.
2907 @param security Security of the wifi network. Default to "psk" if
2908 passphase is given without security. Possible values
2909 are "none", "psk", "802_1x".
2910
2911 @return True if succeed, False if not.
2912 """
2913 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2914 if passphrase:
2915 cmd += ' ' + passphrase
2916 if security:
2917 cmd += ' ' + security
2918 return self.run(cmd, ignore_status=True).exit_status == 0
Otabek Kasimov6825b762020-06-23 23:42:44 -07002919
2920 def get_device_repair_state(self):
2921 """Get device repair state"""
2922 return self._device_repair_state
2923
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002924 def set_device_repair_state(self, state, resultdir=None):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002925 """Set device repair state.
2926
2927 The special device state will be written to the 'dut_state.repair'
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002928 file in result directory. The file will be read by Lucifer. The
2929 file will not be created if result directory not specified.
2930
2931 @params state: The new state for the device.
2932 @params resultdir: The path to result directory. If path not provided
2933 will be attempt to get retrieve it from job
2934 if present.
Otabek Kasimov6825b762020-06-23 23:42:44 -07002935 """
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002936 resultdir = resultdir or getattr(self.job, 'resultdir', '')
2937 if resultdir:
2938 target = os.path.join(resultdir, 'dut_state.repair')
Otabek Kasimov6825b762020-06-23 23:42:44 -07002939 common_utils.open_write_close(target, state)
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002940 logging.info('Set device state as %s. '
2941 'Created dut_state.repair file.', state)
Otabek Kasimov6825b762020-06-23 23:42:44 -07002942 else:
2943 logging.debug('Cannot write the device state due missing info '
2944 'about result dir.')
2945 self._device_repair_state = state
2946
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002947 def set_device_needs_replacement(self, resultdir=None):
2948 """Set device as required replacement.
2949
2950 @params resultdir: The path to result directory. If path not provided
2951 will be attempt to get retrieve it from job
2952 if present.
2953 """
2954 self.set_device_repair_state(
2955 cros_constants.DEVICE_STATE_NEEDS_REPLACEMENT,
2956 resultdir=resultdir)
2957
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002958 def _dut_is_accessible_by_verifier(self):
2959 """Check if DUT accessible by SSH or PING verifier.
Otabek Kasimov86062d02020-11-17 13:30:22 -08002960
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002961 @returns: bool, True - verifier marked as success.
2962 False - result not reachable, verifier did not success.
Otabek Kasimov86062d02020-11-17 13:30:22 -08002963 """
2964 if not self._repair_strategy:
2965 return False
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002966 dut_ssh = self._repair_strategy.verifier_is_good('ssh')
2967 dut_ping = self._repair_strategy.verifier_is_good('ping')
2968 return dut_ssh == hosts.VERIFY_SUCCESS or dut_ssh == hosts.VERIFY_SUCCESS
Otabek Kasimov86062d02020-11-17 13:30:22 -08002969
Otabek Kasimovd48389b2020-12-07 02:38:34 -08002970 def _stat_if_pingable_but_not_sshable(self):
2971 """Check if DUT pingable but failed SSH verifier."""
2972 if not self._repair_strategy:
2973 return
2974 dut_ssh = self._repair_strategy.verifier_is_good('ssh')
2975 dut_ping = self._repair_strategy.verifier_is_good('ping')
2976 if (dut_ping == hosts.VERIFY_FAILED
2977 and dut_ssh == hosts.VERIFY_FAILED):
2978 metrics.Counter('chromeos/autotest/dut_pingable_no_ssh').increment(
2979 fields={'host': self.hostname})
2980
Otabek Kasimov7cad9ce2020-07-28 14:58:48 -07002981 def try_set_device_needs_manual_repair(self):
Otabek Kasimov6825b762020-06-23 23:42:44 -07002982 """Check if device require manual attention to be fixed.
2983
2984 The state 'needs_manual_repair' can be set when auto repair cannot
2985 fix the device due hardware or cable issues.
2986 """
2987 # ignore the logic if state present
2988 # state can be set by any cros repair actions
Otabek Kasimov86062d02020-11-17 13:30:22 -08002989 if self.get_device_repair_state():
Otabek Kasimov6825b762020-06-23 23:42:44 -07002990 return
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08002991 if self._dut_is_accessible_by_verifier():
2992 # DUT is accessible and we still have many options to repair it.
Otabek Kasimovc9812582020-10-08 18:52:52 -07002993 return
Otabek Kasimov9189ede2020-11-09 14:08:58 -08002994 needs_manual_repair = False
2995 dhp = self.health_profile
Otabek Kasimov69253822020-11-24 10:52:27 -08002996 if dhp and dhp.get_repair_fail_count() > 49:
2997 # 42 = 6 times during 7 days. (every 4 hour repair)
2998 # round up to 50 in case somebody will run some attempt on it.
Otabek Kasimovc9812582020-10-08 18:52:52 -07002999 logging.info(
Otabek Kasimov9189ede2020-11-09 14:08:58 -08003000 'DUT is not sshable and fail %s times.'
Otabek Kasimov69253822020-11-24 10:52:27 -08003001 ' Limit to try repair is 50 times',
Otabek Kasimov9189ede2020-11-09 14:08:58 -08003002 dhp.get_repair_fail_count())
3003 needs_manual_repair = True
3004
3005 if not needs_manual_repair:
3006 # We cannot ssh to the DUT and we have hardware or set-up issues
3007 # with servo then we need request manual repair for the DUT.
3008 servo_state_required_manual_fix = [
3009 servo_constants.SERVO_STATE_DUT_NOT_CONNECTED,
3010 servo_constants.SERVO_STATE_NEED_REPLACEMENT,
3011 ]
3012 if self.get_servo_state() in servo_state_required_manual_fix:
3013 logging.info(
3014 'DUT required manual repair because it is not sshable'
3015 ' and possible have setup issue with Servo. Please'
3016 ' verify all connections and present of devices.')
3017 needs_manual_repair = True
3018
3019 if needs_manual_repair:
Otabek Kasimovc9812582020-10-08 18:52:52 -07003020 self.set_device_repair_state(
3021 cros_constants.DEVICE_STATE_NEEDS_MANUAL_REPAIR)
Otabek Kasimov42506d02020-07-29 14:44:57 -07003022
Otabek Kasimov86062d02020-11-17 13:30:22 -08003023 def _reboot_labstation_if_needed(self):
3024 """Place request to reboot the labstation if DUT is not sshable.
3025
3026 @returns: None
3027 """
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003028 message_prefix = "Don't need to request servo-host reboot"
Otabek Kasimovc7fa5072021-01-20 21:24:18 -08003029 if self._dut_is_accessible_by_verifier():
Otabek Kasimov86062d02020-11-17 13:30:22 -08003030 return
3031 if not self._servo_host:
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003032 logging.debug('%s as it not initialized', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003033 return
3034 if not self._servo_host.is_up_fast():
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003035 logging.debug('%s as servo-host is not sshable', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003036 return
3037 if not self._servo_host.is_labstation():
3038 logging.debug('Servo_v3 is not requested to reboot for the DUT')
3039 return
3040 usb_path = self._servo_host.get_main_servo_usb_path()
3041 if usb_path:
3042 connected_port = os.path.basename(os.path.normpath(usb_path))
3043 # Directly connected servo to the labstation looks like '1-5.3'
3044 # and when connected by hub - '1-5.2.3' or '1-5.2.1.3'. Where:
3045 # - '1-5' - port on labstation
3046 # - '2' or '2.1' - port on the hub or smart-hub
3047 # - '3' - port on servo hub
3048 if len(connected_port.split('.')) > 2:
Puthikorn Voravootivat58f739b2021-01-07 08:28:36 -08003049 logging.debug('%s as servo connected by hub', message_prefix)
Otabek Kasimov86062d02020-11-17 13:30:22 -08003050 return
3051 self._servo_host.request_reboot()
3052 logging.info('Requested labstation reboot because DUT is not sshable')
3053
Otabek Kasimov42506d02020-07-29 14:44:57 -07003054 def is_file_system_writable(self, testdirs=None):
3055 """Check is the file systems are writable.
3056
3057 The standard linux response to certain unexpected file system errors
3058 (including hardware errors in block devices) is to change the file
3059 system status to read-only. This checks that that hasn't happened.
3060
3061 @param testdirs: List of directories to check. If no data provided
3062 then '/mnt/stateful_partition' and '/var/tmp'
3063 directories will be checked.
3064
3065 @returns boolean whether file-system writable.
3066 """
3067 def _check_dir(testdir):
3068 # check if we can create a file
3069 filename = os.path.join(testdir, 'writable_my_test_file')
3070 command = 'touch %s && rm %s' % (filename, filename)
3071 rv = self.run(command=command,
3072 timeout=30,
3073 ignore_status=True)
3074 is_writable = rv.exit_status == 0
3075 if not is_writable:
3076 logging.info('Cannot create a file in "%s"!'
3077 ' Probably the FS is read-only', testdir)
3078 logging.info("FileSystem is not writable!")
3079 return False
3080 return True
3081
3082 if not testdirs or len(testdirs) == 0:
3083 # N.B. Order matters here: Encrypted stateful is loop-mounted
3084 # from a file in unencrypted stateful, so we don't test for
3085 # errors in encrypted stateful if unencrypted fails.
3086 testdirs = ['/mnt/stateful_partition', '/var/tmp']
3087
3088 for dir in testdirs:
3089 # loop will be stopped if any directory fill fail the check
3090 try:
3091 if not _check_dir(dir):
3092 return False
3093 except Exception as e:
3094 # here expected only timeout error, all other will
3095 # be catch by 'ignore_status=True'
3096 logging.debug('Fail to check %s to write in it', dir)
3097 return False
3098 return True
Garry Wang1a493d82020-08-31 21:01:19 -07003099
Dana Goyettec172b172020-07-29 16:26:15 -07003100 def blocking_sync(self, freeze_for_reset=False):
3101 """Sync root device and internal device, via script.
3102
3103 The actual calls end up logged by the run() call, since they're printed
3104 to stdout/stderr in the script.
3105
3106 @param freeze_for_reset: if True, prepare for reset by blocking writes
3107 (only if enable_fs_sync_fsfreeze=True)
3108 """
3109
3110 if freeze_for_reset and self.USE_FSFREEZE:
3111 logging.info('Blocking sync and freeze')
3112 elif freeze_for_reset:
3113 logging.info('Blocking sync for reset')
3114 else:
3115 logging.info('Blocking sync')
3116
3117 # client/bin is installed on the DUT as /usr/local/autotest/bin
3118 sync_cmd = '/usr/local/autotest/bin/fs_sync.py'
3119 if freeze_for_reset and self.USE_FSFREEZE:
3120 sync_cmd += ' --freeze'
3121 return self.run(sync_cmd)
3122
Garry Wanga2e78172020-09-09 23:49:07 -07003123 def set_health_profile_dut_state(self, state):
3124 if not self.health_profile:
3125 logging.debug('Device health profile is not initialized, skip'
3126 ' set dut state.')
3127 return
3128 reset_counters = state in profile_constants.STATES_NEED_RESET_COUNTER
3129 self.health_profile.update_dut_state(state, reset_counters)
Garry Wang53fc8f32020-09-18 13:30:08 -07003130
3131 def require_snk_mode_in_recovery(self):
3132 """Check whether we need to switch servo_v4 role to snk when
3133 booting into recovery mode. (See crbug.com/1129165)
3134 """
Garry Wanga8739cc2020-10-30 00:49:23 -07003135 has_battery = True
3136 # Determine if the host has battery based on host_info first.
3137 power_info = self.host_info_store.get().get_label_value('power')
3138 if power_info:
3139 has_battery = power_info == 'battery'
3140 elif self.is_up_fast():
3141 # when running local tests host_info is not available, so we
3142 # need to determine whether the host has battery by checking
3143 # from host side.
3144 logging.debug('Label `power` is not found in host_info, checking'
3145 ' if the host has battery from host side.')
3146 has_battery = self.has_battery()
3147
3148 if not has_battery:
Garry Wang53fc8f32020-09-18 13:30:08 -07003149 logging.info(
3150 '%s does not has battery, snk mode is not needed'
3151 ' for recovery.', self.hostname)
3152 return False
Garry Wanga8739cc2020-10-30 00:49:23 -07003153
Garry Wang53fc8f32020-09-18 13:30:08 -07003154 if not self.servo.supports_built_in_pd_control():
3155 logging.info('Power delivery is not supported on this servo, snk'
3156 ' mode is not needed for recovery.')
3157 return False
3158 try:
Garry Wang53fc8f32020-09-18 13:30:08 -07003159 battery_percent = self.servo.get('battery_charge_percent')
Otabek Kasimov58e22562020-11-03 17:17:41 -08003160 if battery_percent < cros_constants.MIN_BATTERY_LEVEL:
Garry Wang53fc8f32020-09-18 13:30:08 -07003161 logging.info(
3162 'Current battery level %s%% below %s%% threshold, we'
3163 ' will attempt to boot host in recovery mode without'
3164 ' changing servo to snk mode. Please note the host may'
3165 ' not able to see usb drive in recovery mode later due'
3166 ' to servo not in snk mode.', battery_percent,
Otabek Kasimov58e22562020-11-03 17:17:41 -08003167 cros_constants.MIN_BATTERY_LEVEL)
Garry Wang53fc8f32020-09-18 13:30:08 -07003168 return False
3169 except Exception as e:
3170 logging.info(
3171 'Unexpected error occurred when getting'
3172 ' battery_charge_percent from servo; %s', str(e))
3173 return False
3174 return True
Otabek Kasimov382c3bb2020-10-28 13:22:45 -07003175
3176 def _set_servo_topology(self):
3177 """Set servo-topology info to the host-info."""
3178 logging.debug('Try to save servo topology to host-info.')
3179 if not self._servo_host:
3180 logging.info('Servo host is not initilized.')
3181 return
3182 if not self._servo_host.is_servo_topology_supported():
3183 logging.info('Servo-topology is not supported.')
3184 return
3185 servo_topology = self._servo_host.get_topology()
3186 if not servo_topology or servo_topology.is_empty():
3187 logging.info('Servo topology is empty')
3188 return
3189 servo_topology.save(self.host_info_store)