blob: 63ecbc3a03711193966851f6548b80dace43de9a [file] [log] [blame]
J. Richard Barnette24adbf42012-04-11 15:04:53 -07001# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
Dale Curtisaa5eedb2011-08-23 16:18:52 -07002# Use of this source code is governed by a BSD-style license that can be
3# found in the LICENSE file.
4
J. Richard Barnette1d78b012012-05-15 13:56:30 -07005import logging
Dan Shi0f466e82013-02-22 15:44:58 -08006import os
Simran Basid5e5e272012-09-24 15:23:59 -07007import re
Vincent Palatindf2372c2016-10-07 17:03:00 +02008import sys
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07009import time
10
mussa584b4462014-06-20 15:13:28 -070011import common
J. Richard Barnette45e93de2012-04-11 17:24:15 -070012from autotest_lib.client.bin import utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070013from autotest_lib.client.common_lib import autotemp
Richard Barnette0c73ffc2012-11-19 15:21:18 -080014from autotest_lib.client.common_lib import error
15from autotest_lib.client.common_lib import global_config
J. Richard Barnette91137f02016-03-10 16:52:26 -080016from autotest_lib.client.common_lib import hosts
Dan Shi549fb822015-03-24 18:01:11 -070017from autotest_lib.client.common_lib import lsbrelease_utils
Greg Edelstona7b05d12020-04-01 16:00:51 -060018from autotest_lib.client.common_lib.cros import cros_config
Richard Barnette03a0c132012-11-05 12:40:35 -080019from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070020from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000021from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080022from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080023from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070024from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070025from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050026from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070027from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070028from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070029from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070030from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080031from autotest_lib.server.hosts import chameleon_host
Richard Barnetted31580e2018-05-14 19:58:00 +000032from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080033from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070034from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070035from autotest_lib.server.hosts import servo_host
Garry Wang11b5e872020-03-11 15:14:08 -070036from autotest_lib.server.hosts import servo_constants
Simran Basidcff4252012-11-20 16:13:20 -080037from autotest_lib.site_utils.rpm_control_system import rpm_client
Simran Basid5e5e272012-09-24 15:23:59 -070038
Simran Basi382506b2016-09-13 14:58:15 -070039# In case cros_host is being ran via SSP on an older Moblab version with an
40# older chromite version.
41try:
42 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080043except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080044 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080045
Simran Basid5e5e272012-09-24 15:23:59 -070046
Dan Shib8540a52015-07-16 14:18:23 -070047CONFIG = global_config.global_config
48
Dan Shid07ee2e2015-09-24 14:49:25 -070049
beepsc87ff602013-07-31 21:53:00 -070050class FactoryImageCheckerException(error.AutoservError):
51 """Exception raised when an image is a factory image."""
52 pass
53
54
Fang Deng0ca40e22013-08-27 17:47:44 -070055class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070056 """Chromium OS specific subclass of Host."""
57
Simran Basi5ace6f22016-01-06 17:30:44 -080058 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
59
Scott Zawalski62bacae2013-03-05 10:40:32 -050060 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070061
Richard Barnette03a0c132012-11-05 12:40:35 -080062 # Timeout values (in seconds) associated with various Chrome OS
63 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070064 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080065 # In general, a good rule of thumb is that the timeout can be up
66 # to twice the typical measured value on the slowest platform.
67 # The times here have not necessarily been empirically tested to
68 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070069 #
70 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080071 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
72 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080073 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070074 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080075 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070076 # screen delay, time to start the network on the DUT, and the
77 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070078 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080079 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080080 # network.
beepsf079cfb2013-09-18 17:49:51 -070081 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080082 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
83 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070084
85 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080086 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080087 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070088 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070089 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070090 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080091 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070092
Dan Shica503482015-03-30 17:23:25 -070093 # Minimum OS version that supports server side packaging. Older builds may
94 # not have server side package built or with Autotest code change to support
95 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070096 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -070097 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -070098
J. Richard Barnette84890bd2014-02-21 11:05:47 -080099 # REBOOT_TIMEOUT: How long to wait for a reboot.
100 #
Chris Sosab76e0ee2013-05-22 16:55:41 -0700101 # We have a long timeout to ensure we don't flakily fail due to other
102 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700103 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
104 # return from reboot' bug is solved.
105 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700106
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800107 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
108 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700109 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
110 # since changing servo role will reset USB state
111 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800112 _USB_POWER_TIMEOUT = 5
113 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700114 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800115
Fang Dengdeba14f2014-11-14 11:54:09 -0800116 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
117 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700118
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800119 # Constants used in ping_wait_up() and ping_wait_down().
120 #
121 # _PING_WAIT_COUNT is the approximate number of polling
122 # cycles to use when waiting for a host state change.
123 #
124 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
125 # for arguments to the internal _ping_wait_for_status()
126 # method.
127 _PING_WAIT_COUNT = 40
128 _PING_STATUS_DOWN = False
129 _PING_STATUS_UP = True
130
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800131 # Allowed values for the power_method argument.
132
Garry Wang5e5538a2019-04-08 15:36:18 -0700133 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
134 # DUTs except those with servo_v4 CCD.
135 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
136 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800137 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
138 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
139 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700140 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800141 POWER_CONTROL_SERVO = 'servoj10'
142 POWER_CONTROL_MANUAL = 'manual'
143
144 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700145 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800146 POWER_CONTROL_SERVO,
147 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800148
Simran Basi5e6339a2013-03-21 11:34:32 -0700149 _RPM_OUTLET_CHANGED = 'outlet_changed'
150
Dan Shi9cb0eec2014-06-03 09:04:50 -0700151 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700152 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700153 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700154
Brent Peterson1cb623a2020-01-09 13:14:28 -0800155 # Regular expression for extracting EC version string
156 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
157
158 # Regular expression for extracting BIOS version string
159 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
160
Brent Petersonc70a1832020-01-24 15:54:35 -0800161 # Command to update firmware located on DUT
Brent Peterson669edf42020-02-07 15:07:54 -0800162 _FW_UPDATE_CMD = 'chromeos-firmwareupdate --mode=recovery -i %s %s'
Brent Petersonc70a1832020-01-24 15:54:35 -0800163
J. Richard Barnette964fba02012-10-24 17:34:29 -0700164 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800165 def check_host(host, timeout=10):
166 """
167 Check if the given host is a chrome-os host.
168
169 @param host: An ssh host representing a device.
170 @param timeout: The timeout for the run command.
171
172 @return: True if the host device is chromeos.
173
beeps46dadc92013-11-07 14:07:10 -0800174 """
175 try:
Allen Liad719c12017-06-27 23:48:04 +0000176 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700177 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700178 '! grep -q moblab /etc/lsb-release && '
179 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700180 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700181 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000182 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700183 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
184 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800185 return not (
186 lsbrelease_utils.is_jetstream(
187 lsb_release_content=lsb_release_content) or
188 lsbrelease_utils.is_gce_board(
189 lsb_release_content=lsb_release_content))
190
beeps46dadc92013-11-07 14:07:10 -0800191 except (error.AutoservRunError, error.AutoservSSHTimeout):
192 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700193
194 return False
beeps46dadc92013-11-07 14:07:10 -0800195
196
197 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800198 def get_chameleon_arguments(args_dict):
199 """Extract chameleon options from `args_dict` and return the result.
200
201 Recommended usage:
202 ~~~~~~~~
203 args_dict = utils.args_to_dict(args)
204 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
205 host = hosts.create_host(machine, chameleon_args=chameleon_args)
206 ~~~~~~~~
207
208 @param args_dict Dictionary from which to extract the chameleon
209 arguments.
210 """
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800211 return {key: args_dict[key]
Allen Li083866b2016-08-18 10:07:10 -0700212 for key in ('chameleon_host', 'chameleon_port')
213 if key in args_dict}
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800214
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800215 @staticmethod
216 def get_btpeer_arguments(args_dict):
217 """Extract btpeer options from `args_dict` and return the result.
218
219 This is used to parse details of Bluetooth peer.
220 Recommended usage:
221 ~~~~~~~~
222 args_dict = utils.args_to_dict(args)
223 btpeer_args = hosts.CrosHost.get_btpeer_arguments(args_dict)
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700224 host = hosts.create_host(machine, btpeer_args=btpeer_args)
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800225 ~~~~~~~~
226
227 @param args_dict: Dictionary from which to extract the btpeer
228 arguments.
229 """
230 if 'btpeer_host_list' in args_dict:
231 result = []
232 for btpeer in args_dict['btpeer_host_list'].split(','):
233 result.append({key: value for key,value in
234 zip(('btpeer_host','btpeer_port'),
235 btpeer.split(':'))})
236 return result
237 else:
238 return {key: args_dict[key]
239 for key in ('btpeer_host', 'btpeer_port')
240 if key in args_dict}
241
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800242
243 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700244 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800245 """Extract chameleon options from `args_dict` and return the result.
246
247 Recommended usage:
248 ~~~~~~~~
249 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700250 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
251 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800252 ~~~~~~~~
253
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700254 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800255 arguments.
256 """
Allen Li083866b2016-08-18 10:07:10 -0700257 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700258 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700259 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800260
261
262 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800263 def get_servo_arguments(args_dict):
264 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800265
266 Recommended usage:
267 ~~~~~~~~
268 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700269 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800270 host = hosts.create_host(machine, servo_args=servo_args)
271 ~~~~~~~~
272
273 @param args_dict Dictionary from which to extract the servo
274 arguments.
275 """
Garry Wang11b5e872020-03-11 15:14:08 -0700276 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
277 servo_constants.SERVO_PORT_ATTR,
278 servo_constants.SERVO_BOARD_ATTR,
279 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200280 servo_args = {key: args_dict[key]
281 for key in servo_attrs
282 if key in args_dict}
283 return (
284 None
Garry Wang11b5e872020-03-11 15:14:08 -0700285 if servo_constants.SERVO_HOST_ATTR in servo_args
286 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200287 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700288
J. Richard Barnette964fba02012-10-24 17:34:29 -0700289
J. Richard Barnette91137f02016-03-10 16:52:26 -0800290 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700291 pdtester_args=None, try_lab_servo=False,
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700292 try_servo_repair=False, btpeer_args=[],
J. Richard Barnette91137f02016-03-10 16:52:26 -0800293 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700294 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800295 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700296
Fang Denge545abb2014-12-30 18:43:47 -0800297 This method will attempt to create the test-assistant object
298 (chameleon/servo) when it is needed by the test. Check
299 the docstring of chameleon_host.create_chameleon_host and
300 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700301
Fang Denge545abb2014-12-30 18:43:47 -0800302 @param hostname: Hostname of the dut.
303 @param chameleon_args: A dictionary that contains args for creating
304 a ChameleonHost. See chameleon_host for details.
305 @param servo_args: A dictionary that contains args for creating
306 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700307 @param try_lab_servo: When true, indicates that an attempt should
308 be made to create a ServoHost for a DUT in
309 the test lab, even if not required by
310 `servo_args`. See servo_host for details.
311 @param try_servo_repair: If a servo host is created, check it
312 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800313 See servo_host for details.
314 @param ssh_verbosity_flag: String, to pass to the ssh command to control
315 verbosity.
316 @param ssh_options: String, other ssh options to pass to the ssh
317 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700318 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700319 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700320 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800321 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Kevin Chenga2619dc2016-03-28 11:42:08 -0700322 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700323 # self.env is a dictionary of environment variable settings
324 # to be exported for commands run on the host.
325 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
326 # errors that might happen.
327 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700328 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700329 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700330 _servo_host, servo_state = servo_host.create_servo_host(
331 dut=self,
332 servo_args=servo_args,
333 try_lab_servo=try_lab_servo,
334 try_servo_repair=try_servo_repair,
335 dut_host_info=self.host_info_store.get())
336 self.set_servo_host(_servo_host, servo_state)
Garry Wang5e5538a2019-04-08 15:36:18 -0700337 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700338
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800339 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800340 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700341 dut=self.hostname,
342 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800343 if self._chameleon_host:
344 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800345 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800346 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700347
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700348 # Initialize Bluetooth peers.
349 try:
350 self.initialize_btpeer(btpeer_args)
351 except Exception as e:
352 logging.error('Exception %s in initialize_btpeer', str(e))
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800353
howardchung83e55272019-08-08 14:08:05 +0800354 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700355 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700356 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800357
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700358 if self._pdtester_host:
359 self.pdtester_servo = self._pdtester_host.get_servo()
360 logging.info('pdtester_servo: %r', self.pdtester_servo)
361 # Create the pdtester object used to access the ec uart
362 self.pdtester = pdtester.PDTester(self.pdtester_servo,
363 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800364 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700365 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800366
Fang Deng5d518f42013-08-02 14:04:32 -0700367
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800368 def initialize_btpeer(self, btpeer_args):
369 """ Initialize the Bluetooth peers
370
371 Initialize Bluetooth peer devices given in the arguments. Bluetooth peer
372 is chameleon host on Raspberry Pi.
373 @param btpeer_args: A dictionary that contains args for creating
374 a ChameleonHost. See chameleon_host for details.
375
376 """
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700377 #TODO (b:142486063) Remove the try..except
378 try:
379 self._btpeer_host_list = []
380 self.btpeer_list = []
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800381 self.btpeer = None
382
Shijin Abraham22f24cb2020-03-26 09:07:41 -0700383 if type(btpeer_args) is list:
384 btpeer_args_list = btpeer_args
385 else:
386 btpeer_args_list = [btpeer_args]
387
388 self._btpeer_host_list = chameleon_host.create_btpeer_host(
389 dut=self.hostname, btpeer_args_list=btpeer_args_list)
390 logging.debug('Bluetooth peer hosts are %s',
391 self._btpeer_host_list)
392 self.btpeer_list = [_host.create_chameleon_board() for _host in
393 self._btpeer_host_list if _host is not None]
394
395 if len(self.btpeer_list) > 0:
396 self.btpeer = self.btpeer_list[0]
397
398 logging.debug('After initialize_btpeer btpeer_list %s '
399 'btpeer_host_list is %s and btpeer is %s',
400 self.btpeer_list, self._btpeer_host_list,
401 self.btpeer)
402 except Exception as e:
403 logging.error('Exception %s in initialize_btpeer', str(e))
404
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800405
406
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000407 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700408 """Get latest stable cros image name from AFE.
409
410 Use the board name from the info store. Should that fail, try to
411 retrieve the board name from the host's installed image itself.
412
413 @returns: current stable cros image name for this host.
414 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700415 info = self.host_info_store.get()
416 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700417 logging.warn('No board label value found. Trying to infer '
418 'from the host itself.')
419 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700420 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700421 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
422 logging.error('Also failed to get the board name from the DUT '
423 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700424 raise error.AutoservError('Cannot determine board of the DUT'
425 ' while getting repair image name.')
426 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500427
428
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700429 def host_version_prefix(self, image):
430 """Return version label prefix.
431
432 In case the CrOS provisioning version is something other than the
433 standard CrOS version e.g. CrOS TH version, this function will
434 find the prefix from provision.py.
435
436 @param image: The image name to find its version prefix.
437 @returns: A prefix string for the image type.
438 """
439 return provision.get_version_label_prefix(image)
440
Andrew Luo3332ab22020-04-28 16:42:03 -0700441 def stage_build_to_usb(self, build):
442 """Stage the current ChromeOS image on the USB stick connected to the
443 servo.
444
445 @param build: The build to download and send to USB.
446 """
447 if not self.servo:
448 raise error.TestError('Host %s does not have servo.' %
449 self.hostname)
450
451 _, update_url = self.stage_image_for_servo(build)
452 self.servo.image_to_servo_usb(update_url)
453 logging.debug('ChromeOS image %s is staged on the USB stick.',
454 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700455
beepsdae65fd2013-07-26 16:24:41 -0700456 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700457 """
458 Make sure job_repo_url of this host is valid.
459
joychen03eaad92013-06-26 09:55:21 -0700460 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700461 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
462 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
463 download and extract it. If the devserver embedded in the url is
464 unresponsive, update the job_repo_url of the host after staging it on
465 another devserver.
466
467 @param job_repo_url: A url pointing to the devserver where the autotest
468 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700469 @param tag: The tag from the server job, in the format
470 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700471
472 @raises DevServerException: If we could not resolve a devserver.
473 @raises AutoservError: If we're unable to save the new job_repo_url as
474 a result of choosing a new devserver because the old one failed to
475 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700476 @raises urllib2.URLError: If the devserver embedded in job_repo_url
477 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700478 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800479 info = self.host_info_store.get()
480 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700481 if not job_repo_url:
482 logging.warning('No job repo url set on host %s', self.hostname)
483 return
484
485 logging.info('Verifying job repo url %s', job_repo_url)
486 devserver_url, image_name = tools.get_devserver_build_from_package_url(
487 job_repo_url)
488
beeps0c865032013-07-30 11:37:06 -0700489 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700490
491 logging.info('Staging autotest artifacts for %s on devserver %s',
492 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700493
494 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700495 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700496 stage_time = time.time() - start_time
497
498 # Record how much of the verification time comes from a devserver
499 # restage. If we're doing things right we should not see multiple
500 # devservers for a given board/build/branch path.
501 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800502 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700503 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800504 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700505 pass
506 else:
beeps0c865032013-07-30 11:37:06 -0700507 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700508 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700509 stats_key = {
510 'board': board,
511 'build_type': build_type,
512 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700513 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700514 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800515
516 monarch_fields = {
517 'board': board,
518 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800519 'branch': branch,
520 'dev_server': devserver,
521 }
522 metrics.Counter(
523 'chromeos/autotest/provision/verify_url'
524 ).increment(fields=monarch_fields)
525 metrics.SecondsDistribution(
526 'chromeos/autotest/provision/verify_url_duration'
527 ).add(stage_time, fields=monarch_fields)
528
529
Dan Shicf4d2032015-03-12 15:04:21 -0700530 def stage_server_side_package(self, image=None):
531 """Stage autotest server-side package on devserver.
532
533 @param image: Full path of an OS image to install or a build name.
534
535 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700536
537 @raise: error.AutoservError if fail to locate the build to test with, or
538 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700539 """
Dan Shid37736b2016-07-06 15:10:29 -0700540 # If enable_drone_in_restricted_subnet is False, do not set hostname
541 # in devserver.resolve call, so a devserver in non-restricted subnet
542 # is picked to stage autotest server package for drone to download.
543 hostname = self.hostname
544 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
545 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700546 if image:
547 image_name = tools.get_build_from_image(image)
548 if not image_name:
549 raise error.AutoservError(
550 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700551 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700552 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700553 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800554 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700555 if job_repo_url:
556 devserver_url, image_name = (
557 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700558 # If enable_drone_in_restricted_subnet is True, use the
559 # existing devserver. Otherwise, resolve a new one in
560 # non-restricted subnet.
561 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
562 ds = dev_server.ImageServer(devserver_url)
563 else:
564 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800565 elif info.build is not None:
566 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700567 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700568 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800569 raise error.AutoservError(
570 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700571 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700572
573 # Get the OS version of the build, for any build older than
574 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
575 match = re.match('.*/R\d+-(\d+)\.', image_name)
576 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700577 raise error.AutoservError(
578 'Build %s is older than %s. Server side packaging is '
579 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700580
Dan Shicf4d2032015-03-12 15:04:21 -0700581 ds.stage_artifacts(image_name, ['autotest_server_package'])
582 return '%s/static/%s/%s' % (ds.url(), image_name,
583 'autotest_server_package.tar.bz2')
584
585
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700586 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700587 """Stage a build on a devserver and return the update_url.
588
589 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700590 @param artifact: a string like 'test_image'. Requests
591 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700592 @returns a tuple of (image_name, URL) like
593 (lumpy-release/R27-3837.0.0,
594 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700595 """
596 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000597 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700598 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800599 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700600 devserver.stage_artifacts(image_name, [artifact])
601 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700602 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700603 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700604 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700605 else:
606 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700607
608
beepse539be02013-07-31 21:57:39 -0700609 def stage_factory_image_for_servo(self, image_name):
610 """Stage a build on a devserver and return the update_url.
611
612 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700613
beepse539be02013-07-31 21:57:39 -0700614 @return: An update URL, eg:
615 http://<devserver>/static/canary-channel/\
616 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700617
618 @raises: ValueError if the factory artifact name is missing from
619 the config.
620
beepse539be02013-07-31 21:57:39 -0700621 """
622 if not image_name:
623 logging.error('Need an image_name to stage a factory image.')
624 return
625
Dan Shib8540a52015-07-16 14:18:23 -0700626 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700627 'CROS', 'factory_artifact', type=str, default='')
628 if not factory_artifact:
629 raise ValueError('Cannot retrieve the factory artifact name from '
630 'autotest config, and hence cannot stage factory '
631 'artifacts.')
632
beepse539be02013-07-31 21:57:39 -0700633 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800634 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700635 devserver.stage_artifacts(
636 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700637 [factory_artifact],
638 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700639
640 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
641
642
Laurence Goodby778c9a42017-05-24 19:24:07 -0700643 def prepare_for_update(self):
644 """Prepares the DUT for an update.
645
646 Subclasses may override this to perform any special actions
647 required before updating.
648 """
Laurence Goodby468de252017-06-08 17:22:53 -0700649 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700650
651
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800652 def _clear_fw_version_labels(self, rw_only):
653 """Clear firmware version labels from the machine.
654
655 @param rw_only: True to only clear fwrw_version; otherewise, clear
656 both fwro_version and fwrw_version.
657 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700658 info = self.host_info_store.get()
659 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800660 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700661 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
662 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700663
664
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800665 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700666 """Add firmware version label to the machine.
667
668 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800669 @param rw_only: True to only add fwrw_version; otherwise, add both
670 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700671
672 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700673 info = self.host_info_store.get()
674 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800675 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700676 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
677 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700678
679
Namyoon Woo33f38852020-04-13 17:26:58 -0700680 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800681 """Search for the latest package release version from the image archive,
682 and return it.
683
Namyoon Woo33f38852020-04-13 17:26:58 -0700684 @param platform: platform name, a.k.a. board or model
685 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800686
Namyoon Woo33f38852020-04-13 17:26:58 -0700687 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
688 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800689 or None if LATEST release file does not exist.
690 """
691
Namyoon Woo33f38852020-04-13 17:26:58 -0700692 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800693
Namyoon Woo33f38852020-04-13 17:26:58 -0700694 # Search the image path in reference board archive as well.
695 # For example, bob has its binary image under its reference board (gru)
696 # image archive.
697 if ref_board:
698 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800699
Namyoon Woo33f38852020-04-13 17:26:58 -0700700 for board in platforms:
701 # Read 'LATEST-1.0.0' file
702 branch_dir = provision.FW_BRANCH_GLOB % board
703 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
704 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800705
Namyoon Woo33f38852020-04-13 17:26:58 -0700706 try:
707 # The result could be one or more.
708 result = utils.system_output('gsutil ls -d ' + latest_file)
709
710 candidates = re.findall('gs://.*', result)
711
712 # Found the directory candidates. No need to check the other
713 # board name cadidates. Let's break the loop.
714 break
715 except error.CmdError:
716 # It doesn't exist. Let's move on to the next item.
717 pass
718 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800719 logging.error('No LATEST release info is available.')
720 return None
721
Namyoon Woo406c7d42020-01-24 15:57:11 -0800722 for cand_dir in candidates:
723 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800724
Namyoon Woo406c7d42020-01-24 15:57:11 -0800725 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700726 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800727 try:
728 # Check if release_path does exist.
729 release = utils.system_output('gsutil ls -d ' + release_path)
730 # Now 'release' has a full directory path: e.g.
731 # gs://chromeos-image-archive/firmware-octopus-11297.B-
732 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
733
734 # Remove "gs://chromeos-image-archive".
735 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
736
737 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
738 return release.strip('/')
739 except error.CmdError:
740 # The directory might not exist. Let's try next candidate.
741 pass
742 else:
743 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800744
Brent Peterson1cb623a2020-01-09 13:14:28 -0800745 @staticmethod
746 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800747 """Get version string from binary image using regular expression.
748
749 @param image: Binary image to search
750 @param version_regex: Regular expression to search for
751
752 @return Version string
753
754 @raises TestFail if no version string is found in image
755 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800756 with open(image, 'rb') as f:
757 image_data = f.read()
758 match = re.findall(version_regex, image_data)
759 if match:
760 return match[0]
761 else:
762 raise error.TestFail('Failed to read version from %s.' % image)
763
764
Garry Wangad2a1712020-03-26 15:06:43 -0700765 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800766 local_tarball=None, verify_version=False,
767 try_scp=False):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700768 """Install firmware to the DUT.
769
770 Use stateful update if the DUT is already running the same build.
771 Stateful update does not update kernel and tends to run much faster
772 than a full reimage. If the DUT is running a different build, or it
773 failed to do a stateful update, full update, including kernel update,
774 will be applied to the DUT.
775
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800776 Once a host enters firmware_install its fw[ro|rw]_version label will
777 be removed. After the firmware is updated successfully, a new
778 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700779
780 @param build: The build version to which we want to provision the
781 firmware of the machine,
782 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800783 @param rw_only: True to only install firmware to its RW portions. Keep
784 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700785 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800786 @param local_tarball: Path to local firmware image for installing
787 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800788 @param verify_version: True to verify EC and BIOS versions after
789 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800790 @param try_scp: False to always program using servo, true to try copying
791 the firmware and programming from the DUT.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700792
793 TODO(dshi): After bug 381718 is fixed, update here with corresponding
794 exceptions that could be raised.
795
796 """
797 if not self.servo:
798 raise error.TestError('Host %s does not have servo.' %
799 self.hostname)
800
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700801 # Get the DUT board name from AFE.
802 info = self.host_info_store.get()
803 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700804 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800805
806 if board is None or board == '':
807 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700808
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700809 if model is None or model == '':
810 model = self.get_platform_from_fwid()
811
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800812 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700813 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800814 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700815 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800816
817 ds = dev_server.ImageServer.resolve(build, self.hostname)
818 ds.stage_artifacts(build, ['firmware'])
819
820 if not dest:
821 tmpd = autotemp.tempdir(unique_id='fwimage')
822 dest = tmpd.name
823
824 # Download firmware image
Dan Shi9cb0eec2014-06-03 09:04:50 -0700825 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700826 local_tarball = os.path.join(dest, os.path.basename(fwurl))
xixuan4e116822016-11-17 15:32:10 -0800827 ds.download_file(fwurl, local_tarball)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700828
Brent Peterson1cb623a2020-01-09 13:14:28 -0800829 # Extract EC image from tarball
830 logging.info('Extracting EC image.')
831 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
832
833 # Extract BIOS image from tarball
834 logging.info('Extracting BIOS image.')
835 bios_image = self.servo.extract_bios_image(board, model, local_tarball)
836
Brent Petersonc70a1832020-01-24 15:54:35 -0800837 # Clear firmware version labels
838 self._clear_fw_version_labels(rw_only)
839
840 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800841 try:
Brent Petersonc70a1832020-01-24 15:54:35 -0800842 # Check if DUT is available and copying to DUT is enabled
843 if self.is_up() and try_scp:
844 # DUT is available, make temp firmware directory to store images
845 logging.info('Making temp folder.')
846 dest_folder = '/tmp/firmware'
847 self.run('mkdir -p ' + dest_folder)
848
Brent Petersonc70a1832020-01-24 15:54:35 -0800849 # Send BIOS firmware image to DUT
850 logging.info('Sending BIOS firmware.')
851 dest_bios_path = os.path.join(dest_folder,
852 os.path.basename(bios_image))
853 self.send_file(bios_image, dest_bios_path)
854
Brent Peterson669edf42020-02-07 15:07:54 -0800855 # Initialize firmware update command for BIOS image
856 fw_cmd = self._FW_UPDATE_CMD % (dest_bios_path,
Brent Petersonc70a1832020-01-24 15:54:35 -0800857 '--wp=1' if rw_only else '')
Brent Peterson669edf42020-02-07 15:07:54 -0800858
859 # Send EC firmware image to DUT when EC image was found
860 if ec_image:
861 logging.info('Sending EC firmware.')
862 dest_ec_path = os.path.join(dest_folder,
863 os.path.basename(ec_image))
864 self.send_file(ec_image, dest_ec_path)
865
866 # Add EC image to firmware update command
867 fw_cmd += ' -e %s' % dest_ec_path
868
869 # Update firmware on DUT
870 logging.info('Updating firmware.')
Brent Petersonc70a1832020-01-24 15:54:35 -0800871 self.run(fw_cmd)
872 else:
873 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800874 if ec_image:
875 self.servo.program_ec(ec_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800876 self.servo.program_bios(bios_image, rw_only)
877 if utils.host_is_in_lab_zone(self.hostname):
878 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800879
880 # Reboot and wait for DUT after installing firmware
881 logging.info('Rebooting DUT.')
882 self.servo.get_power_state_controller().reset()
883 time.sleep(self.servo.BOOT_DELAY)
884 self.test_wait_for_boot()
885
886 # When enabled verify EC and BIOS firmware version after programming
887 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800888 # Check programmed EC firmware when EC image was found
889 if ec_image:
890 logging.info('Checking EC firmware version.')
891 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800892 ec_version_prefix = dest_ec_version.split('_', 1)[0]
893 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800894 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800895 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800896 if dest_ec_version != image_ec_version:
897 raise error.TestFail(
898 'Failed to update EC RO, version %s (expected %s)' %
899 (dest_ec_version, image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800900
901 # Check programmed BIOS firmware against expected version
902 logging.info('Checking BIOS firmware version.')
903 dest_bios_version = self.get_firmware_version()
904 bios_version_prefix = dest_bios_version.split('.', 1)[0]
905 bios_regex = self._BIOS_REGEX % bios_version_prefix
906 image_bios_version = self.get_version_from_image(bios_image,
907 bios_regex)
908 if dest_bios_version != image_bios_version:
909 raise error.TestFail(
910 'Failed to update BIOS RO, version %s (expected %s)' %
911 (dest_bios_version, image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700912 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700913 if tmpd:
914 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700915
916
beepsf079cfb2013-09-18 17:49:51 -0700917 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
918 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500919 """
920 Re-install the OS on the DUT by:
921 1) installing a test image on a USB storage device attached to the Servo
922 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800923 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700924 3) installing the image with chromeos-install.
925
Scott Zawalski62bacae2013-03-05 10:40:32 -0500926 @param image_url: If specified use as the url to install on the DUT.
927 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700928 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
929 Factory images need a longer usb_boot_timeout than regular
930 cros images.
931 @param install_timeout: The timeout to use when installing the chromeos
932 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800933
Scott Zawalski62bacae2013-03-05 10:40:32 -0500934 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700935
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800936 """
Garry Wang7b0e1b72020-03-25 19:08:59 -0700937 if image_url:
938 logging.info('Downloading image to USB, then booting from it.'
939 ' Usb boot timeout = %s', usb_boot_timeout)
940 else:
941 logging.info('Booting from USB directly. Usb boot timeout = %s',
942 usb_boot_timeout)
943
944 metrics_field = {'download': bool(image_url)}
945 metrics.Counter(
946 'chromeos/autotest/provision/servo_install/download_image'
947 ).increment(fields=metrics_field)
948
Allen Li48a13fe2016-11-22 14:10:40 -0800949 with metrics.SecondsTimer(
950 'chromeos/autotest/provision/servo_install/boot_duration'):
951 self.servo.install_recovery_image(image_url)
952 if not self.wait_up(timeout=usb_boot_timeout):
953 raise hosts.AutoservRepairError(
954 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -0700955 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500956
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800957 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
958 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800959 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800960 try:
961 self.run('chromeos-tpm-recovery')
962 except error.AutoservRunError:
963 logging.warn('chromeos-tpm-recovery is too old.')
964
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800965
Allen Li48a13fe2016-11-22 14:10:40 -0800966 with metrics.SecondsTimer(
967 'chromeos/autotest/provision/servo_install/install_duration'):
968 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -0700969 try:
970 self.run('chromeos-install --yes',timeout=install_timeout)
971 self.halt()
972 finally:
973 # We need reset the DUT no matter re-install success or not,
974 # as we don't want leave the DUT in boot from usb state.
975 logging.info('Power cycling DUT through servo.')
976 self.servo.get_power_state_controller().power_off()
977 self.servo.switch_usbkey('off')
978 # N.B. The Servo API requires that we use power_on() here
979 # for two reasons:
980 # 1) After turning on a DUT in recovery mode, you must turn
981 # it off and then on with power_on() once more to
982 # disable recovery mode (this is a Parrot specific
983 # requirement).
984 # 2) After power_off(), the only way to turn on is with
985 # power_on() (this is a Storm specific requirement).
986 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -0700987
988 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -0800989 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -0700990 raise hosts.AutoservRepairError('DUT failed to reboot installed '
991 'test image after %d seconds' %
992 self.BOOT_TIMEOUT,
993 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500994
995
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700996 def set_servo_host(self, host, servo_state = None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700997 """Set our servo host member, and associated servo.
998
999 @param host Our new `ServoHost`.
1000 """
1001 self._servo_host = host
1002 if self._servo_host is not None:
1003 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001004 servo_state = self._servo_host.get_servo_state()
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001005 else:
1006 self.servo = None
Otabek Kasimov41301a22020-05-10 15:28:21 -07001007 self.set_servo_type()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001008 self.set_servo_state(servo_state)
1009
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001010
Richard Barnette9a26ad62016-06-10 12:03:08 -07001011 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001012 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001013 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001014
Richard Barnette9a26ad62016-06-10 12:03:08 -07001015 If the servo object is missing, attempt to repair the servo
1016 host. Repair failures are passed back to the caller.
1017
1018 @raise AutoservError: If there is no servo host for this CrOS
1019 host.
1020 """
1021 if self.servo:
1022 return
1023 if not self._servo_host:
1024 raise error.AutoservError('No servo host for %s.' %
1025 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001026 try:
1027 self._servo_host.repair()
1028 except:
1029 raise
1030 finally:
1031 self.set_servo_host(self._servo_host)
1032
1033
Otabek Kasimov41301a22020-05-10 15:28:21 -07001034 def set_servo_type(self):
1035 """Set servo info labels to dut host_info"""
1036 if not self.servo:
1037 logging.warning('Servo is not initialized to get servo_type.')
1038 return
1039 servo_type = self.servo.get_servo_type()
1040 if not servo_type:
1041 logging.warning('Cannot collect servo_type from servo'
1042 ' by `dut-control servo_type`! Please file a bug'
1043 ' and inform infra team as we are not expected '
1044 ' to reach this point.')
1045 return
1046 host_info = self.host_info_store.get()
1047 prefix = servo_constants.SERVO_TYPE_LABEL_PREFIX
1048 old_type = host_info.get_label_value(prefix)
1049 if old_type == servo_type:
1050 # do not need update
1051 return
1052 host_info.set_version_label(prefix, servo_type)
1053 self.host_info_store.commit(host_info)
1054 logging.info('ServoHost: servo_type updated to %s '
1055 '(previous: %s)', servo_type, old_type)
1056
1057
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001058 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001059 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001060 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001061 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001062 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001063 old_state = host_info.get_label_value(servo_state_prefix)
1064 if old_state == servo_state:
1065 # do not need update
1066 return
1067 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001068 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001069 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1070 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001071
1072
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001073 def get_servo_state(self):
1074 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001075 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001076 return host_info.get_label_value(servo_state_prefix)
1077
Otabek Kasimov41301a22020-05-10 15:28:21 -07001078
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001079 def repair(self):
1080 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001081
1082 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001083 not call back to the parent class, but instead relies on
1084 `self._repair_strategy` to coordinate the verification and
1085 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001086 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001087 message = 'Beginning repair for host %s board %s model %s'
1088 info = self.host_info_store.get()
1089 message %= (self.hostname, info.board, info.model)
1090 self.record('INFO', None, None, message)
J. Richard Barnette91137f02016-03-10 16:52:26 -08001091 self._repair_strategy.repair(self)
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001092
Richard Barnette82c35912012-11-20 10:09:10 -08001093
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001094 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001095 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001096 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001097
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001098 if self._chameleon_host:
1099 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001100
1101 if self._servo_host:
1102 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001103
1104
Dan Shi49ca0932014-11-14 11:22:27 -08001105 def get_power_supply_info(self):
1106 """Get the output of power_supply_info.
1107
1108 power_supply_info outputs the info of each power supply, e.g.,
1109 Device: Line Power
1110 online: no
1111 type: Mains
1112 voltage (V): 0
1113 current (A): 0
1114 Device: Battery
1115 state: Discharging
1116 percentage: 95.9276
1117 technology: Li-ion
1118
1119 Above output shows two devices, Line Power and Battery, with details of
1120 each device listed. This function parses the output into a dictionary,
1121 with key being the device name, and value being a dictionary of details
1122 of the device info.
1123
1124 @return: The dictionary of power_supply_info, e.g.,
1125 {'Line Power': {'online': 'yes', 'type': 'main'},
1126 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001127 @raise error.AutoservRunError if power_supply_info tool is not found in
1128 the DUT. Caller should handle this error to avoid false failure
1129 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001130 """
1131 result = self.run('power_supply_info').stdout.strip()
1132 info = {}
1133 device_name = None
1134 device_info = {}
1135 for line in result.split('\n'):
1136 pair = [v.strip() for v in line.split(':')]
1137 if len(pair) != 2:
1138 continue
1139 if pair[0] == 'Device':
1140 if device_name:
1141 info[device_name] = device_info
1142 device_name = pair[1]
1143 device_info = {}
1144 else:
1145 device_info[pair[0]] = pair[1]
1146 if device_name and not device_name in info:
1147 info[device_name] = device_info
1148 return info
1149
1150
1151 def get_battery_percentage(self):
1152 """Get the battery percentage.
1153
1154 @return: The percentage of battery level, value range from 0-100. Return
1155 None if the battery info cannot be retrieved.
1156 """
1157 try:
1158 info = self.get_power_supply_info()
1159 logging.info(info)
1160 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001161 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001162 return None
1163
1164
Philip Chenaf69ead2020-03-27 13:06:42 -07001165 def get_battery_state(self):
1166 """Get the battery charging state.
1167
1168 @return: A string representing the battery charging state. It can be
1169 'Charging', 'Fully charged', or 'Discharging'.
1170 """
1171 try:
1172 info = self.get_power_supply_info()
1173 logging.info(info)
1174 return info['Battery']['state']
1175 except (KeyError, ValueError, error.AutoservRunError):
1176 return None
1177
1178
Daniel Campello8ca25c22019-12-13 16:48:26 -07001179 def get_battery_display_percentage(self):
1180 """Get the battery display percentage.
1181
1182 @return: The display percentage of battery level, value range from
1183 0-100. Return None if the battery info cannot be retrieved.
1184 """
1185 try:
1186 info = self.get_power_supply_info()
1187 logging.info(info)
1188 return float(info['Battery']['display percentage'])
1189 except (KeyError, ValueError, error.AutoservRunError):
1190 return None
1191
1192
Dan Shi49ca0932014-11-14 11:22:27 -08001193 def is_ac_connected(self):
1194 """Check if the dut has power adapter connected and charging.
1195
1196 @return: True if power adapter is connected and charging.
1197 """
1198 try:
1199 info = self.get_power_supply_info()
1200 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001201 except (KeyError, error.AutoservRunError):
1202 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001203
1204
Simran Basi5e6339a2013-03-21 11:34:32 -07001205 def _cleanup_poweron(self):
1206 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001207 info = self.host_info_store.get()
1208 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001209 return
1210 logging.debug('This host has recently interacted with the RPM'
1211 ' Infrastructure. Ensuring power is on.')
1212 try:
1213 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001214 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001215 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001216 logging.error('Failed to turn Power On for this host after '
1217 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001218
1219 battery_percentage = self.get_battery_percentage()
Dan Shif01ebe22014-12-05 13:10:57 -08001220 if battery_percentage and battery_percentage < 50:
Dan Shi49ca0932014-11-14 11:22:27 -08001221 raise
1222 elif self.is_ac_connected():
1223 logging.info('The device has power adapter connected and '
1224 'charging. No need to try to turn RPM on '
1225 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001226 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001227 logging.info('Battery level is now at %s%%. The device may '
1228 'still have enough power to run test, so no '
1229 'exception will be raised.', battery_percentage)
1230
Simran Basi5e6339a2013-03-21 11:34:32 -07001231
Garry Wangad4d4fd2019-01-30 17:00:38 -08001232 def _remove_rpm_changed_tag(self):
1233 info = self.host_info_store.get()
1234 del info.attributes[self._RPM_OUTLET_CHANGED]
1235 self.host_info_store.commit(info)
1236
1237
1238 def _add_rpm_changed_tag(self):
1239 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001240 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001241 self.host_info_store.commit(info)
1242
1243
1244
beepsc87ff602013-07-31 21:53:00 -07001245 def _is_factory_image(self):
1246 """Checks if the image on the DUT is a factory image.
1247
1248 @return: True if the image on the DUT is a factory image.
1249 False otherwise.
1250 """
1251 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1252 return result.exit_status == 0
1253
1254
1255 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001256 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001257
1258 @raises: FactoryImageCheckerException for factory images, since
1259 we cannot attempt to restart ui on them.
1260 error.AutoservRunError for any other type of error that
1261 occurs while restarting ui.
1262 """
1263 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001264 raise FactoryImageCheckerException('Cannot restart ui on factory '
1265 'images')
beepsc87ff602013-07-31 21:53:00 -07001266
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001267 # TODO(jrbarnette): The command to stop/start the ui job
1268 # should live inside cros_ui, too. However that would seem
1269 # to imply interface changes to the existing start()/restart()
1270 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001271 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001272 self.run('stop ui; start ui')
1273 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001274
1275
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001276 def _start_powerd_if_needed(self):
1277 """Start powerd if it isn't already running."""
1278 self.run('start powerd', ignore_status=True)
1279
1280
xixuana3bbc422017-05-04 15:57:21 -07001281 def _get_lsb_release_content(self):
1282 """Return the content of lsb-release file of host."""
1283 return self.run(
1284 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1285
1286
Dan Shi549fb822015-03-24 18:01:11 -07001287 def get_release_version(self):
1288 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1289
1290 @returns The version string in lsb-release, under attribute
1291 CHROMEOS_RELEASE_VERSION.
1292 """
Dan Shi549fb822015-03-24 18:01:11 -07001293 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001294 lsb_release_content=self._get_lsb_release_content())
1295
1296
Don Garrettb9f35802018-01-22 18:25:40 -08001297 def get_release_builder_path(self):
1298 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1299
1300 @returns The version string in lsb-release, under attribute
1301 CHROMEOS_RELEASE_BUILDER_PATH.
1302 """
1303 return lsbrelease_utils.get_chromeos_release_builder_path(
1304 lsb_release_content=self._get_lsb_release_content())
1305
1306
xixuana3bbc422017-05-04 15:57:21 -07001307 def get_chromeos_release_milestone(self):
1308 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1309 from lsb-release.
1310
1311 @returns The version string in lsb-release, under attribute
1312 CHROMEOS_RELEASE_BUILD_TYPE.
1313 """
1314 return lsbrelease_utils.get_chromeos_release_milestone(
1315 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001316
1317
1318 def verify_cros_version_label(self):
1319 """ Make sure host's cros-version label match the actual image in dut.
1320
1321 Remove any cros-version: label that doesn't match that installed in
1322 the dut.
1323
1324 @param raise_error: Set to True to raise exception if any mismatch found
1325
1326 @raise error.AutoservError: If any mismatch between cros-version label
1327 and the build installed in dut is found.
1328 """
Prathmesh Prabhuce2da3a2019-10-04 11:54:51 -07001329 # crbug.com/1007333: This check is being removed.
1330 return True
Dan Shi549fb822015-03-24 18:01:11 -07001331
1332
Laurence Goodby778c9a42017-05-24 19:24:07 -07001333 def cleanup_services(self):
1334 """Reinitializes the device for cleanup.
1335
1336 Subclasses may override this to customize the cleanup method.
1337
1338 To indicate failure of the reset, the implementation may raise
1339 any of:
1340 error.AutoservRunError
1341 error.AutotestRunError
1342 FactoryImageCheckerException
1343
1344 @raises error.AutoservRunError
1345 @raises error.AutotestRunError
1346 @raises error.FactoryImageCheckerException
1347 """
1348 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001349 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001350
1351
beepsc87ff602013-07-31 21:53:00 -07001352 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001353 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001354 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001355 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001356 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001357 except (error.AutotestRunError, error.AutoservRunError,
1358 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001359 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001360
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001361 # cleanup routines, i.e. reboot the machine.
1362 super(CrosHost, self).cleanup()
1363
Simran Basi5e6339a2013-03-21 11:34:32 -07001364 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001365 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001366 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001367 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001368
1369
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001370 def reboot(self, **dargs):
1371 """
1372 This function reboots the site host. The more generic
1373 RemoteHost.reboot() performs sync and sleeps for 5
1374 seconds. This is not necessary for Chrome OS devices as the
1375 sync should be finished in a short time during the reboot
1376 command.
1377 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001378 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001379 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001380 dargs['reboot_cmd'] = ('sleep 1; '
1381 'reboot & sleep %d; '
1382 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001383 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001384 if 'fastsync' not in dargs:
1385 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001386
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001387 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001388 # Record who called us
1389 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001390 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001391 'dut_host_name' : self.hostname,
1392 'success' : True}
1393 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001394 'caller' : "%s:%s" % (orig.co_filename,
1395 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001396 'success' : True,
1397 'error' : ''}
1398
Vincent Palatin80780b22016-07-27 16:02:37 +02001399 t0 = time.time()
1400 try:
1401 super(CrosHost, self).reboot(**dargs)
1402 except Exception as e:
1403 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001404 metric_debug_fields['success'] = False
1405 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001406 raise
1407 finally:
1408 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001409 metrics.Counter(
1410 'chromeos/autotest/autoserv/reboot_count').increment(
1411 fields=metric_fields)
1412 metrics.Counter(
1413 'chromeos/autotest/autoserv/reboot_debug').increment(
1414 fields=metric_debug_fields)
1415 metrics.SecondsDistribution(
1416 'chromeos/autotest/autoserv/reboot_duration').add(
1417 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001418
1419
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001420 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001421 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001422 """
1423 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001424
1425 @param suspend_time: How long to suspend as integer seconds.
1426 @param suspend_cmd: Suspend command to execute.
1427 @param allow_early_resume: If False and if device resumes before
1428 |suspend_time|, throw an error.
1429
1430 @exception AutoservSuspendError Host resumed earlier than
1431 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001432 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001433
1434 if suspend_cmd is None:
1435 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001436 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001437 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001438 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001439 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1440 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001441
1442
Simran Basiec564392014-08-25 16:48:09 -07001443 def upstart_status(self, service_name):
1444 """Check the status of an upstart init script.
1445
1446 @param service_name: Service to look up.
1447
1448 @returns True if the service is running, False otherwise.
1449 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001450 return 'start/running' in self.run('status %s' % service_name,
1451 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001452
Tom Hughese9552342018-12-18 14:29:25 -08001453 def upstart_stop(self, service_name):
1454 """Stops an upstart job if it's running.
1455
1456 @param service_name: Service to stop
1457
1458 @returns True if service has been stopped or was already stopped
1459 False otherwise.
1460 """
1461 if not self.upstart_status(service_name):
1462 return True
1463
1464 result = self.run('stop %s' % service_name, ignore_status=True)
1465 if result.exit_status != 0:
1466 return False
1467 return True
1468
1469 def upstart_restart(self, service_name):
1470 """Restarts (or starts) an upstart job.
1471
1472 @param service_name: Service to start/restart
1473
1474 @returns True if service has been started/restarted, False otherwise.
1475 """
1476 cmd = 'start'
1477 if self.upstart_status(service_name):
1478 cmd = 'restart'
1479 cmd = cmd + ' %s' % service_name
1480 result = self.run(cmd)
1481 if result.exit_status != 0:
1482 return False
1483 return True
Simran Basiec564392014-08-25 16:48:09 -07001484
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001485 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001486 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001487
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001488 Tests for the following conditions:
1489 1. All conditions tested by the parent version of this
1490 function.
1491 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001492 3. Sufficient space in /mnt/stateful_partition/encrypted.
1493 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001494
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001495 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001496 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001497 default_kilo_inodes_required = CONFIG.get_config_value(
1498 'SERVER', 'kilo_inodes_required', type=int, default=100)
1499 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1500 kilo_inodes_required = CONFIG.get_config_value(
1501 'SERVER', 'kilo_inodes_required_%s' % board,
1502 type=int, default=default_kilo_inodes_required)
1503 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001504 self.check_diskspace(
1505 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001506 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001507 'SERVER', 'gb_diskspace_required', type=float,
1508 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001509 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1510 # Not all targets build with encrypted stateful support.
1511 if self.path_exists(encrypted_stateful_path):
1512 self.check_diskspace(
1513 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001514 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001515 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1516 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001517
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001518 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001519
beepsc87ff602013-07-31 21:53:00 -07001520 # Factory images don't run update engine,
1521 # goofy controls dbus on these DUTs.
1522 if not self._is_factory_image():
1523 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001524
Dan Shi549fb822015-03-24 18:01:11 -07001525 self.verify_cros_version_label()
1526
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001527
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001528 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1529 def wait_for_system_services(self):
1530 """Waits for system-services to be running.
1531
1532 Sometimes, update_engine will take a while to update firmware, so we
1533 should give this some time to finish. See crbug.com/765686#c38 for
1534 details.
1535 """
1536 if not self.upstart_status('system-services'):
1537 raise error.AutoservError('Chrome failed to reach login. '
1538 'System services not running.')
1539
1540
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001541 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001542 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001543 message = 'Beginning verify for host %s board %s model %s'
1544 info = self.host_info_store.get()
1545 message %= (self.hostname, info.board, info.model)
1546 self.record('INFO', None, None, message)
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001547 self._repair_strategy.verify(self)
1548
1549
Fang Deng96667ca2013-08-01 17:46:18 -07001550 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001551 connect_timeout=None, alive_interval=None,
1552 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001553 """Override default make_ssh_command to use options tuned for Chrome OS.
1554
1555 Tuning changes:
1556 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1557 connection failure. Consistency with remote_access.sh.
1558
Samuel Tan2ce155b2015-06-23 18:24:38 -07001559 - ServerAliveInterval=900; which causes SSH to ping connection every
1560 900 seconds. In conjunction with ServerAliveCountMax ensures
1561 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001562 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001563 the test completed successfully. Later increased from 180 seconds to
1564 900 seconds to account for tests where the DUT is suspended for
1565 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001566
1567 - ServerAliveCountMax=3; consistency with remote_access.sh.
1568
1569 - ConnectAttempts=4; reduce flakiness in connection errors;
1570 consistency with remote_access.sh.
1571
1572 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1573 Host keys change with every new installation, don't waste
1574 memory/space saving them.
1575
1576 - SSH protocol forced to 2; needed for ServerAliveInterval.
1577
1578 @param user User name to use for the ssh connection.
1579 @param port Port on the target host to use for ssh connection.
1580 @param opts Additional options to the ssh command.
1581 @param hosts_file Ignored.
1582 @param connect_timeout Ignored.
1583 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001584 @param alive_count_max Ignored.
1585 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001586 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001587 options = ' '.join([opts, '-o Protocol=2'])
1588 return super(CrosHost, self).make_ssh_command(
1589 user=user, port=port, opts=options, hosts_file='/dev/null',
1590 connect_timeout=30, alive_interval=900, alive_count_max=3,
1591 connection_attempts=4)
1592
1593
Jason Abeleb6f924f2013-11-13 16:01:54 -08001594 def syslog(self, message, tag='autotest'):
1595 """Logs a message to syslog on host.
1596
1597 @param message String message to log into syslog
1598 @param tag String tag prefix for syslog
1599
1600 """
1601 self.run('logger -t "%s" "%s"' % (tag, message))
1602
1603
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001604 def _ping_check_status(self, status):
1605 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001606
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001607 @param status Check the ping status against this value.
1608 @return True iff `status` and the result of ping are the same
1609 (i.e. both True or both False).
1610
1611 """
1612 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1613 return not (status ^ (ping_val == 0))
1614
1615 def _ping_wait_for_status(self, status, timeout):
1616 """Wait for the host to have a given status (UP or DOWN).
1617
1618 Status is checked by polling. Polling will not last longer
1619 than the number of seconds in `timeout`. The polling
1620 interval will be long enough that only approximately
1621 _PING_WAIT_COUNT polling cycles will be executed, subject
1622 to a maximum interval of about one minute.
1623
1624 @param status Waiting will stop immediately if `ping` of the
1625 host returns this status.
1626 @param timeout Poll for at most this many seconds.
1627 @return True iff the host status from `ping` matched the
1628 requested status at the time of return.
1629
1630 """
1631 # _ping_check_status() takes about 1 second, hence the
1632 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001633 # FIXME: if the ping command errors then _ping_check_status()
1634 # returns instantly. If timeout is also smaller than twice
1635 # _PING_WAIT_COUNT then the while loop below forks many
1636 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1637 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1638 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001639 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1640 end_time = time.time() + timeout
1641 while time.time() <= end_time:
1642 if self._ping_check_status(status):
1643 return True
1644 if poll_interval > 0:
1645 time.sleep(poll_interval)
1646
1647 # The last thing we did was sleep(poll_interval), so it may
1648 # have been too long since the last `ping`. Check one more
1649 # time, just to be sure.
1650 return self._ping_check_status(status)
1651
1652 def ping_wait_up(self, timeout):
1653 """Wait for the host to respond to `ping`.
1654
1655 N.B. This method is not a reliable substitute for
1656 `wait_up()`, because a host that responds to ping will not
1657 necessarily respond to ssh. This method should only be used
1658 if the target DUT can be considered functional even if it
1659 can't be reached via ssh.
1660
1661 @param timeout Minimum time to allow before declaring the
1662 host to be non-responsive.
1663 @return True iff the host answered to ping before the timeout.
1664
1665 """
1666 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001667
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001668 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001669 """Wait until the host no longer responds to `ping`.
1670
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001671 This function can be used as a slightly faster version of
1672 `wait_down()`, by avoiding potentially long ssh timeouts.
1673
1674 @param timeout Minimum time to allow for the host to become
1675 non-responsive.
1676 @return True iff the host quit answering ping before the
1677 timeout.
1678
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001679 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001680 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001681
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001682 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001683 """Wait for the client to enter low-power sleep mode.
1684
1685 The test for "is asleep" can't distinguish a system that is
1686 powered off; to confirm that the unit was asleep, it is
1687 necessary to force resume, and then call
1688 `test_wait_for_resume()`.
1689
1690 This function is expected to be called from a test as part
1691 of a sequence like the following:
1692
1693 ~~~~~~~~
1694 boot_id = host.get_boot_id()
1695 # trigger sleep on the host
1696 host.test_wait_for_sleep()
1697 # trigger resume on the host
1698 host.test_wait_for_resume(boot_id)
1699 ~~~~~~~~
1700
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001701 @param sleep_timeout time limit in seconds to allow the host sleep.
1702
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001703 @exception TestFail The host did not go to sleep within
1704 the allowed time.
1705 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001706 if sleep_timeout is None:
1707 sleep_timeout = self.SLEEP_TIMEOUT
1708
1709 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001710 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001711 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001712
1713
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001714 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001715 """Wait for the client to resume from low-power sleep mode.
1716
1717 The `old_boot_id` parameter should be the value from
1718 `get_boot_id()` obtained prior to entering sleep mode. A
1719 `TestFail` exception is raised if the boot id changes.
1720
1721 See @ref test_wait_for_sleep for more on this function's
1722 usage.
1723
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001724 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001725 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001726 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001727
1728 @exception TestFail The host did not respond within the
1729 allowed time.
1730 @exception TestFail The host responded, but the boot id test
1731 indicated a reboot rather than a sleep
1732 cycle.
1733 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001734 if resume_timeout is None:
1735 resume_timeout = self.RESUME_TIMEOUT
1736
1737 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001738 raise error.TestFail(
1739 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001740 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001741 else:
1742 new_boot_id = self.get_boot_id()
1743 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001744 logging.error('client rebooted (old boot %s, new boot %s)',
1745 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001746 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001747 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001748
1749
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001750 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001751 """Wait for the client to shut down.
1752
1753 The test for "has shut down" can't distinguish a system that
1754 is merely asleep; to confirm that the unit was down, it is
1755 necessary to force boot, and then call test_wait_for_boot().
1756
1757 This function is expected to be called from a test as part
1758 of a sequence like the following:
1759
1760 ~~~~~~~~
1761 boot_id = host.get_boot_id()
1762 # trigger shutdown on the host
1763 host.test_wait_for_shutdown()
1764 # trigger boot on the host
1765 host.test_wait_for_boot(boot_id)
1766 ~~~~~~~~
1767
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001768 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001769 @exception TestFail The host did not shut down within the
1770 allowed time.
1771 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001772 if shutdown_timeout is None:
1773 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1774
1775 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001776 raise error.TestFail(
1777 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001778 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001779
1780
1781 def test_wait_for_boot(self, old_boot_id=None):
1782 """Wait for the client to boot from cold power.
1783
1784 The `old_boot_id` parameter should be the value from
1785 `get_boot_id()` obtained prior to shutting down. A
1786 `TestFail` exception is raised if the boot id does not
1787 change. The boot id test is omitted if `old_boot_id` is not
1788 specified.
1789
1790 See @ref test_wait_for_shutdown for more on this function's
1791 usage.
1792
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001793 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001794 shut down.
1795
1796 @exception TestFail The host did not respond within the
1797 allowed time.
1798 @exception TestFail The host responded, but the boot id test
1799 indicated that there was no reboot.
1800 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001801 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001802 raise error.TestFail(
1803 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001804 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001805 elif old_boot_id:
1806 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001807 logging.error('client not rebooted (boot %s)',
1808 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001809 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001810 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001811
1812
1813 @staticmethod
1814 def check_for_rpm_support(hostname):
1815 """For a given hostname, return whether or not it is powered by an RPM.
1816
Simran Basi1df55112013-09-06 11:25:09 -07001817 @param hostname: hostname to check for rpm support.
1818
Simran Basid5e5e272012-09-24 15:23:59 -07001819 @return None if this host does not follows the defined naming format
1820 for RPM powered DUT's in the lab. If it does follow the format,
1821 it returns a regular expression MatchObject instead.
1822 """
Fang Dengbaff9082015-01-06 13:46:15 -08001823 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001824
1825
1826 def has_power(self):
1827 """For this host, return whether or not it is powered by an RPM.
1828
1829 @return True if this host is in the CROS lab and follows the defined
1830 naming format.
1831 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001832 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001833
1834
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001835 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001836 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001837
1838 @param state Specifies which power state to set to DUT
1839 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001840 use. By default "RPM" or "CCD" will be used based
1841 on servo type. Valid values from
1842 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001843
1844 """
1845 ACCEPTABLE_STATES = ['ON', 'OFF']
1846
Garry Wang5e5538a2019-04-08 15:36:18 -07001847 if not power_method:
1848 power_method = self.get_default_power_method()
1849
1850 state = state.upper()
1851 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001852 raise error.TestError('State must be one of: %s.'
1853 % (ACCEPTABLE_STATES,))
1854
1855 if power_method == self.POWER_CONTROL_SERVO:
1856 logging.info('Setting servo port J10 to %s', state)
1857 self.servo.set('prtctl3_pwren', state.lower())
1858 time.sleep(self._USB_POWER_TIMEOUT)
1859 elif power_method == self.POWER_CONTROL_MANUAL:
1860 logging.info('You have %d seconds to set the AC power to %s.',
1861 self._POWER_CYCLE_TIMEOUT, state)
1862 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001863 elif power_method == self.POWER_CONTROL_CCD:
1864 servo_role = 'src' if state == 'ON' else 'snk'
1865 logging.info('servo ccd power pass through detected,'
1866 ' changing servo_role to %s.', servo_role)
1867 self.servo.set_servo_v4_role(servo_role)
1868 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001869 # Make sure we don't leave DUT with no power(servo_role=snk)
1870 # when DUT is not pingable, as we raise a exception here
1871 # that may break a power cycle in the middle.
1872 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001873 raise error.AutoservError(
1874 'DUT failed to regain network connection after %d seconds.'
1875 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001876 else:
1877 if not self.has_power():
1878 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001879 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07001880 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07001881
1882
Garry Wang5e5538a2019-04-08 15:36:18 -07001883 def power_off(self, power_method=None):
1884 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001885
1886 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001887 use. By default "RPM" or "CCD" will be used based
1888 on servo type. Valid values from
1889 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001890
1891 """
1892 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07001893
1894
Garry Wang5e5538a2019-04-08 15:36:18 -07001895 def power_on(self, power_method=None):
1896 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001897
1898 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001899 use. By default "RPM" or "CCD" will be used based
1900 on servo type. Valid values from
1901 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001902
1903 """
1904 self._set_power('ON', power_method)
1905
1906
Garry Wang5e5538a2019-04-08 15:36:18 -07001907 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001908 """Cycle power to this host by turning it OFF, then ON.
1909
1910 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001911 use. By default "RPM" or "CCD" will be used based
1912 on servo type. Valid values from
1913 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001914
1915 """
Garry Wang5e5538a2019-04-08 15:36:18 -07001916 if not power_method:
1917 power_method = self.get_default_power_method()
1918
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001919 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07001920 self.POWER_CONTROL_MANUAL,
1921 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001922 self.power_off(power_method=power_method)
1923 time.sleep(self._POWER_CYCLE_TIMEOUT)
1924 self.power_on(power_method=power_method)
1925 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08001926 self._add_rpm_changed_tag()
1927 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07001928
1929
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001930 def get_platform_from_fwid(self):
1931 """Determine the platform from the crossystem fwid.
1932
1933 @returns a string representing this host's platform.
1934 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06001935 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001936 crossystem = utils.Crossystem(self)
1937 crossystem.init()
1938 # Extract fwid value and use the leading part as the platform id.
1939 # fwid generally follow the format of {platform}.{firmware version}
1940 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
1941 platform = crossystem.fwid().split('.')[0].lower()
1942 # Newer platforms start with 'Google_' while the older ones do not.
1943 return platform.replace('google_', '')
1944
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001945
Simran Basic6f1f7a2012-10-16 10:47:46 -07001946 def get_platform(self):
1947 """Determine the correct platform label for this host.
1948
1949 @returns a string representing this host's platform.
1950 """
C Shapiroed87c6f2018-04-19 09:13:58 -06001951 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1952 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06001953 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001954 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06001955 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001956 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07001957
1958
Greg Edelstona7b05d12020-04-01 16:00:51 -06001959 def get_model_from_cros_config(self):
1960 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001961
Greg Edelstona7b05d12020-04-01 16:00:51 -06001962 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001963 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06001964 return cros_config.call_cros_config_get_output('/ name',
1965 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001966
1967
Hung-ying Tyanb1328032014-04-01 14:18:54 +08001968 def get_architecture(self):
1969 """Determine the correct architecture label for this host.
1970
1971 @returns a string representing this host's architecture.
1972 """
1973 crossystem = utils.Crossystem(self)
1974 crossystem.init()
1975 return crossystem.arch()
1976
1977
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001978 def get_chrome_version(self):
1979 """Gets the Chrome version number and milestone as strings.
1980
1981 Invokes "chrome --version" to get the version number and milestone.
1982
1983 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
1984 current Chrome version number as a string (in the form "W.X.Y.Z")
1985 and "milestone" is the first component of the version number
1986 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
1987 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
1988 of "chrome --version" and the milestone will be the empty string.
1989
1990 """
MK Ryu35d661e2014-09-25 17:44:10 -07001991 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001992 return utils.parse_chrome_version(version_string)
1993
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001994
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001995 def get_ec_version(self):
1996 """Get the ec version as strings.
1997
1998 @returns a string representing this host's ec version.
1999 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002000 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002001 result = self.run(command, ignore_status=True)
2002 if result.exit_status != 0:
2003 return ''
2004 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002005
2006
2007 def get_firmware_version(self):
2008 """Get the firmware version as strings.
2009
2010 @returns a string representing this host's firmware version.
2011 """
2012 crossystem = utils.Crossystem(self)
2013 crossystem.init()
2014 return crossystem.fwid()
2015
2016
2017 def get_hardware_revision(self):
2018 """Get the hardware revision as strings.
2019
2020 @returns a string representing this host's hardware revision.
2021 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08002022 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08002023 result = self.run(command, ignore_status=True)
2024 if result.exit_status != 0:
2025 return ''
2026 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08002027
2028
2029 def get_kernel_version(self):
2030 """Get the kernel version as strings.
2031
2032 @returns a string representing this host's kernel version.
2033 """
2034 return self.run('uname -r').stdout.strip()
2035
2036
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002037 def get_cpu_name(self):
2038 """Get the cpu name as strings.
2039
2040 @returns a string representing this host's cpu name.
2041 """
2042
2043 # Try get cpu name from device tree first
2044 if self.path_exists('/proc/device-tree/compatible'):
2045 command = ' | '.join(
2046 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2047 'tail -1'])
2048 return self.run(command).stdout.strip().replace(',', ' ')
2049
2050 # Get cpu name from uname -p
2051 command = 'uname -p'
2052 ret = self.run(command).stdout.strip()
2053
2054 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2055 # Try get cpu name from /proc/cpuinfo instead
2056 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2057 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2058 self = self.run(command).stdout.strip()
2059
2060 # Remove bloat from CPU name, for example
2061 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2062 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2063 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2064 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2065 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2066 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2067
2068
2069 def get_screen_resolution(self):
2070 """Get the screen(s) resolution as strings.
2071 In case of more than 1 monitor, return resolution for each monitor
2072 separate with plus sign.
2073
2074 @returns a string representing this host's screen(s) resolution.
2075 """
2076 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2077 ret = self.run(command, ignore_status=True)
2078 # We might have Chromebox without a screen
2079 if ret.exit_status != 0:
2080 return ''
2081 return ret.stdout.strip().replace('\n', '+')
2082
2083
2084 def get_mem_total_gb(self):
2085 """Get total memory available in the system in GiB (2^20).
2086
2087 @returns an integer representing total memory
2088 """
2089 mem_total_kb = self.read_from_meminfo('MemTotal')
2090 kb_in_gb = float(2 ** 20)
2091 return int(round(mem_total_kb / kb_in_gb))
2092
2093
2094 def get_disk_size_gb(self):
2095 """Get size of disk in GB (10^9)
2096
2097 @returns an integer representing size of disk, 0 in Error Case
2098 """
2099 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2100 result = self.run(command, ignore_status=True)
2101 if result.exit_status != 0:
2102 return 0
2103 _, _, block, _ = re.split(r' +', result.stdout.strip())
2104 byte_per_block = 1024.0
2105 disk_kb_in_gb = 1e9
2106 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2107
2108
2109 def get_battery_size(self):
2110 """Get size of battery in Watt-hour via sysfs
2111
2112 This method assumes that battery support voltage_min_design and
2113 charge_full_design sysfs.
2114
2115 @returns a float representing Battery size, 0 if error.
2116 """
2117 # sysfs report data in micro scale
2118 battery_scale = 1e6
2119
2120 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2121 result = self.run(command, ignore_status=True)
2122 if result.exit_status != 0:
2123 return 0
2124 voltage = float(result.stdout.strip()) / battery_scale
2125
2126 command = 'cat /sys/class/power_supply/*/charge_full_design'
2127 result = self.run(command, ignore_status=True)
2128 if result.exit_status != 0:
2129 return 0
2130 amphereHour = float(result.stdout.strip()) / battery_scale
2131
2132 return voltage * amphereHour
2133
2134
2135 def get_low_battery_shutdown_percent(self):
2136 """Get the percent-based low-battery shutdown threshold.
2137
2138 @returns a float representing low-battery shutdown percent, 0 if error.
2139 """
2140 ret = 0.0
2141 try:
2142 command = 'check_powerd_config --low_battery_shutdown_percent'
2143 ret = float(self.run(command).stdout)
2144 except error.CmdError:
2145 logging.debug("Can't run %s", command)
2146 except ValueError:
2147 logging.debug("Didn't get number from %s", command)
2148
2149 return ret
2150
2151
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002152 def has_hammer(self):
2153 """Check whether DUT has hammer device or not.
2154
2155 @returns boolean whether device has hammer or not
2156 """
2157 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2158 return self.run(command, ignore_status=True).exit_status == 0
2159
2160
Niranjan Kumar34618872017-05-31 12:57:09 -07002161 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002162 """Returns True if the specified switch was provided to Chrome.
2163
2164 @param switch The chrome switch to search for.
2165 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002166
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002167 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2168 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002169
2170
2171 def oobe_triggers_update(self):
2172 """Returns True if this host has an OOBE flow during which
2173 it will perform an update check and perhaps an update.
2174 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2175 As more such flows are developed, code handling them needs
2176 to be added here.
2177
2178 @return Boolean indicating whether this host's OOBE triggers an update.
2179 """
2180 return self.is_chrome_switch_present(
2181 '--enterprise-enable-zero-touch-enrollment=hands-off')
2182
2183
Kevin Chenga2619dc2016-03-28 11:42:08 -07002184 # TODO(kevcheng): change this to just return the board without the
2185 # 'board:' prefix and fix up all the callers. Also look into removing the
2186 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002187 def get_board(self):
2188 """Determine the correct board label for this host.
2189
2190 @returns a string representing this host's board.
2191 """
2192 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2193 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002194 return (ds_constants.BOARD_PREFIX +
2195 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002196
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002197 def get_channel(self):
2198 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002199
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002200 @returns: a string represeting this host's build channel.
2201 (stable, dev, beta). None on fail.
2202 """
2203 return lsbrelease_utils.get_chromeos_channel(
2204 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002205
Kevin Chenga328da62016-03-31 10:49:04 -07002206 def get_power_supply(self):
2207 """
2208 Determine what type of power supply the host has
2209
2210 @returns a string representing this host's power supply.
2211 'power:battery' when the device has a battery intended for
2212 extended use
2213 'power:AC_primary' when the device has a battery not intended
2214 for extended use (for moving the machine, etc)
2215 'power:AC_only' when the device has no battery at all.
2216 """
2217 psu = self.run(command='mosys psu type', ignore_status=True)
2218 if psu.exit_status:
2219 # The psu command for mosys is not included for all platforms. The
2220 # assumption is that the device will have a battery if the command
2221 # is not found.
2222 return 'power:battery'
2223
2224 psu_str = psu.stdout.strip()
2225 if psu_str == 'unknown':
2226 return None
2227
2228 return 'power:%s' % psu_str
2229
2230
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002231 def has_battery(self):
2232 """Determine if DUT has a battery.
2233
2234 Returns:
2235 Boolean, False if known not to have battery, True otherwise.
2236 """
2237 rv = True
2238 power_supply = self.get_power_supply()
2239 if power_supply == 'power:battery':
2240 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2241 board_type = self.get_board_type()
2242 if board_type in _NO_BATTERY_BOARD_TYPE:
2243 logging.warn('Do NOT believe type %s has battery. '
2244 'See debug for mosys details', board_type)
2245 psu = self.system_output('mosys -vvvv psu type',
2246 ignore_status=True)
2247 logging.debug(psu)
2248 rv = False
2249 elif power_supply == 'power:AC_only':
2250 rv = False
2251
2252 return rv
2253
2254
Kevin Chenga328da62016-03-31 10:49:04 -07002255 def get_servo(self):
2256 """Determine if the host has a servo attached.
2257
2258 If the host has a working servo attached, it should have a servo label.
2259
2260 @return: string 'servo' if the host has servo attached. Otherwise,
2261 returns None.
2262 """
2263 return 'servo' if self._servo_host else None
2264
2265
Kevin Chenga328da62016-03-31 10:49:04 -07002266 def has_internal_display(self):
2267 """Determine if the device under test is equipped with an internal
2268 display.
2269
2270 @return: 'internal_display' if one is present; None otherwise.
2271 """
2272 from autotest_lib.client.cros.graphics import graphics_utils
2273 from autotest_lib.client.common_lib import utils as common_utils
2274
2275 def __system_output(cmd):
2276 return self.run(cmd).stdout
2277
2278 def __read_file(remote_path):
2279 return self.run('cat %s' % remote_path).stdout
2280
2281 # Hijack the necessary client functions so that we can take advantage
2282 # of the client lib here.
2283 # FIXME: find a less hacky way than this
2284 original_system_output = utils.system_output
2285 original_read_file = common_utils.read_file
2286 utils.system_output = __system_output
2287 common_utils.read_file = __read_file
2288 try:
2289 return ('internal_display' if graphics_utils.has_internal_display()
2290 else None)
2291 finally:
2292 utils.system_output = original_system_output
2293 common_utils.read_file = original_read_file
2294
2295
Dan Shi85276d42014-04-08 22:11:45 -07002296 def is_boot_from_usb(self):
2297 """Check if DUT is boot from USB.
2298
2299 @return: True if DUT is boot from usb.
2300 """
2301 device = self.run('rootdev -s -d').stdout.strip()
2302 removable = int(self.run('cat /sys/block/%s/removable' %
2303 os.path.basename(device)).stdout.strip())
2304 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002305
2306
David Haddock19569d82020-05-06 05:30:21 -07002307 def get_active_boot_slot(self):
2308 """Returns the active boot slot."""
2309 return self.run(('rootdev', '-s')).stdout.strip()
2310
2311
Helen Zhang17dae2b2014-11-11 09:25:52 -08002312 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002313 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002314
2315 @param key: meminfo requested
2316
2317 @return the memory value as a string
2318
2319 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002320 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2321 logging.debug('%s', meminfo)
2322 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002323
2324
Rohit Makasana98e696f2016-06-03 18:48:10 -07002325 def get_cpu_arch(self):
2326 """Returns CPU arch of the device.
2327
2328 @return CPU architecture of the DUT.
2329 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002330 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002331 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2332 ignore_status=True).stdout:
2333 return 'x86_64'
2334 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2335 ignore_status=True).stdout:
2336 return 'arm'
2337 return 'i386'
2338
2339
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002340 def get_board_type(self):
2341 """
2342 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002343 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2344
2345 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002346 """
Danny Chan471a8d12015-08-18 14:57:41 -07002347 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2348 ignore_status=True).stdout
2349 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002350 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002351 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002352
2353
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002354 def get_arc_version(self):
2355 """Return ARC version installed on the DUT.
2356
2357 @returns ARC version as string if the CrOS build has ARC, else None.
2358 """
2359 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2360 ignore_status=True).stdout
2361 if arc_version:
2362 return arc_version.split('=')[-1].strip()
2363 return None
2364
2365
Gilad Arnolda76bef02015-09-29 13:55:15 -07002366 def get_os_type(self):
2367 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002368
2369
Kevin Chenga2619dc2016-03-28 11:42:08 -07002370 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002371 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002372 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002373
2374
2375 def get_default_power_method(self):
2376 """
2377 Get the default power method for power_on/off/cycle() methods.
2378 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2379 """
2380 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002381 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002382 if self.servo and self.servo.supports_built_in_pd_control():
2383 self._default_power_method = self.POWER_CONTROL_CCD
2384 else:
2385 logging.debug('Either servo is unitialized or the servo '
2386 'setup does not support pd controls. Falling '
2387 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002388 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002389
2390
2391 def find_usb_devices(self, idVendor, idProduct):
2392 """
2393 Get usb device sysfs name for specific device.
2394
2395 @param idVendor Vendor ID to search in sysfs directory.
2396 @param idProduct Product ID to search in sysfs directory.
2397
2398 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2399 """
2400 # Look for matching file and cut at position 7 to get dir name.
2401 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2402
2403 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2404 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2405
2406 # Use uniq -d to print duplicate line from both command
2407 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2408
2409 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2410
2411
2412 def bind_usb_device(self, usb_node):
2413 """
2414 Bind usb device
2415
2416 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2417 """
2418 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2419 self.run(cmd, ignore_status=True)
2420
2421
2422 def unbind_usb_device(self, usb_node):
2423 """
2424 Unbind usb device
2425
2426 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2427 """
2428 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2429 self.run(cmd, ignore_status=True)
2430
2431
2432 def get_wlan_ip(self):
2433 """
2434 Get ip address of wlan interface.
2435
2436 @return ip address of wlan or empty string if wlan is not connected.
2437 """
2438 cmds = [
2439 'iw dev', # List wlan physical device
2440 'grep Interface', # Grep only interface name
2441 'cut -f 2 -d" "', # Cut the name part
2442 'xargs ifconfig', # Feed it to ifconfig to get ip
2443 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2444 'cut -f 2 -d " "' # Cut the ip part
2445 ]
2446 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002447
2448 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2449 """
2450 Connect to wifi network
2451
2452 @param ssid SSID of the wifi network.
2453 @param passphrase Passphrase of the wifi network. None if not existed.
2454 @param security Security of the wifi network. Default to "psk" if
2455 passphase is given without security. Possible values
2456 are "none", "psk", "802_1x".
2457
2458 @return True if succeed, False if not.
2459 """
2460 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2461 if passphrase:
2462 cmd += ' ' + passphrase
2463 if security:
2464 cmd += ' ' + security
2465 return self.run(cmd, ignore_status=True).exit_status == 0