blob: c3f79dcc44d2978373223cc614ad86c876ee0d5d [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)
224 host = hosts.create_host(machine)
225 host.initialize_btpeer(btpeer_args)
226 ~~~~~~~~
227
228 @param args_dict: Dictionary from which to extract the btpeer
229 arguments.
230 """
231 if 'btpeer_host_list' in args_dict:
232 result = []
233 for btpeer in args_dict['btpeer_host_list'].split(','):
234 result.append({key: value for key,value in
235 zip(('btpeer_host','btpeer_port'),
236 btpeer.split(':'))})
237 return result
238 else:
239 return {key: args_dict[key]
240 for key in ('btpeer_host', 'btpeer_port')
241 if key in args_dict}
242
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800243
244 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700245 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800246 """Extract chameleon options from `args_dict` and return the result.
247
248 Recommended usage:
249 ~~~~~~~~
250 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700251 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
252 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800253 ~~~~~~~~
254
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700255 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800256 arguments.
257 """
Allen Li083866b2016-08-18 10:07:10 -0700258 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700259 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700260 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800261
262
263 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800264 def get_servo_arguments(args_dict):
265 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800266
267 Recommended usage:
268 ~~~~~~~~
269 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700270 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800271 host = hosts.create_host(machine, servo_args=servo_args)
272 ~~~~~~~~
273
274 @param args_dict Dictionary from which to extract the servo
275 arguments.
276 """
Garry Wang11b5e872020-03-11 15:14:08 -0700277 servo_attrs = (servo_constants.SERVO_HOST_ATTR,
278 servo_constants.SERVO_PORT_ATTR,
279 servo_constants.SERVO_BOARD_ATTR,
280 servo_constants.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200281 servo_args = {key: args_dict[key]
282 for key in servo_attrs
283 if key in args_dict}
284 return (
285 None
Garry Wang11b5e872020-03-11 15:14:08 -0700286 if servo_constants.SERVO_HOST_ATTR in servo_args
287 and not servo_args[servo_constants.SERVO_HOST_ATTR]
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200288 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700289
J. Richard Barnette964fba02012-10-24 17:34:29 -0700290
J. Richard Barnette91137f02016-03-10 16:52:26 -0800291 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700292 pdtester_args=None, try_lab_servo=False,
Richard Barnette9a26ad62016-06-10 12:03:08 -0700293 try_servo_repair=False,
J. Richard Barnette91137f02016-03-10 16:52:26 -0800294 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700295 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800296 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700297
Fang Denge545abb2014-12-30 18:43:47 -0800298 This method will attempt to create the test-assistant object
299 (chameleon/servo) when it is needed by the test. Check
300 the docstring of chameleon_host.create_chameleon_host and
301 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700302
Fang Denge545abb2014-12-30 18:43:47 -0800303 @param hostname: Hostname of the dut.
304 @param chameleon_args: A dictionary that contains args for creating
305 a ChameleonHost. See chameleon_host for details.
306 @param servo_args: A dictionary that contains args for creating
307 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700308 @param try_lab_servo: When true, indicates that an attempt should
309 be made to create a ServoHost for a DUT in
310 the test lab, even if not required by
311 `servo_args`. See servo_host for details.
312 @param try_servo_repair: If a servo host is created, check it
313 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800314 See servo_host for details.
315 @param ssh_verbosity_flag: String, to pass to the ssh command to control
316 verbosity.
317 @param ssh_options: String, other ssh options to pass to the ssh
318 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700319 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700320 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700321 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800322 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Kevin Chenga2619dc2016-03-28 11:42:08 -0700323 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700324 # self.env is a dictionary of environment variable settings
325 # to be exported for commands run on the host.
326 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
327 # errors that might happen.
328 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700329 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700330 self._ssh_options = ssh_options
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700331 _servo_host, servo_state = servo_host.create_servo_host(
332 dut=self,
333 servo_args=servo_args,
334 try_lab_servo=try_lab_servo,
335 try_servo_repair=try_servo_repair,
336 dut_host_info=self.host_info_store.get())
337 self.set_servo_host(_servo_host, servo_state)
Garry Wang5e5538a2019-04-08 15:36:18 -0700338 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700339
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800340 # TODO(waihong): Do the simplication on Chameleon too.
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800341 self._chameleon_host = chameleon_host.create_chameleon_host(
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700342 dut=self.hostname,
343 chameleon_args=chameleon_args)
Shijin Abraham783a7dd2020-02-14 15:36:11 -0800344 if self._chameleon_host:
345 self.chameleon = self._chameleon_host.create_chameleon_board()
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800346 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800347 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700348
Shijin Abrahamc09587d2020-02-14 20:46:55 -0800349 # Bluetooth peers. These will be initialized by test if required.
350 self._btpeer_host_list = []
351 self.btpeer_list = []
352 self.btpeer = None
353
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 """
377
378 if type(btpeer_args) is list:
379 btpeer_args_list = btpeer_args
380 else:
381 btpeer_args_list = [btpeer_args]
382
383 self._btpeer_host_list = chameleon_host.create_btpeer_host(
384 dut=self.hostname, btpeer_args_list=btpeer_args_list)
385 logging.debug('Bluetooth peer hosts are %s', self._btpeer_host_list)
386 self.btpeer_list = [_host.create_chameleon_board() for _host in
387 self._btpeer_host_list if _host is not None]
388
389 if len(self.btpeer_list) > 0:
390 self.btpeer = self.btpeer_list[0]
391 else:
392 self.btpeer = None
393
394 logging.debug('After initialize_btpeer btpeer_list %s btpeer_host_list'
395 'is %s and btpeer is %s',self.btpeer_list,
396 self._btpeer_host_list, self.btpeer)
397
398
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000399 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700400 """Get latest stable cros image name from AFE.
401
402 Use the board name from the info store. Should that fail, try to
403 retrieve the board name from the host's installed image itself.
404
405 @returns: current stable cros image name for this host.
406 """
Garry Wange8a8fc22020-04-13 15:04:53 -0700407 info = self.host_info_store.get()
408 if not info.board:
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700409 logging.warn('No board label value found. Trying to infer '
410 'from the host itself.')
411 try:
Garry Wange8a8fc22020-04-13 15:04:53 -0700412 info.labels.append(self.get_board())
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700413 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
414 logging.error('Also failed to get the board name from the DUT '
415 'itself. %s.', str(e))
Garry Wange8a8fc22020-04-13 15:04:53 -0700416 raise error.AutoservError('Cannot determine board of the DUT'
417 ' while getting repair image name.')
418 return afe_utils.get_stable_cros_image_name_v2(info)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500419
420
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700421 def host_version_prefix(self, image):
422 """Return version label prefix.
423
424 In case the CrOS provisioning version is something other than the
425 standard CrOS version e.g. CrOS TH version, this function will
426 find the prefix from provision.py.
427
428 @param image: The image name to find its version prefix.
429 @returns: A prefix string for the image type.
430 """
431 return provision.get_version_label_prefix(image)
432
Andrew Luo3332ab22020-04-28 16:42:03 -0700433 def stage_build_to_usb(self, build):
434 """Stage the current ChromeOS image on the USB stick connected to the
435 servo.
436
437 @param build: The build to download and send to USB.
438 """
439 if not self.servo:
440 raise error.TestError('Host %s does not have servo.' %
441 self.hostname)
442
443 _, update_url = self.stage_image_for_servo(build)
444 self.servo.image_to_servo_usb(update_url)
445 logging.debug('ChromeOS image %s is staged on the USB stick.',
446 build)
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700447
beepsdae65fd2013-07-26 16:24:41 -0700448 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700449 """
450 Make sure job_repo_url of this host is valid.
451
joychen03eaad92013-06-26 09:55:21 -0700452 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700453 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
454 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
455 download and extract it. If the devserver embedded in the url is
456 unresponsive, update the job_repo_url of the host after staging it on
457 another devserver.
458
459 @param job_repo_url: A url pointing to the devserver where the autotest
460 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700461 @param tag: The tag from the server job, in the format
462 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700463
464 @raises DevServerException: If we could not resolve a devserver.
465 @raises AutoservError: If we're unable to save the new job_repo_url as
466 a result of choosing a new devserver because the old one failed to
467 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700468 @raises urllib2.URLError: If the devserver embedded in job_repo_url
469 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700470 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800471 info = self.host_info_store.get()
472 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700473 if not job_repo_url:
474 logging.warning('No job repo url set on host %s', self.hostname)
475 return
476
477 logging.info('Verifying job repo url %s', job_repo_url)
478 devserver_url, image_name = tools.get_devserver_build_from_package_url(
479 job_repo_url)
480
beeps0c865032013-07-30 11:37:06 -0700481 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700482
483 logging.info('Staging autotest artifacts for %s on devserver %s',
484 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700485
486 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700487 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700488 stage_time = time.time() - start_time
489
490 # Record how much of the verification time comes from a devserver
491 # restage. If we're doing things right we should not see multiple
492 # devservers for a given board/build/branch path.
493 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800494 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700495 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800496 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700497 pass
498 else:
beeps0c865032013-07-30 11:37:06 -0700499 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700500 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700501 stats_key = {
502 'board': board,
503 'build_type': build_type,
504 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700505 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700506 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800507
508 monarch_fields = {
509 'board': board,
510 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800511 'branch': branch,
512 'dev_server': devserver,
513 }
514 metrics.Counter(
515 'chromeos/autotest/provision/verify_url'
516 ).increment(fields=monarch_fields)
517 metrics.SecondsDistribution(
518 'chromeos/autotest/provision/verify_url_duration'
519 ).add(stage_time, fields=monarch_fields)
520
521
Dan Shicf4d2032015-03-12 15:04:21 -0700522 def stage_server_side_package(self, image=None):
523 """Stage autotest server-side package on devserver.
524
525 @param image: Full path of an OS image to install or a build name.
526
527 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700528
529 @raise: error.AutoservError if fail to locate the build to test with, or
530 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700531 """
Dan Shid37736b2016-07-06 15:10:29 -0700532 # If enable_drone_in_restricted_subnet is False, do not set hostname
533 # in devserver.resolve call, so a devserver in non-restricted subnet
534 # is picked to stage autotest server package for drone to download.
535 hostname = self.hostname
536 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
537 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700538 if image:
539 image_name = tools.get_build_from_image(image)
540 if not image_name:
541 raise error.AutoservError(
542 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700543 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700544 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700545 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800546 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700547 if job_repo_url:
548 devserver_url, image_name = (
549 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700550 # If enable_drone_in_restricted_subnet is True, use the
551 # existing devserver. Otherwise, resolve a new one in
552 # non-restricted subnet.
553 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
554 ds = dev_server.ImageServer(devserver_url)
555 else:
556 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800557 elif info.build is not None:
558 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700559 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700560 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800561 raise error.AutoservError(
562 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700563 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700564
565 # Get the OS version of the build, for any build older than
566 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
567 match = re.match('.*/R\d+-(\d+)\.', image_name)
568 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700569 raise error.AutoservError(
570 'Build %s is older than %s. Server side packaging is '
571 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700572
Dan Shicf4d2032015-03-12 15:04:21 -0700573 ds.stage_artifacts(image_name, ['autotest_server_package'])
574 return '%s/static/%s/%s' % (ds.url(), image_name,
575 'autotest_server_package.tar.bz2')
576
577
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700578 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700579 """Stage a build on a devserver and return the update_url.
580
581 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700582 @param artifact: a string like 'test_image'. Requests
583 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700584 @returns a tuple of (image_name, URL) like
585 (lumpy-release/R27-3837.0.0,
586 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700587 """
588 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000589 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700590 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800591 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700592 devserver.stage_artifacts(image_name, [artifact])
593 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700594 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700595 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700596 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700597 else:
598 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700599
600
beepse539be02013-07-31 21:57:39 -0700601 def stage_factory_image_for_servo(self, image_name):
602 """Stage a build on a devserver and return the update_url.
603
604 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700605
beepse539be02013-07-31 21:57:39 -0700606 @return: An update URL, eg:
607 http://<devserver>/static/canary-channel/\
608 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700609
610 @raises: ValueError if the factory artifact name is missing from
611 the config.
612
beepse539be02013-07-31 21:57:39 -0700613 """
614 if not image_name:
615 logging.error('Need an image_name to stage a factory image.')
616 return
617
Dan Shib8540a52015-07-16 14:18:23 -0700618 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700619 'CROS', 'factory_artifact', type=str, default='')
620 if not factory_artifact:
621 raise ValueError('Cannot retrieve the factory artifact name from '
622 'autotest config, and hence cannot stage factory '
623 'artifacts.')
624
beepse539be02013-07-31 21:57:39 -0700625 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800626 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700627 devserver.stage_artifacts(
628 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700629 [factory_artifact],
630 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700631
632 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
633
634
Laurence Goodby778c9a42017-05-24 19:24:07 -0700635 def prepare_for_update(self):
636 """Prepares the DUT for an update.
637
638 Subclasses may override this to perform any special actions
639 required before updating.
640 """
Laurence Goodby468de252017-06-08 17:22:53 -0700641 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700642
643
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800644 def _clear_fw_version_labels(self, rw_only):
645 """Clear firmware version labels from the machine.
646
647 @param rw_only: True to only clear fwrw_version; otherewise, clear
648 both fwro_version and fwrw_version.
649 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700650 info = self.host_info_store.get()
651 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800652 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700653 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
654 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700655
656
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800657 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700658 """Add firmware version label to the machine.
659
660 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800661 @param rw_only: True to only add fwrw_version; otherwise, add both
662 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700663
664 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700665 info = self.host_info_store.get()
666 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800667 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700668 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
669 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700670
671
Namyoon Woo33f38852020-04-13 17:26:58 -0700672 def get_latest_release_version(self, platform, ref_board=None):
Namyoon Woo5f894662019-11-15 15:23:23 -0800673 """Search for the latest package release version from the image archive,
674 and return it.
675
Namyoon Woo33f38852020-04-13 17:26:58 -0700676 @param platform: platform name, a.k.a. board or model
677 @param ref_board: reference board name, a.k.a. baseboard, parent
Namyoon Woo5f894662019-11-15 15:23:23 -0800678
Namyoon Woo33f38852020-04-13 17:26:58 -0700679 @return 'firmware-{platform}-{branch}-firmwarebranch/{release-version}/'
680 '{platform}'
Namyoon Woo5f894662019-11-15 15:23:23 -0800681 or None if LATEST release file does not exist.
682 """
683
Namyoon Woo33f38852020-04-13 17:26:58 -0700684 platforms = [ platform ]
Namyoon Woo5f894662019-11-15 15:23:23 -0800685
Namyoon Woo33f38852020-04-13 17:26:58 -0700686 # Search the image path in reference board archive as well.
687 # For example, bob has its binary image under its reference board (gru)
688 # image archive.
689 if ref_board:
690 platforms.append(ref_board)
Namyoon Woo5f894662019-11-15 15:23:23 -0800691
Namyoon Woo33f38852020-04-13 17:26:58 -0700692 for board in platforms:
693 # Read 'LATEST-1.0.0' file
694 branch_dir = provision.FW_BRANCH_GLOB % board
695 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
696 'LATEST-1.0.0')
Namyoon Woo406c7d42020-01-24 15:57:11 -0800697
Namyoon Woo33f38852020-04-13 17:26:58 -0700698 try:
699 # The result could be one or more.
700 result = utils.system_output('gsutil ls -d ' + latest_file)
701
702 candidates = re.findall('gs://.*', result)
703
704 # Found the directory candidates. No need to check the other
705 # board name cadidates. Let's break the loop.
706 break
707 except error.CmdError:
708 # It doesn't exist. Let's move on to the next item.
709 pass
710 else:
Namyoon Woo5f894662019-11-15 15:23:23 -0800711 logging.error('No LATEST release info is available.')
712 return None
713
Namyoon Woo406c7d42020-01-24 15:57:11 -0800714 for cand_dir in candidates:
715 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800716
Namyoon Woo406c7d42020-01-24 15:57:11 -0800717 release_path = cand_dir.replace('LATEST-1.0.0', result)
Namyoon Woo33f38852020-04-13 17:26:58 -0700718 release_path = os.path.join(release_path, platform)
Namyoon Woo406c7d42020-01-24 15:57:11 -0800719 try:
720 # Check if release_path does exist.
721 release = utils.system_output('gsutil ls -d ' + release_path)
722 # Now 'release' has a full directory path: e.g.
723 # gs://chromeos-image-archive/firmware-octopus-11297.B-
724 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
725
726 # Remove "gs://chromeos-image-archive".
727 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
728
729 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
730 return release.strip('/')
731 except error.CmdError:
732 # The directory might not exist. Let's try next candidate.
733 pass
734 else:
735 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800736
Brent Peterson1cb623a2020-01-09 13:14:28 -0800737 @staticmethod
738 def get_version_from_image(image, version_regex):
Brent Peterson8039b472020-02-14 10:51:23 -0800739 """Get version string from binary image using regular expression.
740
741 @param image: Binary image to search
742 @param version_regex: Regular expression to search for
743
744 @return Version string
745
746 @raises TestFail if no version string is found in image
747 """
Brent Peterson1cb623a2020-01-09 13:14:28 -0800748 with open(image, 'rb') as f:
749 image_data = f.read()
750 match = re.findall(version_regex, image_data)
751 if match:
752 return match[0]
753 else:
754 raise error.TestFail('Failed to read version from %s.' % image)
755
756
Garry Wangad2a1712020-03-26 15:06:43 -0700757 def firmware_install(self, build, rw_only=False, dest=None,
Brent Petersonc70a1832020-01-24 15:54:35 -0800758 local_tarball=None, verify_version=False,
759 try_scp=False):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700760 """Install firmware to the DUT.
761
762 Use stateful update if the DUT is already running the same build.
763 Stateful update does not update kernel and tends to run much faster
764 than a full reimage. If the DUT is running a different build, or it
765 failed to do a stateful update, full update, including kernel update,
766 will be applied to the DUT.
767
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800768 Once a host enters firmware_install its fw[ro|rw]_version label will
769 be removed. After the firmware is updated successfully, a new
770 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700771
772 @param build: The build version to which we want to provision the
773 firmware of the machine,
774 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800775 @param rw_only: True to only install firmware to its RW portions. Keep
776 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700777 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800778 @param local_tarball: Path to local firmware image for installing
779 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800780 @param verify_version: True to verify EC and BIOS versions after
781 programming firmware, default is False.
Brent Petersonc70a1832020-01-24 15:54:35 -0800782 @param try_scp: False to always program using servo, true to try copying
783 the firmware and programming from the DUT.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700784
785 TODO(dshi): After bug 381718 is fixed, update here with corresponding
786 exceptions that could be raised.
787
788 """
789 if not self.servo:
790 raise error.TestError('Host %s does not have servo.' %
791 self.hostname)
792
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700793 # Get the DUT board name from AFE.
794 info = self.host_info_store.get()
795 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700796 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800797
798 if board is None or board == '':
799 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700800
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700801 if model is None or model == '':
802 model = self.get_platform_from_fwid()
803
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800804 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700805 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800806 if not local_tarball:
Garry Wangad2a1712020-03-26 15:06:43 -0700807 logging.info('Will install firmware from build %s.', build)
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800808
809 ds = dev_server.ImageServer.resolve(build, self.hostname)
810 ds.stage_artifacts(build, ['firmware'])
811
812 if not dest:
813 tmpd = autotemp.tempdir(unique_id='fwimage')
814 dest = tmpd.name
815
816 # Download firmware image
Dan Shi9cb0eec2014-06-03 09:04:50 -0700817 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700818 local_tarball = os.path.join(dest, os.path.basename(fwurl))
xixuan4e116822016-11-17 15:32:10 -0800819 ds.download_file(fwurl, local_tarball)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700820
Brent Peterson1cb623a2020-01-09 13:14:28 -0800821 # Extract EC image from tarball
822 logging.info('Extracting EC image.')
823 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
824
825 # Extract BIOS image from tarball
826 logging.info('Extracting BIOS image.')
827 bios_image = self.servo.extract_bios_image(board, model, local_tarball)
828
Brent Petersonc70a1832020-01-24 15:54:35 -0800829 # Clear firmware version labels
830 self._clear_fw_version_labels(rw_only)
831
832 # Install firmware from local tarball
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800833 try:
Brent Petersonc70a1832020-01-24 15:54:35 -0800834 # Check if DUT is available and copying to DUT is enabled
835 if self.is_up() and try_scp:
836 # DUT is available, make temp firmware directory to store images
837 logging.info('Making temp folder.')
838 dest_folder = '/tmp/firmware'
839 self.run('mkdir -p ' + dest_folder)
840
Brent Petersonc70a1832020-01-24 15:54:35 -0800841 # Send BIOS firmware image to DUT
842 logging.info('Sending BIOS firmware.')
843 dest_bios_path = os.path.join(dest_folder,
844 os.path.basename(bios_image))
845 self.send_file(bios_image, dest_bios_path)
846
Brent Peterson669edf42020-02-07 15:07:54 -0800847 # Initialize firmware update command for BIOS image
848 fw_cmd = self._FW_UPDATE_CMD % (dest_bios_path,
Brent Petersonc70a1832020-01-24 15:54:35 -0800849 '--wp=1' if rw_only else '')
Brent Peterson669edf42020-02-07 15:07:54 -0800850
851 # Send EC firmware image to DUT when EC image was found
852 if ec_image:
853 logging.info('Sending EC firmware.')
854 dest_ec_path = os.path.join(dest_folder,
855 os.path.basename(ec_image))
856 self.send_file(ec_image, dest_ec_path)
857
858 # Add EC image to firmware update command
859 fw_cmd += ' -e %s' % dest_ec_path
860
861 # Update firmware on DUT
862 logging.info('Updating firmware.')
Brent Petersonc70a1832020-01-24 15:54:35 -0800863 self.run(fw_cmd)
864 else:
865 # Host is not available, program firmware using servo
Brent Peterson669edf42020-02-07 15:07:54 -0800866 if ec_image:
867 self.servo.program_ec(ec_image, rw_only)
Brent Petersonc70a1832020-01-24 15:54:35 -0800868 self.servo.program_bios(bios_image, rw_only)
869 if utils.host_is_in_lab_zone(self.hostname):
870 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800871
872 # Reboot and wait for DUT after installing firmware
873 logging.info('Rebooting DUT.')
874 self.servo.get_power_state_controller().reset()
875 time.sleep(self.servo.BOOT_DELAY)
876 self.test_wait_for_boot()
877
878 # When enabled verify EC and BIOS firmware version after programming
879 if verify_version:
Brent Peterson669edf42020-02-07 15:07:54 -0800880 # Check programmed EC firmware when EC image was found
881 if ec_image:
882 logging.info('Checking EC firmware version.')
883 dest_ec_version = self.get_ec_version()
Brent Peterson8039b472020-02-14 10:51:23 -0800884 ec_version_prefix = dest_ec_version.split('_', 1)[0]
885 ec_regex = self._EC_REGEX % ec_version_prefix
Brent Peterson669edf42020-02-07 15:07:54 -0800886 image_ec_version = self.get_version_from_image(ec_image,
Brent Peterson8039b472020-02-14 10:51:23 -0800887 ec_regex)
Brent Peterson669edf42020-02-07 15:07:54 -0800888 if dest_ec_version != image_ec_version:
889 raise error.TestFail(
890 'Failed to update EC RO, version %s (expected %s)' %
891 (dest_ec_version, image_ec_version))
Brent Peterson1cb623a2020-01-09 13:14:28 -0800892
893 # Check programmed BIOS firmware against expected version
894 logging.info('Checking BIOS firmware version.')
895 dest_bios_version = self.get_firmware_version()
896 bios_version_prefix = dest_bios_version.split('.', 1)[0]
897 bios_regex = self._BIOS_REGEX % bios_version_prefix
898 image_bios_version = self.get_version_from_image(bios_image,
899 bios_regex)
900 if dest_bios_version != image_bios_version:
901 raise error.TestFail(
902 'Failed to update BIOS RO, version %s (expected %s)' %
903 (dest_bios_version, image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700904 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700905 if tmpd:
906 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700907
908
beepsf079cfb2013-09-18 17:49:51 -0700909 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
910 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500911 """
912 Re-install the OS on the DUT by:
913 1) installing a test image on a USB storage device attached to the Servo
914 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800915 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700916 3) installing the image with chromeos-install.
917
Scott Zawalski62bacae2013-03-05 10:40:32 -0500918 @param image_url: If specified use as the url to install on the DUT.
919 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700920 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
921 Factory images need a longer usb_boot_timeout than regular
922 cros images.
923 @param install_timeout: The timeout to use when installing the chromeos
924 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800925
Scott Zawalski62bacae2013-03-05 10:40:32 -0500926 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700927
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800928 """
Garry Wang7b0e1b72020-03-25 19:08:59 -0700929 if image_url:
930 logging.info('Downloading image to USB, then booting from it.'
931 ' Usb boot timeout = %s', usb_boot_timeout)
932 else:
933 logging.info('Booting from USB directly. Usb boot timeout = %s',
934 usb_boot_timeout)
935
936 metrics_field = {'download': bool(image_url)}
937 metrics.Counter(
938 'chromeos/autotest/provision/servo_install/download_image'
939 ).increment(fields=metrics_field)
940
Allen Li48a13fe2016-11-22 14:10:40 -0800941 with metrics.SecondsTimer(
942 'chromeos/autotest/provision/servo_install/boot_duration'):
943 self.servo.install_recovery_image(image_url)
944 if not self.wait_up(timeout=usb_boot_timeout):
945 raise hosts.AutoservRepairError(
946 'DUT failed to boot from USB after %d seconds' %
Garry Wang9ced7aa2020-04-10 17:26:35 -0700947 usb_boot_timeout, 'failed_to_boot_pre_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500948
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800949 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
950 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800951 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800952 try:
953 self.run('chromeos-tpm-recovery')
954 except error.AutoservRunError:
955 logging.warn('chromeos-tpm-recovery is too old.')
956
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800957
Allen Li48a13fe2016-11-22 14:10:40 -0800958 with metrics.SecondsTimer(
959 'chromeos/autotest/provision/servo_install/install_duration'):
960 logging.info('Installing image through chromeos-install.')
Garry Wang033a31e2020-04-10 17:20:49 -0700961 try:
962 self.run('chromeos-install --yes',timeout=install_timeout)
963 self.halt()
964 finally:
965 # We need reset the DUT no matter re-install success or not,
966 # as we don't want leave the DUT in boot from usb state.
967 logging.info('Power cycling DUT through servo.')
968 self.servo.get_power_state_controller().power_off()
969 self.servo.switch_usbkey('off')
970 # N.B. The Servo API requires that we use power_on() here
971 # for two reasons:
972 # 1) After turning on a DUT in recovery mode, you must turn
973 # it off and then on with power_on() once more to
974 # disable recovery mode (this is a Parrot specific
975 # requirement).
976 # 2) After power_off(), the only way to turn on is with
977 # power_on() (this is a Storm specific requirement).
978 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -0700979
980 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -0800981 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
Garry Wang9ced7aa2020-04-10 17:26:35 -0700982 raise hosts.AutoservRepairError('DUT failed to reboot installed '
983 'test image after %d seconds' %
984 self.BOOT_TIMEOUT,
985 'failed_to_boot_post_install')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500986
987
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700988 def set_servo_host(self, host, servo_state = None):
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700989 """Set our servo host member, and associated servo.
990
991 @param host Our new `ServoHost`.
992 """
993 self._servo_host = host
994 if self._servo_host is not None:
995 self.servo = self._servo_host.get_servo()
Otabek Kasimova7ba91a2020-03-09 08:31:01 -0700996 servo_state = self._servo_host.get_servo_state()
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700997 else:
998 self.servo = None
999
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001000 self.set_servo_state(servo_state)
1001
Richard Barnette4aeb01c2018-09-20 09:36:12 -07001002
Richard Barnette9a26ad62016-06-10 12:03:08 -07001003 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -07001004 """
Richard Barnette9a26ad62016-06-10 12:03:08 -07001005 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -07001006
Richard Barnette9a26ad62016-06-10 12:03:08 -07001007 If the servo object is missing, attempt to repair the servo
1008 host. Repair failures are passed back to the caller.
1009
1010 @raise AutoservError: If there is no servo host for this CrOS
1011 host.
1012 """
1013 if self.servo:
1014 return
1015 if not self._servo_host:
1016 raise error.AutoservError('No servo host for %s.' %
1017 self.hostname)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001018 try:
1019 self._servo_host.repair()
1020 except:
1021 raise
1022 finally:
1023 self.set_servo_host(self._servo_host)
1024
1025
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001026 def set_servo_state(self, servo_state):
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001027 """Set servo info labels to dut host_info"""
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001028 if servo_state is not None:
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001029 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001030 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001031 old_state = host_info.get_label_value(servo_state_prefix)
1032 if old_state == servo_state:
1033 # do not need update
1034 return
1035 host_info.set_version_label(servo_state_prefix, servo_state)
Otabek Kasimovcc9738e2020-02-14 16:17:15 -08001036 self.host_info_store.commit(host_info)
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001037 logging.info('ServoHost: servo_state updated to %s (previous: %s)',
1038 servo_state, old_state)
Dan Shi90466352015-09-22 15:01:05 -07001039
1040
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001041 def get_servo_state(self):
1042 host_info = self.host_info_store.get()
Garry Wang11b5e872020-03-11 15:14:08 -07001043 servo_state_prefix = servo_constants.SERVO_STATE_LABEL_PREFIX
Otabek Kasimova7ba91a2020-03-09 08:31:01 -07001044 return host_info.get_label_value(servo_state_prefix)
1045
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -08001046 def repair(self):
1047 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -08001048
1049 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -08001050 not call back to the parent class, but instead relies on
1051 `self._repair_strategy` to coordinate the verification and
1052 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -08001053 """
Richard Barnetteabbdc252018-07-26 16:57:42 -07001054 message = 'Beginning repair for host %s board %s model %s'
1055 info = self.host_info_store.get()
1056 message %= (self.hostname, info.board, info.model)
1057 self.record('INFO', None, None, message)
J. Richard Barnette91137f02016-03-10 16:52:26 -08001058 self._repair_strategy.repair(self)
Scott Zawalski89c44dd2013-02-26 09:28:02 -05001059
Richard Barnette82c35912012-11-20 10:09:10 -08001060
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001061 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -08001062 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -07001063 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +08001064
Shijin Abraham783a7dd2020-02-14 15:36:11 -08001065 if self._chameleon_host:
1066 self._chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -08001067
1068 if self._servo_host:
1069 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -07001070
1071
Dan Shi49ca0932014-11-14 11:22:27 -08001072 def get_power_supply_info(self):
1073 """Get the output of power_supply_info.
1074
1075 power_supply_info outputs the info of each power supply, e.g.,
1076 Device: Line Power
1077 online: no
1078 type: Mains
1079 voltage (V): 0
1080 current (A): 0
1081 Device: Battery
1082 state: Discharging
1083 percentage: 95.9276
1084 technology: Li-ion
1085
1086 Above output shows two devices, Line Power and Battery, with details of
1087 each device listed. This function parses the output into a dictionary,
1088 with key being the device name, and value being a dictionary of details
1089 of the device info.
1090
1091 @return: The dictionary of power_supply_info, e.g.,
1092 {'Line Power': {'online': 'yes', 'type': 'main'},
1093 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -08001094 @raise error.AutoservRunError if power_supply_info tool is not found in
1095 the DUT. Caller should handle this error to avoid false failure
1096 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -08001097 """
1098 result = self.run('power_supply_info').stdout.strip()
1099 info = {}
1100 device_name = None
1101 device_info = {}
1102 for line in result.split('\n'):
1103 pair = [v.strip() for v in line.split(':')]
1104 if len(pair) != 2:
1105 continue
1106 if pair[0] == 'Device':
1107 if device_name:
1108 info[device_name] = device_info
1109 device_name = pair[1]
1110 device_info = {}
1111 else:
1112 device_info[pair[0]] = pair[1]
1113 if device_name and not device_name in info:
1114 info[device_name] = device_info
1115 return info
1116
1117
1118 def get_battery_percentage(self):
1119 """Get the battery percentage.
1120
1121 @return: The percentage of battery level, value range from 0-100. Return
1122 None if the battery info cannot be retrieved.
1123 """
1124 try:
1125 info = self.get_power_supply_info()
1126 logging.info(info)
1127 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -08001128 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -08001129 return None
1130
1131
Philip Chenaf69ead2020-03-27 13:06:42 -07001132 def get_battery_state(self):
1133 """Get the battery charging state.
1134
1135 @return: A string representing the battery charging state. It can be
1136 'Charging', 'Fully charged', or 'Discharging'.
1137 """
1138 try:
1139 info = self.get_power_supply_info()
1140 logging.info(info)
1141 return info['Battery']['state']
1142 except (KeyError, ValueError, error.AutoservRunError):
1143 return None
1144
1145
Daniel Campello8ca25c22019-12-13 16:48:26 -07001146 def get_battery_display_percentage(self):
1147 """Get the battery display percentage.
1148
1149 @return: The display percentage of battery level, value range from
1150 0-100. Return None if the battery info cannot be retrieved.
1151 """
1152 try:
1153 info = self.get_power_supply_info()
1154 logging.info(info)
1155 return float(info['Battery']['display percentage'])
1156 except (KeyError, ValueError, error.AutoservRunError):
1157 return None
1158
1159
Dan Shi49ca0932014-11-14 11:22:27 -08001160 def is_ac_connected(self):
1161 """Check if the dut has power adapter connected and charging.
1162
1163 @return: True if power adapter is connected and charging.
1164 """
1165 try:
1166 info = self.get_power_supply_info()
1167 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -08001168 except (KeyError, error.AutoservRunError):
1169 return None
Dan Shi49ca0932014-11-14 11:22:27 -08001170
1171
Simran Basi5e6339a2013-03-21 11:34:32 -07001172 def _cleanup_poweron(self):
1173 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001174 info = self.host_info_store.get()
1175 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001176 return
1177 logging.debug('This host has recently interacted with the RPM'
1178 ' Infrastructure. Ensuring power is on.')
1179 try:
1180 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001181 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001182 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001183 logging.error('Failed to turn Power On for this host after '
1184 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001185
1186 battery_percentage = self.get_battery_percentage()
Dan Shif01ebe22014-12-05 13:10:57 -08001187 if battery_percentage and battery_percentage < 50:
Dan Shi49ca0932014-11-14 11:22:27 -08001188 raise
1189 elif self.is_ac_connected():
1190 logging.info('The device has power adapter connected and '
1191 'charging. No need to try to turn RPM on '
1192 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001193 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001194 logging.info('Battery level is now at %s%%. The device may '
1195 'still have enough power to run test, so no '
1196 'exception will be raised.', battery_percentage)
1197
Simran Basi5e6339a2013-03-21 11:34:32 -07001198
Garry Wangad4d4fd2019-01-30 17:00:38 -08001199 def _remove_rpm_changed_tag(self):
1200 info = self.host_info_store.get()
1201 del info.attributes[self._RPM_OUTLET_CHANGED]
1202 self.host_info_store.commit(info)
1203
1204
1205 def _add_rpm_changed_tag(self):
1206 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001207 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001208 self.host_info_store.commit(info)
1209
1210
1211
beepsc87ff602013-07-31 21:53:00 -07001212 def _is_factory_image(self):
1213 """Checks if the image on the DUT is a factory image.
1214
1215 @return: True if the image on the DUT is a factory image.
1216 False otherwise.
1217 """
1218 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1219 return result.exit_status == 0
1220
1221
1222 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001223 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001224
1225 @raises: FactoryImageCheckerException for factory images, since
1226 we cannot attempt to restart ui on them.
1227 error.AutoservRunError for any other type of error that
1228 occurs while restarting ui.
1229 """
1230 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001231 raise FactoryImageCheckerException('Cannot restart ui on factory '
1232 'images')
beepsc87ff602013-07-31 21:53:00 -07001233
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001234 # TODO(jrbarnette): The command to stop/start the ui job
1235 # should live inside cros_ui, too. However that would seem
1236 # to imply interface changes to the existing start()/restart()
1237 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001238 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001239 self.run('stop ui; start ui')
1240 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001241
1242
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001243 def _start_powerd_if_needed(self):
1244 """Start powerd if it isn't already running."""
1245 self.run('start powerd', ignore_status=True)
1246
1247
xixuana3bbc422017-05-04 15:57:21 -07001248 def _get_lsb_release_content(self):
1249 """Return the content of lsb-release file of host."""
1250 return self.run(
1251 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1252
1253
Dan Shi549fb822015-03-24 18:01:11 -07001254 def get_release_version(self):
1255 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1256
1257 @returns The version string in lsb-release, under attribute
1258 CHROMEOS_RELEASE_VERSION.
1259 """
Dan Shi549fb822015-03-24 18:01:11 -07001260 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001261 lsb_release_content=self._get_lsb_release_content())
1262
1263
Don Garrettb9f35802018-01-22 18:25:40 -08001264 def get_release_builder_path(self):
1265 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1266
1267 @returns The version string in lsb-release, under attribute
1268 CHROMEOS_RELEASE_BUILDER_PATH.
1269 """
1270 return lsbrelease_utils.get_chromeos_release_builder_path(
1271 lsb_release_content=self._get_lsb_release_content())
1272
1273
xixuana3bbc422017-05-04 15:57:21 -07001274 def get_chromeos_release_milestone(self):
1275 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1276 from lsb-release.
1277
1278 @returns The version string in lsb-release, under attribute
1279 CHROMEOS_RELEASE_BUILD_TYPE.
1280 """
1281 return lsbrelease_utils.get_chromeos_release_milestone(
1282 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001283
1284
1285 def verify_cros_version_label(self):
1286 """ Make sure host's cros-version label match the actual image in dut.
1287
1288 Remove any cros-version: label that doesn't match that installed in
1289 the dut.
1290
1291 @param raise_error: Set to True to raise exception if any mismatch found
1292
1293 @raise error.AutoservError: If any mismatch between cros-version label
1294 and the build installed in dut is found.
1295 """
Prathmesh Prabhuce2da3a2019-10-04 11:54:51 -07001296 # crbug.com/1007333: This check is being removed.
1297 return True
Dan Shi549fb822015-03-24 18:01:11 -07001298
1299
Laurence Goodby778c9a42017-05-24 19:24:07 -07001300 def cleanup_services(self):
1301 """Reinitializes the device for cleanup.
1302
1303 Subclasses may override this to customize the cleanup method.
1304
1305 To indicate failure of the reset, the implementation may raise
1306 any of:
1307 error.AutoservRunError
1308 error.AutotestRunError
1309 FactoryImageCheckerException
1310
1311 @raises error.AutoservRunError
1312 @raises error.AutotestRunError
1313 @raises error.FactoryImageCheckerException
1314 """
1315 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001316 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001317
1318
beepsc87ff602013-07-31 21:53:00 -07001319 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001320 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001321 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001322 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001323 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001324 except (error.AutotestRunError, error.AutoservRunError,
1325 FactoryImageCheckerException):
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001326 logging.warning('Unable to restart ui.')
Namyoon Woo33f38852020-04-13 17:26:58 -07001327
Otabek Kasimovc0d77e82020-03-27 15:01:57 -07001328 # cleanup routines, i.e. reboot the machine.
1329 super(CrosHost, self).cleanup()
1330
Simran Basi5e6339a2013-03-21 11:34:32 -07001331 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001332 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001333 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001334 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001335
1336
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001337 def reboot(self, **dargs):
1338 """
1339 This function reboots the site host. The more generic
1340 RemoteHost.reboot() performs sync and sleeps for 5
1341 seconds. This is not necessary for Chrome OS devices as the
1342 sync should be finished in a short time during the reboot
1343 command.
1344 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001345 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001346 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001347 dargs['reboot_cmd'] = ('sleep 1; '
1348 'reboot & sleep %d; '
1349 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001350 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001351 if 'fastsync' not in dargs:
1352 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001353
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001354 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001355 # Record who called us
1356 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001357 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001358 'dut_host_name' : self.hostname,
1359 'success' : True}
1360 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001361 'caller' : "%s:%s" % (orig.co_filename,
1362 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001363 'success' : True,
1364 'error' : ''}
1365
Vincent Palatin80780b22016-07-27 16:02:37 +02001366 t0 = time.time()
1367 try:
1368 super(CrosHost, self).reboot(**dargs)
1369 except Exception as e:
1370 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001371 metric_debug_fields['success'] = False
1372 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001373 raise
1374 finally:
1375 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001376 metrics.Counter(
1377 'chromeos/autotest/autoserv/reboot_count').increment(
1378 fields=metric_fields)
1379 metrics.Counter(
1380 'chromeos/autotest/autoserv/reboot_debug').increment(
1381 fields=metric_debug_fields)
1382 metrics.SecondsDistribution(
1383 'chromeos/autotest/autoserv/reboot_duration').add(
1384 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001385
1386
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001387 def suspend(self, suspend_time=60, delay_seconds=0,
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001388 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001389 """
1390 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001391
1392 @param suspend_time: How long to suspend as integer seconds.
1393 @param suspend_cmd: Suspend command to execute.
1394 @param allow_early_resume: If False and if device resumes before
1395 |suspend_time|, throw an error.
1396
1397 @exception AutoservSuspendError Host resumed earlier than
1398 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001399 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001400
1401 if suspend_cmd is None:
1402 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001403 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001404 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
Kalin Stoyanov83d9caa2020-01-31 16:58:27 -08001405 'powerd_dbus_suspend --delay=%d' % delay_seconds])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001406 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1407 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001408
1409
Simran Basiec564392014-08-25 16:48:09 -07001410 def upstart_status(self, service_name):
1411 """Check the status of an upstart init script.
1412
1413 @param service_name: Service to look up.
1414
1415 @returns True if the service is running, False otherwise.
1416 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001417 return 'start/running' in self.run('status %s' % service_name,
1418 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001419
Tom Hughese9552342018-12-18 14:29:25 -08001420 def upstart_stop(self, service_name):
1421 """Stops an upstart job if it's running.
1422
1423 @param service_name: Service to stop
1424
1425 @returns True if service has been stopped or was already stopped
1426 False otherwise.
1427 """
1428 if not self.upstart_status(service_name):
1429 return True
1430
1431 result = self.run('stop %s' % service_name, ignore_status=True)
1432 if result.exit_status != 0:
1433 return False
1434 return True
1435
1436 def upstart_restart(self, service_name):
1437 """Restarts (or starts) an upstart job.
1438
1439 @param service_name: Service to start/restart
1440
1441 @returns True if service has been started/restarted, False otherwise.
1442 """
1443 cmd = 'start'
1444 if self.upstart_status(service_name):
1445 cmd = 'restart'
1446 cmd = cmd + ' %s' % service_name
1447 result = self.run(cmd)
1448 if result.exit_status != 0:
1449 return False
1450 return True
Simran Basiec564392014-08-25 16:48:09 -07001451
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001452 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001453 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001454
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001455 Tests for the following conditions:
1456 1. All conditions tested by the parent version of this
1457 function.
1458 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001459 3. Sufficient space in /mnt/stateful_partition/encrypted.
1460 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001461
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001462 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001463 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001464 default_kilo_inodes_required = CONFIG.get_config_value(
1465 'SERVER', 'kilo_inodes_required', type=int, default=100)
1466 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1467 kilo_inodes_required = CONFIG.get_config_value(
1468 'SERVER', 'kilo_inodes_required_%s' % board,
1469 type=int, default=default_kilo_inodes_required)
1470 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001471 self.check_diskspace(
1472 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001473 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001474 'SERVER', 'gb_diskspace_required', type=float,
1475 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001476 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1477 # Not all targets build with encrypted stateful support.
1478 if self.path_exists(encrypted_stateful_path):
1479 self.check_diskspace(
1480 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001481 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001482 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1483 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001484
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001485 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001486
beepsc87ff602013-07-31 21:53:00 -07001487 # Factory images don't run update engine,
1488 # goofy controls dbus on these DUTs.
1489 if not self._is_factory_image():
1490 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001491
Dan Shi549fb822015-03-24 18:01:11 -07001492 self.verify_cros_version_label()
1493
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001494
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001495 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1496 def wait_for_system_services(self):
1497 """Waits for system-services to be running.
1498
1499 Sometimes, update_engine will take a while to update firmware, so we
1500 should give this some time to finish. See crbug.com/765686#c38 for
1501 details.
1502 """
1503 if not self.upstart_status('system-services'):
1504 raise error.AutoservError('Chrome failed to reach login. '
1505 'System services not running.')
1506
1507
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001508 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001509 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001510 message = 'Beginning verify for host %s board %s model %s'
1511 info = self.host_info_store.get()
1512 message %= (self.hostname, info.board, info.model)
1513 self.record('INFO', None, None, message)
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001514 self._repair_strategy.verify(self)
1515
1516
Fang Deng96667ca2013-08-01 17:46:18 -07001517 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001518 connect_timeout=None, alive_interval=None,
1519 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001520 """Override default make_ssh_command to use options tuned for Chrome OS.
1521
1522 Tuning changes:
1523 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1524 connection failure. Consistency with remote_access.sh.
1525
Samuel Tan2ce155b2015-06-23 18:24:38 -07001526 - ServerAliveInterval=900; which causes SSH to ping connection every
1527 900 seconds. In conjunction with ServerAliveCountMax ensures
1528 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001529 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001530 the test completed successfully. Later increased from 180 seconds to
1531 900 seconds to account for tests where the DUT is suspended for
1532 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001533
1534 - ServerAliveCountMax=3; consistency with remote_access.sh.
1535
1536 - ConnectAttempts=4; reduce flakiness in connection errors;
1537 consistency with remote_access.sh.
1538
1539 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1540 Host keys change with every new installation, don't waste
1541 memory/space saving them.
1542
1543 - SSH protocol forced to 2; needed for ServerAliveInterval.
1544
1545 @param user User name to use for the ssh connection.
1546 @param port Port on the target host to use for ssh connection.
1547 @param opts Additional options to the ssh command.
1548 @param hosts_file Ignored.
1549 @param connect_timeout Ignored.
1550 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001551 @param alive_count_max Ignored.
1552 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001553 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001554 options = ' '.join([opts, '-o Protocol=2'])
1555 return super(CrosHost, self).make_ssh_command(
1556 user=user, port=port, opts=options, hosts_file='/dev/null',
1557 connect_timeout=30, alive_interval=900, alive_count_max=3,
1558 connection_attempts=4)
1559
1560
Jason Abeleb6f924f2013-11-13 16:01:54 -08001561 def syslog(self, message, tag='autotest'):
1562 """Logs a message to syslog on host.
1563
1564 @param message String message to log into syslog
1565 @param tag String tag prefix for syslog
1566
1567 """
1568 self.run('logger -t "%s" "%s"' % (tag, message))
1569
1570
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001571 def _ping_check_status(self, status):
1572 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001573
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001574 @param status Check the ping status against this value.
1575 @return True iff `status` and the result of ping are the same
1576 (i.e. both True or both False).
1577
1578 """
1579 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1580 return not (status ^ (ping_val == 0))
1581
1582 def _ping_wait_for_status(self, status, timeout):
1583 """Wait for the host to have a given status (UP or DOWN).
1584
1585 Status is checked by polling. Polling will not last longer
1586 than the number of seconds in `timeout`. The polling
1587 interval will be long enough that only approximately
1588 _PING_WAIT_COUNT polling cycles will be executed, subject
1589 to a maximum interval of about one minute.
1590
1591 @param status Waiting will stop immediately if `ping` of the
1592 host returns this status.
1593 @param timeout Poll for at most this many seconds.
1594 @return True iff the host status from `ping` matched the
1595 requested status at the time of return.
1596
1597 """
1598 # _ping_check_status() takes about 1 second, hence the
1599 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001600 # FIXME: if the ping command errors then _ping_check_status()
1601 # returns instantly. If timeout is also smaller than twice
1602 # _PING_WAIT_COUNT then the while loop below forks many
1603 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1604 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1605 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001606 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1607 end_time = time.time() + timeout
1608 while time.time() <= end_time:
1609 if self._ping_check_status(status):
1610 return True
1611 if poll_interval > 0:
1612 time.sleep(poll_interval)
1613
1614 # The last thing we did was sleep(poll_interval), so it may
1615 # have been too long since the last `ping`. Check one more
1616 # time, just to be sure.
1617 return self._ping_check_status(status)
1618
1619 def ping_wait_up(self, timeout):
1620 """Wait for the host to respond to `ping`.
1621
1622 N.B. This method is not a reliable substitute for
1623 `wait_up()`, because a host that responds to ping will not
1624 necessarily respond to ssh. This method should only be used
1625 if the target DUT can be considered functional even if it
1626 can't be reached via ssh.
1627
1628 @param timeout Minimum time to allow before declaring the
1629 host to be non-responsive.
1630 @return True iff the host answered to ping before the timeout.
1631
1632 """
1633 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001634
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001635 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001636 """Wait until the host no longer responds to `ping`.
1637
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001638 This function can be used as a slightly faster version of
1639 `wait_down()`, by avoiding potentially long ssh timeouts.
1640
1641 @param timeout Minimum time to allow for the host to become
1642 non-responsive.
1643 @return True iff the host quit answering ping before the
1644 timeout.
1645
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001646 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001647 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001648
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001649 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001650 """Wait for the client to enter low-power sleep mode.
1651
1652 The test for "is asleep" can't distinguish a system that is
1653 powered off; to confirm that the unit was asleep, it is
1654 necessary to force resume, and then call
1655 `test_wait_for_resume()`.
1656
1657 This function is expected to be called from a test as part
1658 of a sequence like the following:
1659
1660 ~~~~~~~~
1661 boot_id = host.get_boot_id()
1662 # trigger sleep on the host
1663 host.test_wait_for_sleep()
1664 # trigger resume on the host
1665 host.test_wait_for_resume(boot_id)
1666 ~~~~~~~~
1667
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001668 @param sleep_timeout time limit in seconds to allow the host sleep.
1669
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001670 @exception TestFail The host did not go to sleep within
1671 the allowed time.
1672 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001673 if sleep_timeout is None:
1674 sleep_timeout = self.SLEEP_TIMEOUT
1675
1676 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001677 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001678 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001679
1680
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001681 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001682 """Wait for the client to resume from low-power sleep mode.
1683
1684 The `old_boot_id` parameter should be the value from
1685 `get_boot_id()` obtained prior to entering sleep mode. A
1686 `TestFail` exception is raised if the boot id changes.
1687
1688 See @ref test_wait_for_sleep for more on this function's
1689 usage.
1690
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001691 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001692 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001693 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001694
1695 @exception TestFail The host did not respond within the
1696 allowed time.
1697 @exception TestFail The host responded, but the boot id test
1698 indicated a reboot rather than a sleep
1699 cycle.
1700 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001701 if resume_timeout is None:
1702 resume_timeout = self.RESUME_TIMEOUT
1703
1704 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001705 raise error.TestFail(
1706 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001707 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001708 else:
1709 new_boot_id = self.get_boot_id()
1710 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001711 logging.error('client rebooted (old boot %s, new boot %s)',
1712 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001713 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001714 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001715
1716
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001717 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001718 """Wait for the client to shut down.
1719
1720 The test for "has shut down" can't distinguish a system that
1721 is merely asleep; to confirm that the unit was down, it is
1722 necessary to force boot, and then call test_wait_for_boot().
1723
1724 This function is expected to be called from a test as part
1725 of a sequence like the following:
1726
1727 ~~~~~~~~
1728 boot_id = host.get_boot_id()
1729 # trigger shutdown on the host
1730 host.test_wait_for_shutdown()
1731 # trigger boot on the host
1732 host.test_wait_for_boot(boot_id)
1733 ~~~~~~~~
1734
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001735 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001736 @exception TestFail The host did not shut down within the
1737 allowed time.
1738 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001739 if shutdown_timeout is None:
1740 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1741
1742 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001743 raise error.TestFail(
1744 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001745 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001746
1747
1748 def test_wait_for_boot(self, old_boot_id=None):
1749 """Wait for the client to boot from cold power.
1750
1751 The `old_boot_id` parameter should be the value from
1752 `get_boot_id()` obtained prior to shutting down. A
1753 `TestFail` exception is raised if the boot id does not
1754 change. The boot id test is omitted if `old_boot_id` is not
1755 specified.
1756
1757 See @ref test_wait_for_shutdown for more on this function's
1758 usage.
1759
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001760 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001761 shut down.
1762
1763 @exception TestFail The host did not respond within the
1764 allowed time.
1765 @exception TestFail The host responded, but the boot id test
1766 indicated that there was no reboot.
1767 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001768 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001769 raise error.TestFail(
1770 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001771 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001772 elif old_boot_id:
1773 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001774 logging.error('client not rebooted (boot %s)',
1775 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001776 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001777 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001778
1779
1780 @staticmethod
1781 def check_for_rpm_support(hostname):
1782 """For a given hostname, return whether or not it is powered by an RPM.
1783
Simran Basi1df55112013-09-06 11:25:09 -07001784 @param hostname: hostname to check for rpm support.
1785
Simran Basid5e5e272012-09-24 15:23:59 -07001786 @return None if this host does not follows the defined naming format
1787 for RPM powered DUT's in the lab. If it does follow the format,
1788 it returns a regular expression MatchObject instead.
1789 """
Fang Dengbaff9082015-01-06 13:46:15 -08001790 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001791
1792
1793 def has_power(self):
1794 """For this host, return whether or not it is powered by an RPM.
1795
1796 @return True if this host is in the CROS lab and follows the defined
1797 naming format.
1798 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001799 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001800
1801
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001802 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001803 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001804
1805 @param state Specifies which power state to set to DUT
1806 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001807 use. By default "RPM" or "CCD" will be used based
1808 on servo type. Valid values from
1809 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001810
1811 """
1812 ACCEPTABLE_STATES = ['ON', 'OFF']
1813
Garry Wang5e5538a2019-04-08 15:36:18 -07001814 if not power_method:
1815 power_method = self.get_default_power_method()
1816
1817 state = state.upper()
1818 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001819 raise error.TestError('State must be one of: %s.'
1820 % (ACCEPTABLE_STATES,))
1821
1822 if power_method == self.POWER_CONTROL_SERVO:
1823 logging.info('Setting servo port J10 to %s', state)
1824 self.servo.set('prtctl3_pwren', state.lower())
1825 time.sleep(self._USB_POWER_TIMEOUT)
1826 elif power_method == self.POWER_CONTROL_MANUAL:
1827 logging.info('You have %d seconds to set the AC power to %s.',
1828 self._POWER_CYCLE_TIMEOUT, state)
1829 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001830 elif power_method == self.POWER_CONTROL_CCD:
1831 servo_role = 'src' if state == 'ON' else 'snk'
1832 logging.info('servo ccd power pass through detected,'
1833 ' changing servo_role to %s.', servo_role)
1834 self.servo.set_servo_v4_role(servo_role)
1835 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001836 # Make sure we don't leave DUT with no power(servo_role=snk)
1837 # when DUT is not pingable, as we raise a exception here
1838 # that may break a power cycle in the middle.
1839 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001840 raise error.AutoservError(
1841 'DUT failed to regain network connection after %d seconds.'
1842 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001843 else:
1844 if not self.has_power():
1845 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001846 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07001847 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07001848
1849
Garry Wang5e5538a2019-04-08 15:36:18 -07001850 def power_off(self, power_method=None):
1851 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001852
1853 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001854 use. By default "RPM" or "CCD" will be used based
1855 on servo type. Valid values from
1856 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001857
1858 """
1859 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07001860
1861
Garry Wang5e5538a2019-04-08 15:36:18 -07001862 def power_on(self, power_method=None):
1863 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001864
1865 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001866 use. By default "RPM" or "CCD" will be used based
1867 on servo type. Valid values from
1868 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001869
1870 """
1871 self._set_power('ON', power_method)
1872
1873
Garry Wang5e5538a2019-04-08 15:36:18 -07001874 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001875 """Cycle power to this host by turning it OFF, then ON.
1876
1877 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001878 use. By default "RPM" or "CCD" will be used based
1879 on servo type. Valid values from
1880 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001881
1882 """
Garry Wang5e5538a2019-04-08 15:36:18 -07001883 if not power_method:
1884 power_method = self.get_default_power_method()
1885
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001886 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07001887 self.POWER_CONTROL_MANUAL,
1888 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001889 self.power_off(power_method=power_method)
1890 time.sleep(self._POWER_CYCLE_TIMEOUT)
1891 self.power_on(power_method=power_method)
1892 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08001893 self._add_rpm_changed_tag()
1894 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07001895
1896
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001897 def get_platform_from_fwid(self):
1898 """Determine the platform from the crossystem fwid.
1899
1900 @returns a string representing this host's platform.
1901 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06001902 # Look at the firmware for non-unibuild cases or if cros_config fails.
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001903 crossystem = utils.Crossystem(self)
1904 crossystem.init()
1905 # Extract fwid value and use the leading part as the platform id.
1906 # fwid generally follow the format of {platform}.{firmware version}
1907 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
1908 platform = crossystem.fwid().split('.')[0].lower()
1909 # Newer platforms start with 'Google_' while the older ones do not.
1910 return platform.replace('google_', '')
1911
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001912
Simran Basic6f1f7a2012-10-16 10:47:46 -07001913 def get_platform(self):
1914 """Determine the correct platform label for this host.
1915
1916 @returns a string representing this host's platform.
1917 """
C Shapiroed87c6f2018-04-19 09:13:58 -06001918 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1919 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06001920 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001921 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
Greg Edelstona7b05d12020-04-01 16:00:51 -06001922 platform = self.get_model_from_cros_config()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001923 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07001924
1925
Greg Edelstona7b05d12020-04-01 16:00:51 -06001926 def get_model_from_cros_config(self):
1927 """Get the host model from cros_config command.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001928
Greg Edelstona7b05d12020-04-01 16:00:51 -06001929 @returns a string representing this host's model.
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001930 """
Greg Edelstona7b05d12020-04-01 16:00:51 -06001931 return cros_config.call_cros_config_get_output('/ name',
1932 self.run, ignore_status=True)
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001933
1934
Hung-ying Tyanb1328032014-04-01 14:18:54 +08001935 def get_architecture(self):
1936 """Determine the correct architecture label for this host.
1937
1938 @returns a string representing this host's architecture.
1939 """
1940 crossystem = utils.Crossystem(self)
1941 crossystem.init()
1942 return crossystem.arch()
1943
1944
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001945 def get_chrome_version(self):
1946 """Gets the Chrome version number and milestone as strings.
1947
1948 Invokes "chrome --version" to get the version number and milestone.
1949
1950 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
1951 current Chrome version number as a string (in the form "W.X.Y.Z")
1952 and "milestone" is the first component of the version number
1953 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
1954 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
1955 of "chrome --version" and the milestone will be the empty string.
1956
1957 """
MK Ryu35d661e2014-09-25 17:44:10 -07001958 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001959 return utils.parse_chrome_version(version_string)
1960
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001961
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001962 def get_ec_version(self):
1963 """Get the ec version as strings.
1964
1965 @returns a string representing this host's ec version.
1966 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001967 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001968 result = self.run(command, ignore_status=True)
1969 if result.exit_status != 0:
1970 return ''
1971 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001972
1973
1974 def get_firmware_version(self):
1975 """Get the firmware version as strings.
1976
1977 @returns a string representing this host's firmware version.
1978 """
1979 crossystem = utils.Crossystem(self)
1980 crossystem.init()
1981 return crossystem.fwid()
1982
1983
1984 def get_hardware_revision(self):
1985 """Get the hardware revision as strings.
1986
1987 @returns a string representing this host's hardware revision.
1988 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001989 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001990 result = self.run(command, ignore_status=True)
1991 if result.exit_status != 0:
1992 return ''
1993 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001994
1995
1996 def get_kernel_version(self):
1997 """Get the kernel version as strings.
1998
1999 @returns a string representing this host's kernel version.
2000 """
2001 return self.run('uname -r').stdout.strip()
2002
2003
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002004 def get_cpu_name(self):
2005 """Get the cpu name as strings.
2006
2007 @returns a string representing this host's cpu name.
2008 """
2009
2010 # Try get cpu name from device tree first
2011 if self.path_exists('/proc/device-tree/compatible'):
2012 command = ' | '.join(
2013 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
2014 'tail -1'])
2015 return self.run(command).stdout.strip().replace(',', ' ')
2016
2017 # Get cpu name from uname -p
2018 command = 'uname -p'
2019 ret = self.run(command).stdout.strip()
2020
2021 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
2022 # Try get cpu name from /proc/cpuinfo instead
2023 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
2024 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
2025 self = self.run(command).stdout.strip()
2026
2027 # Remove bloat from CPU name, for example
2028 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
2029 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
2030 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
2031 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
2032 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
2033 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
2034
2035
2036 def get_screen_resolution(self):
2037 """Get the screen(s) resolution as strings.
2038 In case of more than 1 monitor, return resolution for each monitor
2039 separate with plus sign.
2040
2041 @returns a string representing this host's screen(s) resolution.
2042 """
2043 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
2044 ret = self.run(command, ignore_status=True)
2045 # We might have Chromebox without a screen
2046 if ret.exit_status != 0:
2047 return ''
2048 return ret.stdout.strip().replace('\n', '+')
2049
2050
2051 def get_mem_total_gb(self):
2052 """Get total memory available in the system in GiB (2^20).
2053
2054 @returns an integer representing total memory
2055 """
2056 mem_total_kb = self.read_from_meminfo('MemTotal')
2057 kb_in_gb = float(2 ** 20)
2058 return int(round(mem_total_kb / kb_in_gb))
2059
2060
2061 def get_disk_size_gb(self):
2062 """Get size of disk in GB (10^9)
2063
2064 @returns an integer representing size of disk, 0 in Error Case
2065 """
2066 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
2067 result = self.run(command, ignore_status=True)
2068 if result.exit_status != 0:
2069 return 0
2070 _, _, block, _ = re.split(r' +', result.stdout.strip())
2071 byte_per_block = 1024.0
2072 disk_kb_in_gb = 1e9
2073 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
2074
2075
2076 def get_battery_size(self):
2077 """Get size of battery in Watt-hour via sysfs
2078
2079 This method assumes that battery support voltage_min_design and
2080 charge_full_design sysfs.
2081
2082 @returns a float representing Battery size, 0 if error.
2083 """
2084 # sysfs report data in micro scale
2085 battery_scale = 1e6
2086
2087 command = 'cat /sys/class/power_supply/*/voltage_min_design'
2088 result = self.run(command, ignore_status=True)
2089 if result.exit_status != 0:
2090 return 0
2091 voltage = float(result.stdout.strip()) / battery_scale
2092
2093 command = 'cat /sys/class/power_supply/*/charge_full_design'
2094 result = self.run(command, ignore_status=True)
2095 if result.exit_status != 0:
2096 return 0
2097 amphereHour = float(result.stdout.strip()) / battery_scale
2098
2099 return voltage * amphereHour
2100
2101
2102 def get_low_battery_shutdown_percent(self):
2103 """Get the percent-based low-battery shutdown threshold.
2104
2105 @returns a float representing low-battery shutdown percent, 0 if error.
2106 """
2107 ret = 0.0
2108 try:
2109 command = 'check_powerd_config --low_battery_shutdown_percent'
2110 ret = float(self.run(command).stdout)
2111 except error.CmdError:
2112 logging.debug("Can't run %s", command)
2113 except ValueError:
2114 logging.debug("Didn't get number from %s", command)
2115
2116 return ret
2117
2118
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07002119 def has_hammer(self):
2120 """Check whether DUT has hammer device or not.
2121
2122 @returns boolean whether device has hammer or not
2123 """
2124 command = 'grep Hammer /sys/bus/usb/devices/*/product'
2125 return self.run(command, ignore_status=True).exit_status == 0
2126
2127
Niranjan Kumar34618872017-05-31 12:57:09 -07002128 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07002129 """Returns True if the specified switch was provided to Chrome.
2130
2131 @param switch The chrome switch to search for.
2132 """
Niranjan Kumar34618872017-05-31 12:57:09 -07002133
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07002134 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
2135 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07002136
2137
2138 def oobe_triggers_update(self):
2139 """Returns True if this host has an OOBE flow during which
2140 it will perform an update check and perhaps an update.
2141 One example of such a flow is Hands-Off Zero-Touch Enrollment.
2142 As more such flows are developed, code handling them needs
2143 to be added here.
2144
2145 @return Boolean indicating whether this host's OOBE triggers an update.
2146 """
2147 return self.is_chrome_switch_present(
2148 '--enterprise-enable-zero-touch-enrollment=hands-off')
2149
2150
Kevin Chenga2619dc2016-03-28 11:42:08 -07002151 # TODO(kevcheng): change this to just return the board without the
2152 # 'board:' prefix and fix up all the callers. Also look into removing the
2153 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002154 def get_board(self):
2155 """Determine the correct board label for this host.
2156
2157 @returns a string representing this host's board.
2158 """
2159 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
2160 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08002161 return (ds_constants.BOARD_PREFIX +
2162 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07002163
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002164 def get_channel(self):
2165 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07002166
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07002167 @returns: a string represeting this host's build channel.
2168 (stable, dev, beta). None on fail.
2169 """
2170 return lsbrelease_utils.get_chromeos_channel(
2171 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002172
Kevin Chenga328da62016-03-31 10:49:04 -07002173 def get_power_supply(self):
2174 """
2175 Determine what type of power supply the host has
2176
2177 @returns a string representing this host's power supply.
2178 'power:battery' when the device has a battery intended for
2179 extended use
2180 'power:AC_primary' when the device has a battery not intended
2181 for extended use (for moving the machine, etc)
2182 'power:AC_only' when the device has no battery at all.
2183 """
2184 psu = self.run(command='mosys psu type', ignore_status=True)
2185 if psu.exit_status:
2186 # The psu command for mosys is not included for all platforms. The
2187 # assumption is that the device will have a battery if the command
2188 # is not found.
2189 return 'power:battery'
2190
2191 psu_str = psu.stdout.strip()
2192 if psu_str == 'unknown':
2193 return None
2194
2195 return 'power:%s' % psu_str
2196
2197
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002198 def has_battery(self):
2199 """Determine if DUT has a battery.
2200
2201 Returns:
2202 Boolean, False if known not to have battery, True otherwise.
2203 """
2204 rv = True
2205 power_supply = self.get_power_supply()
2206 if power_supply == 'power:battery':
2207 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2208 board_type = self.get_board_type()
2209 if board_type in _NO_BATTERY_BOARD_TYPE:
2210 logging.warn('Do NOT believe type %s has battery. '
2211 'See debug for mosys details', board_type)
2212 psu = self.system_output('mosys -vvvv psu type',
2213 ignore_status=True)
2214 logging.debug(psu)
2215 rv = False
2216 elif power_supply == 'power:AC_only':
2217 rv = False
2218
2219 return rv
2220
2221
Kevin Chenga328da62016-03-31 10:49:04 -07002222 def get_servo(self):
2223 """Determine if the host has a servo attached.
2224
2225 If the host has a working servo attached, it should have a servo label.
2226
2227 @return: string 'servo' if the host has servo attached. Otherwise,
2228 returns None.
2229 """
2230 return 'servo' if self._servo_host else None
2231
2232
Kevin Chenga328da62016-03-31 10:49:04 -07002233 def has_internal_display(self):
2234 """Determine if the device under test is equipped with an internal
2235 display.
2236
2237 @return: 'internal_display' if one is present; None otherwise.
2238 """
2239 from autotest_lib.client.cros.graphics import graphics_utils
2240 from autotest_lib.client.common_lib import utils as common_utils
2241
2242 def __system_output(cmd):
2243 return self.run(cmd).stdout
2244
2245 def __read_file(remote_path):
2246 return self.run('cat %s' % remote_path).stdout
2247
2248 # Hijack the necessary client functions so that we can take advantage
2249 # of the client lib here.
2250 # FIXME: find a less hacky way than this
2251 original_system_output = utils.system_output
2252 original_read_file = common_utils.read_file
2253 utils.system_output = __system_output
2254 common_utils.read_file = __read_file
2255 try:
2256 return ('internal_display' if graphics_utils.has_internal_display()
2257 else None)
2258 finally:
2259 utils.system_output = original_system_output
2260 common_utils.read_file = original_read_file
2261
2262
Dan Shi85276d42014-04-08 22:11:45 -07002263 def is_boot_from_usb(self):
2264 """Check if DUT is boot from USB.
2265
2266 @return: True if DUT is boot from usb.
2267 """
2268 device = self.run('rootdev -s -d').stdout.strip()
2269 removable = int(self.run('cat /sys/block/%s/removable' %
2270 os.path.basename(device)).stdout.strip())
2271 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002272
2273
David Haddock19569d82020-05-06 05:30:21 -07002274 def get_active_boot_slot(self):
2275 """Returns the active boot slot."""
2276 return self.run(('rootdev', '-s')).stdout.strip()
2277
2278
Helen Zhang17dae2b2014-11-11 09:25:52 -08002279 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002280 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002281
2282 @param key: meminfo requested
2283
2284 @return the memory value as a string
2285
2286 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002287 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2288 logging.debug('%s', meminfo)
2289 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002290
2291
Rohit Makasana98e696f2016-06-03 18:48:10 -07002292 def get_cpu_arch(self):
2293 """Returns CPU arch of the device.
2294
2295 @return CPU architecture of the DUT.
2296 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002297 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002298 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2299 ignore_status=True).stdout:
2300 return 'x86_64'
2301 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2302 ignore_status=True).stdout:
2303 return 'arm'
2304 return 'i386'
2305
2306
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002307 def get_board_type(self):
2308 """
2309 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002310 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2311
2312 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002313 """
Danny Chan471a8d12015-08-18 14:57:41 -07002314 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2315 ignore_status=True).stdout
2316 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002317 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002318 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002319
2320
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002321 def get_arc_version(self):
2322 """Return ARC version installed on the DUT.
2323
2324 @returns ARC version as string if the CrOS build has ARC, else None.
2325 """
2326 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2327 ignore_status=True).stdout
2328 if arc_version:
2329 return arc_version.split('=')[-1].strip()
2330 return None
2331
2332
Gilad Arnolda76bef02015-09-29 13:55:15 -07002333 def get_os_type(self):
2334 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002335
2336
Kevin Chenga2619dc2016-03-28 11:42:08 -07002337 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002338 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002339 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002340
2341
2342 def get_default_power_method(self):
2343 """
2344 Get the default power method for power_on/off/cycle() methods.
2345 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2346 """
2347 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002348 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002349 if self.servo and self.servo.supports_built_in_pd_control():
2350 self._default_power_method = self.POWER_CONTROL_CCD
2351 else:
2352 logging.debug('Either servo is unitialized or the servo '
2353 'setup does not support pd controls. Falling '
2354 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002355 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002356
2357
2358 def find_usb_devices(self, idVendor, idProduct):
2359 """
2360 Get usb device sysfs name for specific device.
2361
2362 @param idVendor Vendor ID to search in sysfs directory.
2363 @param idProduct Product ID to search in sysfs directory.
2364
2365 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2366 """
2367 # Look for matching file and cut at position 7 to get dir name.
2368 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2369
2370 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2371 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2372
2373 # Use uniq -d to print duplicate line from both command
2374 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2375
2376 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2377
2378
2379 def bind_usb_device(self, usb_node):
2380 """
2381 Bind usb device
2382
2383 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2384 """
2385 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2386 self.run(cmd, ignore_status=True)
2387
2388
2389 def unbind_usb_device(self, usb_node):
2390 """
2391 Unbind usb device
2392
2393 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2394 """
2395 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2396 self.run(cmd, ignore_status=True)
2397
2398
2399 def get_wlan_ip(self):
2400 """
2401 Get ip address of wlan interface.
2402
2403 @return ip address of wlan or empty string if wlan is not connected.
2404 """
2405 cmds = [
2406 'iw dev', # List wlan physical device
2407 'grep Interface', # Grep only interface name
2408 'cut -f 2 -d" "', # Cut the name part
2409 'xargs ifconfig', # Feed it to ifconfig to get ip
2410 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2411 'cut -f 2 -d " "' # Cut the ip part
2412 ]
2413 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002414
2415 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2416 """
2417 Connect to wifi network
2418
2419 @param ssid SSID of the wifi network.
2420 @param passphrase Passphrase of the wifi network. None if not existed.
2421 @param security Security of the wifi network. Default to "psk" if
2422 passphase is given without security. Possible values
2423 are "none", "psk", "802_1x".
2424
2425 @return True if succeed, False if not.
2426 """
2427 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2428 if passphrase:
2429 cmd += ' ' + passphrase
2430 if security:
2431 cmd += ' ' + security
2432 return self.run(cmd, ignore_status=True).exit_status == 0