blob: f60bb3ba80b971b50f54cb84786272138066c119 [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
Richard Barnette03a0c132012-11-05 12:40:35 -080018from autotest_lib.client.common_lib.cros import dev_server
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -070019from autotest_lib.client.common_lib.cros import retry
Hsinyu Chaoe0b08e62015-08-11 10:50:37 +000020from autotest_lib.client.cros import constants as client_constants
J. Richard Barnette84890bd2014-02-21 11:05:47 -080021from autotest_lib.client.cros import cros_ui
Simran Basi5ace6f22016-01-06 17:30:44 -080022from autotest_lib.server import afe_utils
Dan Shia1ecd5c2013-06-06 11:21:31 -070023from autotest_lib.server import utils as server_utils
Dan Shi9cb0eec2014-06-03 09:04:50 -070024from autotest_lib.server.cros import provision
Scott Zawalski89c44dd2013-02-26 09:28:02 -050025from autotest_lib.server.cros.dynamic_suite import constants as ds_constants
Simran Basi5e6339a2013-03-21 11:34:32 -070026from autotest_lib.server.cros.dynamic_suite import tools, frontend_wrappers
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070027from autotest_lib.server.cros.servo import pdtester
Fang Deng96667ca2013-08-01 17:46:18 -070028from autotest_lib.server.hosts import abstract_ssh
Kevin Chenga2619dc2016-03-28 11:42:08 -070029from autotest_lib.server.hosts import base_label
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +080030from autotest_lib.server.hosts import chameleon_host
Richard Barnetted31580e2018-05-14 19:58:00 +000031from autotest_lib.server.hosts import cros_label
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -080032from autotest_lib.server.hosts import cros_repair
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -070033from autotest_lib.server.hosts import pdtester_host
Fang Deng5d518f42013-08-02 14:04:32 -070034from autotest_lib.server.hosts import servo_host
Simran Basidcff4252012-11-20 16:13:20 -080035from autotest_lib.site_utils.rpm_control_system import rpm_client
Simran Basid5e5e272012-09-24 15:23:59 -070036
Simran Basi382506b2016-09-13 14:58:15 -070037# In case cros_host is being ran via SSP on an older Moblab version with an
38# older chromite version.
39try:
40 from chromite.lib import metrics
Dan Shi5e2efb72017-02-07 11:40:23 -080041except ImportError:
Congbin Guo42427612019-02-12 10:22:06 -080042 metrics = utils.metrics_mock
Dan Shi5e2efb72017-02-07 11:40:23 -080043
Simran Basid5e5e272012-09-24 15:23:59 -070044
Dan Shib8540a52015-07-16 14:18:23 -070045CONFIG = global_config.global_config
46
Dan Shid07ee2e2015-09-24 14:49:25 -070047
beepsc87ff602013-07-31 21:53:00 -070048class FactoryImageCheckerException(error.AutoservError):
49 """Exception raised when an image is a factory image."""
50 pass
51
52
Fang Deng0ca40e22013-08-27 17:47:44 -070053class CrosHost(abstract_ssh.AbstractSSHHost):
J. Richard Barnette45e93de2012-04-11 17:24:15 -070054 """Chromium OS specific subclass of Host."""
55
Simran Basi5ace6f22016-01-06 17:30:44 -080056 VERSION_PREFIX = provision.CROS_VERSION_PREFIX
57
Scott Zawalski62bacae2013-03-05 10:40:32 -050058 _AFE = frontend_wrappers.RetryingAFE(timeout_min=5, delay_sec=10)
J. Richard Barnette45e93de2012-04-11 17:24:15 -070059
Richard Barnette03a0c132012-11-05 12:40:35 -080060 # Timeout values (in seconds) associated with various Chrome OS
61 # state changes.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -070062 #
Richard Barnette0c73ffc2012-11-19 15:21:18 -080063 # In general, a good rule of thumb is that the timeout can be up
64 # to twice the typical measured value on the slowest platform.
65 # The times here have not necessarily been empirically tested to
66 # meet this criterion.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070067 #
68 # SLEEP_TIMEOUT: Time to allow for suspend to memory.
Richard Barnette0c73ffc2012-11-19 15:21:18 -080069 # RESUME_TIMEOUT: Time to allow for resume after suspend, plus
70 # time to restart the netwowrk.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080071 # SHUTDOWN_TIMEOUT: Time to allow for shut down.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070072 # BOOT_TIMEOUT: Time to allow for boot from power off. Among
Richard Barnette0c73ffc2012-11-19 15:21:18 -080073 # other things, this must account for the 30 second dev-mode
J. Richard Barnette417cc792015-10-01 09:56:36 -070074 # screen delay, time to start the network on the DUT, and the
75 # ssh timeout of 120 seconds.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070076 # USB_BOOT_TIMEOUT: Time to allow for boot from a USB device,
Richard Barnette0c73ffc2012-11-19 15:21:18 -080077 # including the 30 second dev-mode delay and time to start the
J. Richard Barnetted4649c62013-03-06 17:42:27 -080078 # network.
beepsf079cfb2013-09-18 17:49:51 -070079 # INSTALL_TIMEOUT: Time to allow for chromeos-install.
J. Richard Barnette84890bd2014-02-21 11:05:47 -080080 # POWERWASH_BOOT_TIMEOUT: Time to allow for a reboot that
81 # includes powerwash.
J. Richard Barnetteeb69d722012-06-18 17:29:44 -070082
83 SLEEP_TIMEOUT = 2
J. Richard Barnetted4649c62013-03-06 17:42:27 -080084 RESUME_TIMEOUT = 10
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +080085 SHUTDOWN_TIMEOUT = 10
J. Richard Barnette417cc792015-10-01 09:56:36 -070086 BOOT_TIMEOUT = 150
J. Richard Barnette5bab5f52015-08-03 13:14:38 -070087 USB_BOOT_TIMEOUT = 300
J. Richard Barnette7817b052014-08-28 09:47:29 -070088 INSTALL_TIMEOUT = 480
Dan Shi2c88eed2013-11-12 10:18:38 -080089 POWERWASH_BOOT_TIMEOUT = 60
Chris Sosab76e0ee2013-05-22 16:55:41 -070090
Dan Shica503482015-03-30 17:23:25 -070091 # Minimum OS version that supports server side packaging. Older builds may
92 # not have server side package built or with Autotest code change to support
93 # server-side packaging.
Dan Shib8540a52015-07-16 14:18:23 -070094 MIN_VERSION_SUPPORT_SSP = CONFIG.get_config_value(
Dan Shiced09e42015-04-17 16:09:34 -070095 'AUTOSERV', 'min_version_support_ssp', type=int)
Dan Shica503482015-03-30 17:23:25 -070096
J. Richard Barnette84890bd2014-02-21 11:05:47 -080097 # REBOOT_TIMEOUT: How long to wait for a reboot.
98 #
Chris Sosab76e0ee2013-05-22 16:55:41 -070099 # We have a long timeout to ensure we don't flakily fail due to other
100 # issues. Shorter timeouts are vetted in platform_RebootAfterUpdate.
Simran Basi1160e2c2013-10-04 16:00:24 -0700101 # TODO(sbasi - crbug.com/276094) Restore to 5 mins once the 'host did not
102 # return from reboot' bug is solved.
103 REBOOT_TIMEOUT = 480
Chris Sosab76e0ee2013-05-22 16:55:41 -0700104
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800105 # _USB_POWER_TIMEOUT: Time to allow for USB to power toggle ON and OFF.
106 # _POWER_CYCLE_TIMEOUT: Time to allow for manual power cycle.
Garry Wang5e5538a2019-04-08 15:36:18 -0700107 # _CHANGE_SERVO_ROLE_TIMEOUT: Time to allow DUT regain network connection
108 # since changing servo role will reset USB state
109 # and causes temporary ethernet drop.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800110 _USB_POWER_TIMEOUT = 5
111 _POWER_CYCLE_TIMEOUT = 10
Garry Wang5e5538a2019-04-08 15:36:18 -0700112 _CHANGE_SERVO_ROLE_TIMEOUT = 180
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800113
Fang Dengdeba14f2014-11-14 11:54:09 -0800114 _RPM_HOSTNAME_REGEX = ('chromeos(\d+)(-row(\d+))?-rack(\d+[a-z]*)'
115 '-host(\d+)')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700116
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -0800117 # Constants used in ping_wait_up() and ping_wait_down().
118 #
119 # _PING_WAIT_COUNT is the approximate number of polling
120 # cycles to use when waiting for a host state change.
121 #
122 # _PING_STATUS_DOWN and _PING_STATUS_UP are names used
123 # for arguments to the internal _ping_wait_for_status()
124 # method.
125 _PING_WAIT_COUNT = 40
126 _PING_STATUS_DOWN = False
127 _PING_STATUS_UP = True
128
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800129 # Allowed values for the power_method argument.
130
Garry Wang5e5538a2019-04-08 15:36:18 -0700131 # POWER_CONTROL_RPM: Used in power_off/on/cycle() methods, default for all
132 # DUTs except those with servo_v4 CCD.
133 # POWER_CONTROL_CCD: Used in power_off/on/cycle() methods, default for all
134 # DUTs with servo_v4 CCD.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800135 # POWER_CONTROL_SERVO: Used in set_power() and power_cycle() methods.
136 # POWER_CONTROL_MANUAL: Used in set_power() and power_cycle() methods.
137 POWER_CONTROL_RPM = 'RPM'
Garry Wang5e5538a2019-04-08 15:36:18 -0700138 POWER_CONTROL_CCD = 'CCD'
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800139 POWER_CONTROL_SERVO = 'servoj10'
140 POWER_CONTROL_MANUAL = 'manual'
141
142 POWER_CONTROL_VALID_ARGS = (POWER_CONTROL_RPM,
Garry Wang5e5538a2019-04-08 15:36:18 -0700143 POWER_CONTROL_CCD,
Ismail Noorbasha07fdb612013-02-14 14:13:31 -0800144 POWER_CONTROL_SERVO,
145 POWER_CONTROL_MANUAL)
Richard Barnette0c73ffc2012-11-19 15:21:18 -0800146
Simran Basi5e6339a2013-03-21 11:34:32 -0700147 _RPM_OUTLET_CHANGED = 'outlet_changed'
148
Dan Shi9cb0eec2014-06-03 09:04:50 -0700149 # URL pattern to download firmware image.
Dan Shib8540a52015-07-16 14:18:23 -0700150 _FW_IMAGE_URL_PATTERN = CONFIG.get_config_value(
Dan Shi9cb0eec2014-06-03 09:04:50 -0700151 'CROS', 'firmware_url_pattern', type=str)
beeps687243d2013-07-18 15:29:27 -0700152
Brent Peterson1cb623a2020-01-09 13:14:28 -0800153 # Regular expression for extracting EC version string
154 _EC_REGEX = '(%s_\w*[-\.]\w*[-\.]\w*[-\.]\w*)'
155
156 # Regular expression for extracting BIOS version string
157 _BIOS_REGEX = '(%s\.\w*\.\w*\.\w*)'
158
J. Richard Barnette964fba02012-10-24 17:34:29 -0700159 @staticmethod
beeps46dadc92013-11-07 14:07:10 -0800160 def check_host(host, timeout=10):
161 """
162 Check if the given host is a chrome-os host.
163
164 @param host: An ssh host representing a device.
165 @param timeout: The timeout for the run command.
166
167 @return: True if the host device is chromeos.
168
beeps46dadc92013-11-07 14:07:10 -0800169 """
170 try:
Allen Liad719c12017-06-27 23:48:04 +0000171 result = host.run(
Simran Basi933c8af2015-04-29 14:05:07 -0700172 'grep -q CHROMEOS /etc/lsb-release && '
Garry Wange4b6d6e2019-06-17 17:08:46 -0700173 '! grep -q moblab /etc/lsb-release && '
174 '! grep -q labstation /etc/lsb-release',
Simran Basi933c8af2015-04-29 14:05:07 -0700175 ignore_status=True, timeout=timeout)
Laurence Goodby468de252017-06-08 17:22:53 -0700176 if result.exit_status == 0:
Allen Liad719c12017-06-27 23:48:04 +0000177 lsb_release_content = host.run(
Laurence Goodby468de252017-06-08 17:22:53 -0700178 'grep CHROMEOS_RELEASE_BOARD /etc/lsb-release',
179 timeout=timeout).stdout
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -0800180 return not (
181 lsbrelease_utils.is_jetstream(
182 lsb_release_content=lsb_release_content) or
183 lsbrelease_utils.is_gce_board(
184 lsb_release_content=lsb_release_content))
185
beeps46dadc92013-11-07 14:07:10 -0800186 except (error.AutoservRunError, error.AutoservSSHTimeout):
187 return False
Laurence Goodby468de252017-06-08 17:22:53 -0700188
189 return False
beeps46dadc92013-11-07 14:07:10 -0800190
191
192 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800193 def get_chameleon_arguments(args_dict):
194 """Extract chameleon options from `args_dict` and return the result.
195
196 Recommended usage:
197 ~~~~~~~~
198 args_dict = utils.args_to_dict(args)
199 chameleon_args = hosts.CrosHost.get_chameleon_arguments(args_dict)
200 host = hosts.create_host(machine, chameleon_args=chameleon_args)
201 ~~~~~~~~
202
203 @param args_dict Dictionary from which to extract the chameleon
204 arguments.
205 """
howardchung83e55272019-08-08 14:08:05 +0800206 if 'chameleon_host_list' in args_dict:
207 result = []
208 for chameleon in args_dict['chameleon_host_list'].split(','):
209 result.append({key: value for key,value in
210 zip(('chameleon_host','chameleon_port'),
211 chameleon.split(':'))})
212
213 logging.info(result)
214 return result
215 else:
216 return {key: args_dict[key]
Allen Li083866b2016-08-18 10:07:10 -0700217 for key in ('chameleon_host', 'chameleon_port')
218 if key in args_dict}
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800219
220
221 @staticmethod
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700222 def get_pdtester_arguments(args_dict):
Scottfe06ed82015-11-05 17:15:01 -0800223 """Extract chameleon options from `args_dict` and return the result.
224
225 Recommended usage:
226 ~~~~~~~~
227 args_dict = utils.args_to_dict(args)
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700228 pdtester_args = hosts.CrosHost.get_pdtester_arguments(args_dict)
229 host = hosts.create_host(machine, pdtester_args=pdtester_args)
Scottfe06ed82015-11-05 17:15:01 -0800230 ~~~~~~~~
231
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700232 @param args_dict Dictionary from which to extract the pdtester
Scottfe06ed82015-11-05 17:15:01 -0800233 arguments.
234 """
Allen Li083866b2016-08-18 10:07:10 -0700235 return {key: args_dict[key]
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700236 for key in ('pdtester_host', 'pdtester_port')
Allen Li083866b2016-08-18 10:07:10 -0700237 if key in args_dict}
Scottfe06ed82015-11-05 17:15:01 -0800238
239
240 @staticmethod
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800241 def get_servo_arguments(args_dict):
242 """Extract servo options from `args_dict` and return the result.
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800243
244 Recommended usage:
245 ~~~~~~~~
246 args_dict = utils.args_to_dict(args)
Fang Deng0ca40e22013-08-27 17:47:44 -0700247 servo_args = hosts.CrosHost.get_servo_arguments(args_dict)
J. Richard Barnette7214e0b2013-02-06 15:20:49 -0800248 host = hosts.create_host(machine, servo_args=servo_args)
249 ~~~~~~~~
250
251 @param args_dict Dictionary from which to extract the servo
252 arguments.
253 """
Richard Barnettee519dcd2016-08-15 17:37:17 -0700254 servo_attrs = (servo_host.SERVO_HOST_ATTR,
255 servo_host.SERVO_PORT_ATTR,
Nick Sanders2f3c9852018-10-24 12:10:24 -0700256 servo_host.SERVO_BOARD_ATTR,
257 servo_host.SERVO_MODEL_ATTR)
Armando Miraglia2ac802e2017-08-15 14:54:47 +0200258 servo_args = {key: args_dict[key]
259 for key in servo_attrs
260 if key in args_dict}
261 return (
262 None
263 if servo_host.SERVO_HOST_ATTR in servo_args
264 and not servo_args[servo_host.SERVO_HOST_ATTR]
265 else servo_args)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -0700266
J. Richard Barnette964fba02012-10-24 17:34:29 -0700267
J. Richard Barnette91137f02016-03-10 16:52:26 -0800268 def _initialize(self, hostname, chameleon_args=None, servo_args=None,
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700269 pdtester_args=None, try_lab_servo=False,
Richard Barnette9a26ad62016-06-10 12:03:08 -0700270 try_servo_repair=False,
J. Richard Barnette91137f02016-03-10 16:52:26 -0800271 ssh_verbosity_flag='', ssh_options='',
Fang Dengd1c2b732013-08-20 12:59:46 -0700272 *args, **dargs):
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800273 """Initialize superclasses, |self.chameleon|, and |self.servo|.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700274
Fang Denge545abb2014-12-30 18:43:47 -0800275 This method will attempt to create the test-assistant object
276 (chameleon/servo) when it is needed by the test. Check
277 the docstring of chameleon_host.create_chameleon_host and
278 servo_host.create_servo_host for how this is determined.
Fang Deng5d518f42013-08-02 14:04:32 -0700279
Fang Denge545abb2014-12-30 18:43:47 -0800280 @param hostname: Hostname of the dut.
281 @param chameleon_args: A dictionary that contains args for creating
282 a ChameleonHost. See chameleon_host for details.
283 @param servo_args: A dictionary that contains args for creating
284 a ServoHost object. See servo_host for details.
Richard Barnette9a26ad62016-06-10 12:03:08 -0700285 @param try_lab_servo: When true, indicates that an attempt should
286 be made to create a ServoHost for a DUT in
287 the test lab, even if not required by
288 `servo_args`. See servo_host for details.
289 @param try_servo_repair: If a servo host is created, check it
290 with `repair()` rather than `verify()`.
Fang Denge545abb2014-12-30 18:43:47 -0800291 See servo_host for details.
292 @param ssh_verbosity_flag: String, to pass to the ssh command to control
293 verbosity.
294 @param ssh_options: String, other ssh options to pass to the ssh
295 command.
J. Richard Barnette67ccb872012-04-19 16:34:56 -0700296 """
Fang Deng0ca40e22013-08-27 17:47:44 -0700297 super(CrosHost, self)._initialize(hostname=hostname,
J. Richard Barnette45e93de2012-04-11 17:24:15 -0700298 *args, **dargs)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800299 self._repair_strategy = cros_repair.create_cros_repair_strategy()
Kevin Chenga2619dc2016-03-28 11:42:08 -0700300 self.labels = base_label.LabelRetriever(cros_label.CROS_LABELS)
J. Richard Barnettef0859852012-08-20 14:55:50 -0700301 # self.env is a dictionary of environment variable settings
302 # to be exported for commands run on the host.
303 # LIBC_FATAL_STDERR_ can be useful for diagnosing certain
304 # errors that might happen.
305 self.env['LIBC_FATAL_STDERR_'] = '1'
Fang Dengd1c2b732013-08-20 12:59:46 -0700306 self._ssh_verbosity_flag = ssh_verbosity_flag
Aviv Keshetc5947fa2013-09-04 14:06:29 -0700307 self._ssh_options = ssh_options
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700308 self.set_servo_host(
309 servo_host.create_servo_host(
Richard Barnetteea3e4602016-06-10 12:36:41 -0700310 dut=self, servo_args=servo_args,
Richard Barnette9a26ad62016-06-10 12:03:08 -0700311 try_lab_servo=try_lab_servo,
Gregory Nisbetde13e2a2019-12-09 22:44:00 -0800312 try_servo_repair=try_servo_repair,
313 dut_host_info=self.host_info_store.get()))
Garry Wang5e5538a2019-04-08 15:36:18 -0700314 self._default_power_method = None
Richard Barnettee519dcd2016-08-15 17:37:17 -0700315
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800316 # TODO(waihong): Do the simplication on Chameleon too.
howardchung83e55272019-08-08 14:08:05 +0800317 if type(chameleon_args) is list:
318 self.multi_chameleon = True
319 chameleon_args_list = chameleon_args
320 else:
321 self.multi_chameleon = False
322 chameleon_args_list = [chameleon_args]
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800323
howardchung83e55272019-08-08 14:08:05 +0800324 self._chameleon_host_list = [
325 chameleon_host.create_chameleon_host(
326 dut=self.hostname, chameleon_args=_args)
327 for _args in chameleon_args_list]
328
329 self.chameleon_list = [_host.create_chameleon_board() for _host in
330 self._chameleon_host_list if _host is not None]
331 if len(self.chameleon_list) > 0:
332 self.chameleon = self.chameleon_list[0]
Tom Wai-Hong Tamefe1c7f2014-01-02 14:00:11 +0800333 else:
Tom Wai-Hong Tam3d6790d2014-04-14 16:15:47 +0800334 self.chameleon = None
Fang Deng5d518f42013-08-02 14:04:32 -0700335
howardchung83e55272019-08-08 14:08:05 +0800336 # Add pdtester host if pdtester args were added on command line
Wai-Hong Tam16e5edb2019-09-17 16:10:07 -0700337 self._pdtester_host = pdtester_host.create_pdtester_host(
Wai-Hong Tam90b164d2019-10-25 13:15:39 -0700338 pdtester_args, self._servo_host)
howardchung83e55272019-08-08 14:08:05 +0800339
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700340 if self._pdtester_host:
341 self.pdtester_servo = self._pdtester_host.get_servo()
342 logging.info('pdtester_servo: %r', self.pdtester_servo)
343 # Create the pdtester object used to access the ec uart
344 self.pdtester = pdtester.PDTester(self.pdtester_servo,
345 self._pdtester_host.get_servod_server_proxy())
Scottfe06ed82015-11-05 17:15:01 -0800346 else:
Wai-Hong Tam0ed9fe12019-05-23 11:00:58 -0700347 self.pdtester = None
Scottfe06ed82015-11-05 17:15:01 -0800348
Fang Deng5d518f42013-08-02 14:04:32 -0700349
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000350 def get_cros_repair_image_name(self):
Ruben Rodriguez Buchilloncee72f72019-05-01 13:20:58 -0700351 """Get latest stable cros image name from AFE.
352
353 Use the board name from the info store. Should that fail, try to
354 retrieve the board name from the host's installed image itself.
355
356 @returns: current stable cros image name for this host.
357 """
358 board = self.host_info_store.get().board
359 if not board:
360 logging.warn('No board label value found. Trying to infer '
361 'from the host itself.')
362 try:
363 board = self.get_board().split(':')[1]
364 except (error.AutoservRunError, error.AutoservSSHTimeout) as e:
365 logging.error('Also failed to get the board name from the DUT '
366 'itself. %s.', str(e))
367 raise error.AutoservError('Cannot obtain repair image name.')
Gregory Nisbet7fe11c22019-11-22 11:06:06 -0800368 return afe_utils.get_stable_cros_image_name_v2(self.host_info_store.get())
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500369
370
Rohit Makasanadf0a3a32017-06-30 13:55:18 -0700371 def host_version_prefix(self, image):
372 """Return version label prefix.
373
374 In case the CrOS provisioning version is something other than the
375 standard CrOS version e.g. CrOS TH version, this function will
376 find the prefix from provision.py.
377
378 @param image: The image name to find its version prefix.
379 @returns: A prefix string for the image type.
380 """
381 return provision.get_version_label_prefix(image)
382
383
beepsdae65fd2013-07-26 16:24:41 -0700384 def verify_job_repo_url(self, tag=''):
beepscb6f1e22013-06-28 19:14:10 -0700385 """
386 Make sure job_repo_url of this host is valid.
387
joychen03eaad92013-06-26 09:55:21 -0700388 Eg: The job_repo_url "http://lmn.cd.ab.xyx:8080/static/\
beepscb6f1e22013-06-28 19:14:10 -0700389 lumpy-release/R29-4279.0.0/autotest/packages" claims to have the
390 autotest package for lumpy-release/R29-4279.0.0. If this isn't the case,
391 download and extract it. If the devserver embedded in the url is
392 unresponsive, update the job_repo_url of the host after staging it on
393 another devserver.
394
395 @param job_repo_url: A url pointing to the devserver where the autotest
396 package for this build should be staged.
beepsdae65fd2013-07-26 16:24:41 -0700397 @param tag: The tag from the server job, in the format
398 <job_id>-<user>/<hostname>, or <hostless> for a server job.
beepscb6f1e22013-06-28 19:14:10 -0700399
400 @raises DevServerException: If we could not resolve a devserver.
401 @raises AutoservError: If we're unable to save the new job_repo_url as
402 a result of choosing a new devserver because the old one failed to
403 respond to a health check.
beeps0c865032013-07-30 11:37:06 -0700404 @raises urllib2.URLError: If the devserver embedded in job_repo_url
405 doesn't respond within the timeout.
beepscb6f1e22013-06-28 19:14:10 -0700406 """
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800407 info = self.host_info_store.get()
408 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
beepscb6f1e22013-06-28 19:14:10 -0700409 if not job_repo_url:
410 logging.warning('No job repo url set on host %s', self.hostname)
411 return
412
413 logging.info('Verifying job repo url %s', job_repo_url)
414 devserver_url, image_name = tools.get_devserver_build_from_package_url(
415 job_repo_url)
416
beeps0c865032013-07-30 11:37:06 -0700417 ds = dev_server.ImageServer(devserver_url)
beepscb6f1e22013-06-28 19:14:10 -0700418
419 logging.info('Staging autotest artifacts for %s on devserver %s',
420 image_name, ds.url())
beeps687243d2013-07-18 15:29:27 -0700421
422 start_time = time.time()
Simran Basi25e7a922014-10-31 11:56:10 -0700423 ds.stage_artifacts(image_name, ['autotest_packages'])
beeps687243d2013-07-18 15:29:27 -0700424 stage_time = time.time() - start_time
425
426 # Record how much of the verification time comes from a devserver
427 # restage. If we're doing things right we should not see multiple
428 # devservers for a given board/build/branch path.
429 try:
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800430 board, build_type, branch = server_utils.ParseBuildName(
beeps687243d2013-07-18 15:29:27 -0700431 image_name)[:3]
J. Richard Barnette3cbd76b2013-11-27 12:11:25 -0800432 except server_utils.ParseBuildNameException:
beeps687243d2013-07-18 15:29:27 -0700433 pass
434 else:
beeps0c865032013-07-30 11:37:06 -0700435 devserver = devserver_url[
Chris Sosa65425082013-10-16 13:26:22 -0700436 devserver_url.find('/') + 2:devserver_url.rfind(':')]
beeps687243d2013-07-18 15:29:27 -0700437 stats_key = {
438 'board': board,
439 'build_type': build_type,
440 'branch': branch,
beeps0c865032013-07-30 11:37:06 -0700441 'devserver': devserver.replace('.', '_'),
beeps687243d2013-07-18 15:29:27 -0700442 }
Dan Shi5e2efb72017-02-07 11:40:23 -0800443
444 monarch_fields = {
445 'board': board,
446 'build_type': build_type,
Dan Shi5e2efb72017-02-07 11:40:23 -0800447 'branch': branch,
448 'dev_server': devserver,
449 }
450 metrics.Counter(
451 'chromeos/autotest/provision/verify_url'
452 ).increment(fields=monarch_fields)
453 metrics.SecondsDistribution(
454 'chromeos/autotest/provision/verify_url_duration'
455 ).add(stage_time, fields=monarch_fields)
456
457
Dan Shicf4d2032015-03-12 15:04:21 -0700458 def stage_server_side_package(self, image=None):
459 """Stage autotest server-side package on devserver.
460
461 @param image: Full path of an OS image to install or a build name.
462
463 @return: A url to the autotest server-side package.
Dan Shi14de7622016-08-22 11:09:06 -0700464
465 @raise: error.AutoservError if fail to locate the build to test with, or
466 fail to stage server-side package.
Dan Shicf4d2032015-03-12 15:04:21 -0700467 """
Dan Shid37736b2016-07-06 15:10:29 -0700468 # If enable_drone_in_restricted_subnet is False, do not set hostname
469 # in devserver.resolve call, so a devserver in non-restricted subnet
470 # is picked to stage autotest server package for drone to download.
471 hostname = self.hostname
472 if not server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
473 hostname = None
Dan Shicf4d2032015-03-12 15:04:21 -0700474 if image:
475 image_name = tools.get_build_from_image(image)
476 if not image_name:
477 raise error.AutoservError(
478 'Failed to parse build name from %s' % image)
Dan Shid37736b2016-07-06 15:10:29 -0700479 ds = dev_server.ImageServer.resolve(image_name, hostname)
Dan Shicf4d2032015-03-12 15:04:21 -0700480 else:
Prathmesh Prabhu9235e4c2017-03-28 13:16:06 -0700481 info = self.host_info_store.get()
Prathmesh Prabhu368abdf2017-02-14 11:23:47 -0800482 job_repo_url = info.attributes.get(ds_constants.JOB_REPO_URL, '')
Dan Shicf4d2032015-03-12 15:04:21 -0700483 if job_repo_url:
484 devserver_url, image_name = (
485 tools.get_devserver_build_from_package_url(job_repo_url))
Dan Shid37736b2016-07-06 15:10:29 -0700486 # If enable_drone_in_restricted_subnet is True, use the
487 # existing devserver. Otherwise, resolve a new one in
488 # non-restricted subnet.
489 if server_utils.ENABLE_DRONE_IN_RESTRICTED_SUBNET:
490 ds = dev_server.ImageServer(devserver_url)
491 else:
492 ds = dev_server.ImageServer.resolve(image_name)
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800493 elif info.build is not None:
494 ds = dev_server.ImageServer.resolve(info.build, hostname)
Prathmesh Prabhu0c1dd4d2017-06-07 13:01:53 -0700495 image_name = info.build
Dan Shicf4d2032015-03-12 15:04:21 -0700496 else:
Prathmesh Prabhub6cea612017-02-09 15:41:19 -0800497 raise error.AutoservError(
498 'Failed to stage server-side package. The host has '
Garry Wang12b9baf2019-06-24 18:58:54 -0700499 'no job_repo_url attribute or cros-version label.')
Dan Shica503482015-03-30 17:23:25 -0700500
501 # Get the OS version of the build, for any build older than
502 # MIN_VERSION_SUPPORT_SSP, server side packaging is not supported.
503 match = re.match('.*/R\d+-(\d+)\.', image_name)
504 if match and int(match.group(1)) < self.MIN_VERSION_SUPPORT_SSP:
Dan Shi14de7622016-08-22 11:09:06 -0700505 raise error.AutoservError(
506 'Build %s is older than %s. Server side packaging is '
507 'disabled.' % (image_name, self.MIN_VERSION_SUPPORT_SSP))
Dan Shica503482015-03-30 17:23:25 -0700508
Dan Shicf4d2032015-03-12 15:04:21 -0700509 ds.stage_artifacts(image_name, ['autotest_server_package'])
510 return '%s/static/%s/%s' % (ds.url(), image_name,
511 'autotest_server_package.tar.bz2')
512
513
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700514 def stage_image_for_servo(self, image_name=None, artifact='test_image'):
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700515 """Stage a build on a devserver and return the update_url.
516
517 @param image_name: a name like lumpy-release/R27-3837.0.0
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700518 @param artifact: a string like 'test_image'. Requests
519 appropriate image to be staged.
Xixuan Wufee57542019-10-15 11:50:27 -0700520 @returns a tuple of (image_name, URL) like
521 (lumpy-release/R27-3837.0.0,
522 http://172.22.50.205:8082/update/lumpy-release/R27-3837.0.0)
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700523 """
524 if not image_name:
Richard Barnetteb4b4d6f2018-05-05 01:47:41 +0000525 image_name = self.get_cros_repair_image_name()
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700526 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800527 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700528 devserver.stage_artifacts(image_name, [artifact])
529 if artifact == 'test_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700530 return image_name, devserver.get_test_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700531 elif artifact == 'recovery_image':
Xixuan Wufee57542019-10-15 11:50:27 -0700532 return image_name, devserver.get_recovery_image_url(image_name)
Kalin Stoyanovb3c11f32018-05-11 09:02:00 -0700533 else:
534 raise error.AutoservError("Bad artifact!")
J. Richard Barnettee4af8b92013-05-01 13:16:12 -0700535
536
beepse539be02013-07-31 21:57:39 -0700537 def stage_factory_image_for_servo(self, image_name):
538 """Stage a build on a devserver and return the update_url.
539
540 @param image_name: a name like <baord>/4262.204.0
beeps12c0a3c2013-09-03 11:58:27 -0700541
beepse539be02013-07-31 21:57:39 -0700542 @return: An update URL, eg:
543 http://<devserver>/static/canary-channel/\
544 <board>/4262.204.0/factory_test/chromiumos_factory_image.bin
beeps12c0a3c2013-09-03 11:58:27 -0700545
546 @raises: ValueError if the factory artifact name is missing from
547 the config.
548
beepse539be02013-07-31 21:57:39 -0700549 """
550 if not image_name:
551 logging.error('Need an image_name to stage a factory image.')
552 return
553
Dan Shib8540a52015-07-16 14:18:23 -0700554 factory_artifact = CONFIG.get_config_value(
beeps12c0a3c2013-09-03 11:58:27 -0700555 'CROS', 'factory_artifact', type=str, default='')
556 if not factory_artifact:
557 raise ValueError('Cannot retrieve the factory artifact name from '
558 'autotest config, and hence cannot stage factory '
559 'artifacts.')
560
beepse539be02013-07-31 21:57:39 -0700561 logging.info('Staging build for servo install: %s', image_name)
Dan Shi216389c2015-12-22 11:03:06 -0800562 devserver = dev_server.ImageServer.resolve(image_name, self.hostname)
beepse539be02013-07-31 21:57:39 -0700563 devserver.stage_artifacts(
564 image_name,
beeps12c0a3c2013-09-03 11:58:27 -0700565 [factory_artifact],
566 archive_url=None)
beepse539be02013-07-31 21:57:39 -0700567
568 return tools.factory_image_url_pattern() % (devserver.url(), image_name)
569
570
Laurence Goodby778c9a42017-05-24 19:24:07 -0700571 def prepare_for_update(self):
572 """Prepares the DUT for an update.
573
574 Subclasses may override this to perform any special actions
575 required before updating.
576 """
Laurence Goodby468de252017-06-08 17:22:53 -0700577 pass
Laurence Goodby778c9a42017-05-24 19:24:07 -0700578
579
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800580 def _clear_fw_version_labels(self, rw_only):
581 """Clear firmware version labels from the machine.
582
583 @param rw_only: True to only clear fwrw_version; otherewise, clear
584 both fwro_version and fwrw_version.
585 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700586 info = self.host_info_store.get()
587 info.clear_version_labels(provision.FW_RW_VERSION_PREFIX)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800588 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700589 info.clear_version_labels(provision.FW_RO_VERSION_PREFIX)
590 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700591
592
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800593 def _add_fw_version_label(self, build, rw_only):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700594 """Add firmware version label to the machine.
595
596 @param build: Build of firmware.
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800597 @param rw_only: True to only add fwrw_version; otherwise, add both
598 fwro_version and fwrw_version.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700599
600 """
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700601 info = self.host_info_store.get()
602 info.set_version_label(provision.FW_RW_VERSION_PREFIX, build)
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800603 if not rw_only:
Prathmesh Prabhuf8ae9a82019-10-04 15:34:13 -0700604 info.set_version_label(provision.FW_RO_VERSION_PREFIX, build)
605 self.host_info_store.commit(info)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700606
607
Namyoon Woo5f894662019-11-15 15:23:23 -0800608 def get_latest_release_version(self, board):
609 """Search for the latest package release version from the image archive,
610 and return it.
611
612 @param board: board name
613
614 @return 'firmware-{board}-{branch}-firmwarebranch/{release-version}'
615 or None if LATEST release file does not exist.
616 """
617
618 # This might be in the format of 'baseboard_model',
619 # e.g. octopus_fleex. In that case, board should be just
620 # 'baseboard' to use in search for image package, e.g. octopus.
621 board = board.split('_')[0]
622
623 # Read 'LATEST-1.0.0' file
624 branch_dir = provision.FW_BRANCH_GLOB % board
625 latest_file = os.path.join(provision.CROS_IMAGE_ARCHIVE, branch_dir,
626 'LATEST-1.0.0')
Namyoon Woo5f894662019-11-15 15:23:23 -0800627
Namyoon Woo406c7d42020-01-24 15:57:11 -0800628 try:
629 # The result could be one or more.
630 result = utils.system_output('gsutil ls -d ' + latest_file)
631
632 candidates = re.findall('gs://.*', result)
Namyoon Woo5f894662019-11-15 15:23:23 -0800633 except error.CmdError:
634 logging.error('No LATEST release info is available.')
635 return None
636
Namyoon Woo406c7d42020-01-24 15:57:11 -0800637 for cand_dir in candidates:
638 result = utils.system_output('gsutil cat ' + cand_dir)
Namyoon Woo5f894662019-11-15 15:23:23 -0800639
Namyoon Woo406c7d42020-01-24 15:57:11 -0800640 release_path = cand_dir.replace('LATEST-1.0.0', result)
641 release_path = os.path.join(release_path, board)
642 try:
643 # Check if release_path does exist.
644 release = utils.system_output('gsutil ls -d ' + release_path)
645 # Now 'release' has a full directory path: e.g.
646 # gs://chromeos-image-archive/firmware-octopus-11297.B-
647 # firmwarebranch/RNone-1.0.0-b4395530/octopus/
648
649 # Remove "gs://chromeos-image-archive".
650 release = release.replace(provision.CROS_IMAGE_ARCHIVE, '')
651
652 # Remove CROS_IMAGE_ARCHIVE and any surrounding '/'s.
653 return release.strip('/')
654 except error.CmdError:
655 # The directory might not exist. Let's try next candidate.
656 pass
657 else:
658 raise error.AutoservError('Cannot find the latest firmware')
Namyoon Woo5f894662019-11-15 15:23:23 -0800659
Brent Peterson1cb623a2020-01-09 13:14:28 -0800660 @staticmethod
661 def get_version_from_image(image, version_regex):
662 with open(image, 'rb') as f:
663 image_data = f.read()
664 match = re.findall(version_regex, image_data)
665 if match:
666 return match[0]
667 else:
668 raise error.TestFail('Failed to read version from %s.' % image)
669
670
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800671 def firmware_install(self, build=None, rw_only=False, dest=None,
Brent Peterson1cb623a2020-01-09 13:14:28 -0800672 local_tarball=None, verify_version=False):
Dan Shi9cb0eec2014-06-03 09:04:50 -0700673 """Install firmware to the DUT.
674
675 Use stateful update if the DUT is already running the same build.
676 Stateful update does not update kernel and tends to run much faster
677 than a full reimage. If the DUT is running a different build, or it
678 failed to do a stateful update, full update, including kernel update,
679 will be applied to the DUT.
680
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800681 Once a host enters firmware_install its fw[ro|rw]_version label will
682 be removed. After the firmware is updated successfully, a new
683 fw[ro|rw]_version label will be added to the host.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700684
685 @param build: The build version to which we want to provision the
686 firmware of the machine,
687 e.g. 'link-firmware/R22-2695.1.144'.
Tom Wai-Hong Tam4ac78982016-01-08 02:34:37 +0800688 @param rw_only: True to only install firmware to its RW portions. Keep
689 the RO portions unchanged.
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700690 @param dest: Directory to store the firmware in.
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800691 @param local_tarball: Path to local firmware image for installing
692 without devserver.
Brent Peterson1cb623a2020-01-09 13:14:28 -0800693 @param verify_version: True to verify EC and BIOS versions after
694 programming firmware, default is False.
Dan Shi9cb0eec2014-06-03 09:04:50 -0700695
696 TODO(dshi): After bug 381718 is fixed, update here with corresponding
697 exceptions that could be raised.
698
699 """
700 if not self.servo:
701 raise error.TestError('Host %s does not have servo.' %
702 self.hostname)
703
Wai-Hong Tam3fa455a2018-07-18 14:40:43 -0700704 # Get the DUT board name from AFE.
705 info = self.host_info_store.get()
706 board = info.board
Shelley Chenac61d5a2019-06-24 15:35:46 -0700707 model = info.model
Namyoon Woo8dbfcf92019-01-15 18:37:12 -0800708
709 if board is None or board == '':
710 board = self.servo.get_board()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700711
Mary Ruthvende14a8b2019-08-23 12:43:52 -0700712 if model is None or model == '':
713 model = self.get_platform_from_fwid()
714
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800715 # If local firmware path not provided fetch it from the dev server
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700716 tmpd = None
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800717 if not local_tarball:
718 # If build is not set, try to install firmware from stable CrOS.
719 if not build:
720 build = afe_utils.get_stable_faft_version_v2(info)
721 if not build:
722 raise error.TestError(
723 'Failed to find stable firmware build for %s.',
724 self.hostname)
725 logging.info('Will install firmware from build %s.', build)
726
727 ds = dev_server.ImageServer.resolve(build, self.hostname)
728 ds.stage_artifacts(build, ['firmware'])
729
730 if not dest:
731 tmpd = autotemp.tempdir(unique_id='fwimage')
732 dest = tmpd.name
733
734 # Download firmware image
Dan Shi9cb0eec2014-06-03 09:04:50 -0700735 fwurl = self._FW_IMAGE_URL_PATTERN % (ds.url(), build)
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700736 local_tarball = os.path.join(dest, os.path.basename(fwurl))
xixuan4e116822016-11-17 15:32:10 -0800737 ds.download_file(fwurl, local_tarball)
Dan Shi9cb0eec2014-06-03 09:04:50 -0700738
Brent Peterson1cb623a2020-01-09 13:14:28 -0800739 # Extract EC image from tarball
740 logging.info('Extracting EC image.')
741 ec_image = self.servo.extract_ec_image(board, model, local_tarball)
742
743 # Extract BIOS image from tarball
744 logging.info('Extracting BIOS image.')
745 bios_image = self.servo.extract_bios_image(board, model, local_tarball)
746
Brent Peterson5b7c5b12020-01-09 13:14:28 -0800747 try:
Brent Peterson1cb623a2020-01-09 13:14:28 -0800748 # Program firmware using servo
Tom Wai-Hong Tam2d00cb22016-01-08 06:40:50 +0800749 self._clear_fw_version_labels(rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800750 self.servo.program_ec(ec_image, rw_only)
751 self.servo.program_bios(bios_image, rw_only)
Wai-Hong Tamb5f66ce2016-11-10 15:45:30 -0800752 if utils.host_is_in_lab_zone(self.hostname):
753 self._add_fw_version_label(build, rw_only)
Brent Peterson1cb623a2020-01-09 13:14:28 -0800754
755 # Reboot and wait for DUT after installing firmware
756 logging.info('Rebooting DUT.')
757 self.servo.get_power_state_controller().reset()
758 time.sleep(self.servo.BOOT_DELAY)
759 self.test_wait_for_boot()
760
761 # When enabled verify EC and BIOS firmware version after programming
762 if verify_version:
763 # Check programmed EC firmware against expected version
764 logging.info('Checking EC firmware version.')
765 dest_ec_version = self.get_ec_version()
766 ec_regex = self._EC_REGEX % model
767 image_ec_version = self.get_version_from_image(ec_image,
768 ec_regex)
769 if dest_ec_version != image_ec_version:
770 raise error.TestFail(
771 'Failed to update EC RO, version %s (expected %s)' %
772 (dest_ec_version, image_ec_version))
773
774 # Check programmed BIOS firmware against expected version
775 logging.info('Checking BIOS firmware version.')
776 dest_bios_version = self.get_firmware_version()
777 bios_version_prefix = dest_bios_version.split('.', 1)[0]
778 bios_regex = self._BIOS_REGEX % bios_version_prefix
779 image_bios_version = self.get_version_from_image(bios_image,
780 bios_regex)
781 if dest_bios_version != image_bios_version:
782 raise error.TestFail(
783 'Failed to update BIOS RO, version %s (expected %s)' %
784 (dest_bios_version, image_bios_version))
Dan Shi9cb0eec2014-06-03 09:04:50 -0700785 finally:
Mary Ruthven6481a9f2019-08-23 12:46:05 -0700786 if tmpd:
787 tmpd.clean()
Dan Shi9cb0eec2014-06-03 09:04:50 -0700788
789
beepsf079cfb2013-09-18 17:49:51 -0700790 def servo_install(self, image_url=None, usb_boot_timeout=USB_BOOT_TIMEOUT,
791 install_timeout=INSTALL_TIMEOUT):
Scott Zawalski62bacae2013-03-05 10:40:32 -0500792 """
793 Re-install the OS on the DUT by:
794 1) installing a test image on a USB storage device attached to the Servo
795 board,
Richard Barnette03a0c132012-11-05 12:40:35 -0800796 2) booting that image in recovery mode, and then
J. Richard Barnette31b2e312013-04-04 16:05:22 -0700797 3) installing the image with chromeos-install.
798
Scott Zawalski62bacae2013-03-05 10:40:32 -0500799 @param image_url: If specified use as the url to install on the DUT.
800 otherwise boot the currently staged image on the USB stick.
beepsf079cfb2013-09-18 17:49:51 -0700801 @param usb_boot_timeout: The usb_boot_timeout to use during reimage.
802 Factory images need a longer usb_boot_timeout than regular
803 cros images.
804 @param install_timeout: The timeout to use when installing the chromeos
805 image. Factory images need a longer install_timeout.
Richard Barnette03a0c132012-11-05 12:40:35 -0800806
Scott Zawalski62bacae2013-03-05 10:40:32 -0500807 @raises AutoservError if the image fails to boot.
beepsf079cfb2013-09-18 17:49:51 -0700808
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800809 """
beepsf079cfb2013-09-18 17:49:51 -0700810 logging.info('Downloading image to USB, then booting from it. Usb boot '
811 'timeout = %s', usb_boot_timeout)
Allen Li48a13fe2016-11-22 14:10:40 -0800812 with metrics.SecondsTimer(
813 'chromeos/autotest/provision/servo_install/boot_duration'):
814 self.servo.install_recovery_image(image_url)
815 if not self.wait_up(timeout=usb_boot_timeout):
816 raise hosts.AutoservRepairError(
817 'DUT failed to boot from USB after %d seconds' %
Garry Wang954f8382019-01-23 13:49:29 -0800818 usb_boot_timeout, 'failed_to_reboot')
Scott Zawalski62bacae2013-03-05 10:40:32 -0500819
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800820 # The new chromeos-tpm-recovery has been merged since R44-7073.0.0.
821 # In old CrOS images, this command fails. Skip the error.
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800822 logging.info('Resetting the TPM status')
Tom Wai-Hong Tamf6b4f812015-08-08 04:14:59 +0800823 try:
824 self.run('chromeos-tpm-recovery')
825 except error.AutoservRunError:
826 logging.warn('chromeos-tpm-recovery is too old.')
827
Tom Wai-Hong Tam27af7332015-07-25 06:09:39 +0800828
Allen Li48a13fe2016-11-22 14:10:40 -0800829 with metrics.SecondsTimer(
830 'chromeos/autotest/provision/servo_install/install_duration'):
831 logging.info('Installing image through chromeos-install.')
Garry Wang83fb5b32019-09-27 13:16:00 -0700832 self.run('chromeos-install --yes',timeout=install_timeout)
Anh Lee21e4032019-07-11 15:01:06 -0700833
Allen Li48a13fe2016-11-22 14:10:40 -0800834 self.halt()
beepsf079cfb2013-09-18 17:49:51 -0700835
836 logging.info('Power cycling DUT through servo.')
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800837 self.servo.get_power_state_controller().power_off()
Fang Dengafb88142013-05-30 17:44:31 -0700838 self.servo.switch_usbkey('off')
J. Richard Barnette0199cc82014-12-05 17:08:40 -0800839 # N.B. The Servo API requires that we use power_on() here
840 # for two reasons:
841 # 1) After turning on a DUT in recovery mode, you must turn
842 # it off and then on with power_on() once more to
843 # disable recovery mode (this is a Parrot specific
844 # requirement).
845 # 2) After power_off(), the only way to turn on is with
846 # power_on() (this is a Storm specific requirement).
J. Richard Barnettefbcc7122013-07-24 18:24:59 -0700847 self.servo.get_power_state_controller().power_on()
beepsf079cfb2013-09-18 17:49:51 -0700848
849 logging.info('Waiting for DUT to come back up.')
Richard Barnette03a0c132012-11-05 12:40:35 -0800850 if not self.wait_up(timeout=self.BOOT_TIMEOUT):
851 raise error.AutoservError('DUT failed to reboot installed '
852 'test image after %d seconds' %
Scott Zawalski62bacae2013-03-05 10:40:32 -0500853 self.BOOT_TIMEOUT)
854
855
Richard Barnette4aeb01c2018-09-20 09:36:12 -0700856 def set_servo_host(self, host):
857 """Set our servo host member, and associated servo.
858
859 @param host Our new `ServoHost`.
860 """
861 self._servo_host = host
862 if self._servo_host is not None:
863 self.servo = self._servo_host.get_servo()
864 else:
865 self.servo = None
866
867
Richard Barnette9a26ad62016-06-10 12:03:08 -0700868 def repair_servo(self):
Dan Shi90466352015-09-22 15:01:05 -0700869 """
Richard Barnette9a26ad62016-06-10 12:03:08 -0700870 Confirm that servo is initialized and verified.
Dan Shi90466352015-09-22 15:01:05 -0700871
Richard Barnette9a26ad62016-06-10 12:03:08 -0700872 If the servo object is missing, attempt to repair the servo
873 host. Repair failures are passed back to the caller.
874
875 @raise AutoservError: If there is no servo host for this CrOS
876 host.
877 """
878 if self.servo:
879 return
880 if not self._servo_host:
881 raise error.AutoservError('No servo host for %s.' %
882 self.hostname)
883 self._servo_host.repair()
884 self.servo = self._servo_host.get_servo()
Dan Shi90466352015-09-22 15:01:05 -0700885
886
J. Richard Barnettec2d99cf2015-11-18 12:46:15 -0800887 def repair(self):
888 """Attempt to get the DUT to pass `self.verify()`.
Richard Barnette82c35912012-11-20 10:09:10 -0800889
890 This overrides the base class function for repair; it does
J. Richard Barnette91137f02016-03-10 16:52:26 -0800891 not call back to the parent class, but instead relies on
892 `self._repair_strategy` to coordinate the verification and
893 repair steps needed to get the DUT working.
Richard Barnette82c35912012-11-20 10:09:10 -0800894 """
Richard Barnetteabbdc252018-07-26 16:57:42 -0700895 message = 'Beginning repair for host %s board %s model %s'
896 info = self.host_info_store.get()
897 message %= (self.hostname, info.board, info.model)
898 self.record('INFO', None, None, message)
J. Richard Barnette91137f02016-03-10 16:52:26 -0800899 self._repair_strategy.repair(self)
Scott Zawalski89c44dd2013-02-26 09:28:02 -0500900
Richard Barnette82c35912012-11-20 10:09:10 -0800901
J. Richard Barnette1d78b012012-05-15 13:56:30 -0700902 def close(self):
David Rileye2c6be12017-12-11 10:20:57 -0800903 """Close connection."""
Fang Deng0ca40e22013-08-27 17:47:44 -0700904 super(CrosHost, self).close()
howardchung83e55272019-08-08 14:08:05 +0800905
906 for chameleon_host in self._chameleon_host_list:
907 if chameleon_host:
908 chameleon_host.close()
xixuand6011f12016-12-08 15:01:58 -0800909
910 if self._servo_host:
911 self._servo_host.close()
J. Richard Barnette1d78b012012-05-15 13:56:30 -0700912
913
Dan Shi49ca0932014-11-14 11:22:27 -0800914 def get_power_supply_info(self):
915 """Get the output of power_supply_info.
916
917 power_supply_info outputs the info of each power supply, e.g.,
918 Device: Line Power
919 online: no
920 type: Mains
921 voltage (V): 0
922 current (A): 0
923 Device: Battery
924 state: Discharging
925 percentage: 95.9276
926 technology: Li-ion
927
928 Above output shows two devices, Line Power and Battery, with details of
929 each device listed. This function parses the output into a dictionary,
930 with key being the device name, and value being a dictionary of details
931 of the device info.
932
933 @return: The dictionary of power_supply_info, e.g.,
934 {'Line Power': {'online': 'yes', 'type': 'main'},
935 'Battery': {'vendor': 'xyz', 'percentage': '100'}}
Dan Shie9b765d2014-12-29 16:59:49 -0800936 @raise error.AutoservRunError if power_supply_info tool is not found in
937 the DUT. Caller should handle this error to avoid false failure
938 on verification.
Dan Shi49ca0932014-11-14 11:22:27 -0800939 """
940 result = self.run('power_supply_info').stdout.strip()
941 info = {}
942 device_name = None
943 device_info = {}
944 for line in result.split('\n'):
945 pair = [v.strip() for v in line.split(':')]
946 if len(pair) != 2:
947 continue
948 if pair[0] == 'Device':
949 if device_name:
950 info[device_name] = device_info
951 device_name = pair[1]
952 device_info = {}
953 else:
954 device_info[pair[0]] = pair[1]
955 if device_name and not device_name in info:
956 info[device_name] = device_info
957 return info
958
959
960 def get_battery_percentage(self):
961 """Get the battery percentage.
962
963 @return: The percentage of battery level, value range from 0-100. Return
964 None if the battery info cannot be retrieved.
965 """
966 try:
967 info = self.get_power_supply_info()
968 logging.info(info)
969 return float(info['Battery']['percentage'])
Dan Shie9b765d2014-12-29 16:59:49 -0800970 except (KeyError, ValueError, error.AutoservRunError):
Dan Shi49ca0932014-11-14 11:22:27 -0800971 return None
972
973
Daniel Campello8ca25c22019-12-13 16:48:26 -0700974 def get_battery_display_percentage(self):
975 """Get the battery display percentage.
976
977 @return: The display percentage of battery level, value range from
978 0-100. Return None if the battery info cannot be retrieved.
979 """
980 try:
981 info = self.get_power_supply_info()
982 logging.info(info)
983 return float(info['Battery']['display percentage'])
984 except (KeyError, ValueError, error.AutoservRunError):
985 return None
986
987
Dan Shi49ca0932014-11-14 11:22:27 -0800988 def is_ac_connected(self):
989 """Check if the dut has power adapter connected and charging.
990
991 @return: True if power adapter is connected and charging.
992 """
993 try:
994 info = self.get_power_supply_info()
995 return info['Line Power']['online'] == 'yes'
Dan Shie9b765d2014-12-29 16:59:49 -0800996 except (KeyError, error.AutoservRunError):
997 return None
Dan Shi49ca0932014-11-14 11:22:27 -0800998
999
Simran Basi5e6339a2013-03-21 11:34:32 -07001000 def _cleanup_poweron(self):
1001 """Special cleanup method to make sure hosts always get power back."""
Garry Wangad4d4fd2019-01-30 17:00:38 -08001002 info = self.host_info_store.get()
1003 if self._RPM_OUTLET_CHANGED not in info.attributes:
Simran Basi5e6339a2013-03-21 11:34:32 -07001004 return
1005 logging.debug('This host has recently interacted with the RPM'
1006 ' Infrastructure. Ensuring power is on.')
1007 try:
1008 self.power_on()
Garry Wangad4d4fd2019-01-30 17:00:38 -08001009 self._remove_rpm_changed_tag()
Simran Basi5e6339a2013-03-21 11:34:32 -07001010 except rpm_client.RemotePowerException:
Simran Basi5e6339a2013-03-21 11:34:32 -07001011 logging.error('Failed to turn Power On for this host after '
1012 'cleanup through the RPM Infrastructure.')
Dan Shi49ca0932014-11-14 11:22:27 -08001013
1014 battery_percentage = self.get_battery_percentage()
Dan Shif01ebe22014-12-05 13:10:57 -08001015 if battery_percentage and battery_percentage < 50:
Dan Shi49ca0932014-11-14 11:22:27 -08001016 raise
1017 elif self.is_ac_connected():
1018 logging.info('The device has power adapter connected and '
1019 'charging. No need to try to turn RPM on '
1020 'again.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001021 self._remove_rpm_changed_tag()
Dan Shi49ca0932014-11-14 11:22:27 -08001022 logging.info('Battery level is now at %s%%. The device may '
1023 'still have enough power to run test, so no '
1024 'exception will be raised.', battery_percentage)
1025
Simran Basi5e6339a2013-03-21 11:34:32 -07001026
Garry Wangad4d4fd2019-01-30 17:00:38 -08001027 def _remove_rpm_changed_tag(self):
1028 info = self.host_info_store.get()
1029 del info.attributes[self._RPM_OUTLET_CHANGED]
1030 self.host_info_store.commit(info)
1031
1032
1033 def _add_rpm_changed_tag(self):
1034 info = self.host_info_store.get()
Garry Wang518831d2019-02-21 15:15:36 -08001035 info.attributes[self._RPM_OUTLET_CHANGED] = 'true'
Garry Wangad4d4fd2019-01-30 17:00:38 -08001036 self.host_info_store.commit(info)
1037
1038
1039
beepsc87ff602013-07-31 21:53:00 -07001040 def _is_factory_image(self):
1041 """Checks if the image on the DUT is a factory image.
1042
1043 @return: True if the image on the DUT is a factory image.
1044 False otherwise.
1045 """
1046 result = self.run('[ -f /root/.factory_test ]', ignore_status=True)
1047 return result.exit_status == 0
1048
1049
1050 def _restart_ui(self):
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001051 """Restart the Chrome UI.
beepsc87ff602013-07-31 21:53:00 -07001052
1053 @raises: FactoryImageCheckerException for factory images, since
1054 we cannot attempt to restart ui on them.
1055 error.AutoservRunError for any other type of error that
1056 occurs while restarting ui.
1057 """
1058 if self._is_factory_image():
Dan Shi549fb822015-03-24 18:01:11 -07001059 raise FactoryImageCheckerException('Cannot restart ui on factory '
1060 'images')
beepsc87ff602013-07-31 21:53:00 -07001061
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001062 # TODO(jrbarnette): The command to stop/start the ui job
1063 # should live inside cros_ui, too. However that would seem
1064 # to imply interface changes to the existing start()/restart()
1065 # functions, which is a bridge too far (for now).
J. Richard Barnette6069aa12015-06-08 09:10:24 -07001066 prompt = cros_ui.get_chrome_session_ident(self)
J. Richard Barnette84890bd2014-02-21 11:05:47 -08001067 self.run('stop ui; start ui')
1068 cros_ui.wait_for_chrome_ready(prompt, self)
beepsc87ff602013-07-31 21:53:00 -07001069
1070
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001071 def _start_powerd_if_needed(self):
1072 """Start powerd if it isn't already running."""
1073 self.run('start powerd', ignore_status=True)
1074
1075
xixuana3bbc422017-05-04 15:57:21 -07001076 def _get_lsb_release_content(self):
1077 """Return the content of lsb-release file of host."""
1078 return self.run(
1079 'cat "%s"' % client_constants.LSB_RELEASE).stdout.strip()
1080
1081
Dan Shi549fb822015-03-24 18:01:11 -07001082 def get_release_version(self):
1083 """Get the value of attribute CHROMEOS_RELEASE_VERSION from lsb-release.
1084
1085 @returns The version string in lsb-release, under attribute
1086 CHROMEOS_RELEASE_VERSION.
1087 """
Dan Shi549fb822015-03-24 18:01:11 -07001088 return lsbrelease_utils.get_chromeos_release_version(
xixuana3bbc422017-05-04 15:57:21 -07001089 lsb_release_content=self._get_lsb_release_content())
1090
1091
Don Garrettb9f35802018-01-22 18:25:40 -08001092 def get_release_builder_path(self):
1093 """Get the value of CHROMEOS_RELEASE_BUILDER_PATH from lsb-release.
1094
1095 @returns The version string in lsb-release, under attribute
1096 CHROMEOS_RELEASE_BUILDER_PATH.
1097 """
1098 return lsbrelease_utils.get_chromeos_release_builder_path(
1099 lsb_release_content=self._get_lsb_release_content())
1100
1101
xixuana3bbc422017-05-04 15:57:21 -07001102 def get_chromeos_release_milestone(self):
1103 """Get the value of attribute CHROMEOS_RELEASE_BUILD_TYPE
1104 from lsb-release.
1105
1106 @returns The version string in lsb-release, under attribute
1107 CHROMEOS_RELEASE_BUILD_TYPE.
1108 """
1109 return lsbrelease_utils.get_chromeos_release_milestone(
1110 lsb_release_content=self._get_lsb_release_content())
Dan Shi549fb822015-03-24 18:01:11 -07001111
1112
1113 def verify_cros_version_label(self):
1114 """ Make sure host's cros-version label match the actual image in dut.
1115
1116 Remove any cros-version: label that doesn't match that installed in
1117 the dut.
1118
1119 @param raise_error: Set to True to raise exception if any mismatch found
1120
1121 @raise error.AutoservError: If any mismatch between cros-version label
1122 and the build installed in dut is found.
1123 """
Prathmesh Prabhuce2da3a2019-10-04 11:54:51 -07001124 # crbug.com/1007333: This check is being removed.
1125 return True
Dan Shi549fb822015-03-24 18:01:11 -07001126
1127
Laurence Goodby778c9a42017-05-24 19:24:07 -07001128 def cleanup_services(self):
1129 """Reinitializes the device for cleanup.
1130
1131 Subclasses may override this to customize the cleanup method.
1132
1133 To indicate failure of the reset, the implementation may raise
1134 any of:
1135 error.AutoservRunError
1136 error.AutotestRunError
1137 FactoryImageCheckerException
1138
1139 @raises error.AutoservRunError
1140 @raises error.AutotestRunError
1141 @raises error.FactoryImageCheckerException
1142 """
1143 self._restart_ui()
Daniel Erat4b5f7e02018-07-04 22:05:30 -07001144 self._start_powerd_if_needed()
Laurence Goodby778c9a42017-05-24 19:24:07 -07001145
1146
beepsc87ff602013-07-31 21:53:00 -07001147 def cleanup(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001148 """Cleanup state on device."""
MK Ryu35d661e2014-09-25 17:44:10 -07001149 self.run('rm -f %s' % client_constants.CLEANUP_LOGS_PAUSED_FILE)
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001150 try:
Laurence Goodby778c9a42017-05-24 19:24:07 -07001151 self.cleanup_services()
beepsc87ff602013-07-31 21:53:00 -07001152 except (error.AutotestRunError, error.AutoservRunError,
1153 FactoryImageCheckerException):
Ilja H. Friedel04be2bd2014-05-07 21:29:59 -07001154 logging.warning('Unable to restart ui, rebooting device.')
Scott Zawalskiddbc31e2012-11-15 11:29:01 -05001155 # Since restarting the UI fails fall back to normal Autotest
1156 # cleanup routines, i.e. reboot the machine.
Fang Deng0ca40e22013-08-27 17:47:44 -07001157 super(CrosHost, self).cleanup()
Simran Basi5e6339a2013-03-21 11:34:32 -07001158 # Check if the rpm outlet was manipulated.
Simran Basid5e5e272012-09-24 15:23:59 -07001159 if self.has_power():
Simran Basi5e6339a2013-03-21 11:34:32 -07001160 self._cleanup_poweron()
Dan Shi549fb822015-03-24 18:01:11 -07001161 self.verify_cros_version_label()
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001162
1163
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001164 def reboot(self, **dargs):
1165 """
1166 This function reboots the site host. The more generic
1167 RemoteHost.reboot() performs sync and sleeps for 5
1168 seconds. This is not necessary for Chrome OS devices as the
1169 sync should be finished in a short time during the reboot
1170 command.
1171 """
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001172 if 'reboot_cmd' not in dargs:
Doug Anderson7d5aeb22014-02-27 15:12:17 -08001173 reboot_timeout = dargs.get('reboot_timeout', 10)
J. Richard Barnette9af19632015-09-25 12:18:03 -07001174 dargs['reboot_cmd'] = ('sleep 1; '
1175 'reboot & sleep %d; '
1176 'reboot -f' % reboot_timeout)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001177 # Enable fastsync to avoid running extra sync commands before reboot.
Tom Wai-Hong Tamf5cd1d42012-08-13 12:04:08 +08001178 if 'fastsync' not in dargs:
1179 dargs['fastsync'] = True
Michael Liangda8c60a2014-06-03 13:24:51 -07001180
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001181 dargs['board'] = self.host_info_store.get().board
Vincent Palatindf2372c2016-10-07 17:03:00 +02001182 # Record who called us
1183 orig = sys._getframe(1).f_code
Vincent Palatin80780b22016-07-27 16:02:37 +02001184 metric_fields = {'board' : dargs['board'],
Vincent Palatindf2372c2016-10-07 17:03:00 +02001185 'dut_host_name' : self.hostname,
1186 'success' : True}
1187 metric_debug_fields = {'board' : dargs['board'],
Prathmesh Prabhu53a4c1c2018-09-14 16:36:27 -07001188 'caller' : "%s:%s" % (orig.co_filename,
1189 orig.co_name),
Vincent Palatindf2372c2016-10-07 17:03:00 +02001190 'success' : True,
1191 'error' : ''}
1192
Vincent Palatin80780b22016-07-27 16:02:37 +02001193 t0 = time.time()
1194 try:
1195 super(CrosHost, self).reboot(**dargs)
1196 except Exception as e:
1197 metric_fields['success'] = False
Vincent Palatindf2372c2016-10-07 17:03:00 +02001198 metric_debug_fields['success'] = False
1199 metric_debug_fields['error'] = type(e).__name__
Vincent Palatin80780b22016-07-27 16:02:37 +02001200 raise
1201 finally:
1202 duration = int(time.time() - t0)
Dan Shi5e2efb72017-02-07 11:40:23 -08001203 metrics.Counter(
1204 'chromeos/autotest/autoserv/reboot_count').increment(
1205 fields=metric_fields)
1206 metrics.Counter(
1207 'chromeos/autotest/autoserv/reboot_debug').increment(
1208 fields=metric_debug_fields)
1209 metrics.SecondsDistribution(
1210 'chromeos/autotest/autoserv/reboot_duration').add(
1211 duration, fields=metric_fields)
Yu-Ju Honga2be94a2012-07-31 09:48:52 -07001212
1213
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001214 def suspend(self, suspend_time=60,
1215 suspend_cmd=None, allow_early_resume=False):
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001216 """
1217 This function suspends the site host.
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001218
1219 @param suspend_time: How long to suspend as integer seconds.
1220 @param suspend_cmd: Suspend command to execute.
1221 @param allow_early_resume: If False and if device resumes before
1222 |suspend_time|, throw an error.
1223
1224 @exception AutoservSuspendError Host resumed earlier than
1225 |suspend_time|.
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001226 """
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001227
1228 if suspend_cmd is None:
1229 suspend_cmd = ' && '.join([
J. Richard Barnette9af19632015-09-25 12:18:03 -07001230 'echo 0 > /sys/class/rtc/rtc0/wakealarm',
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001231 'echo +%d > /sys/class/rtc/rtc0/wakealarm' % suspend_time,
J. Richard Barnette9af19632015-09-25 12:18:03 -07001232 'powerd_dbus_suspend --delay=0'])
Ravi Chandra Sadineni812e61b2018-07-09 11:16:50 -07001233 super(CrosHost, self).suspend(suspend_time, suspend_cmd,
1234 allow_early_resume);
Gwendal Grignou7a61d2f2014-05-23 11:05:51 -07001235
1236
Simran Basiec564392014-08-25 16:48:09 -07001237 def upstart_status(self, service_name):
1238 """Check the status of an upstart init script.
1239
1240 @param service_name: Service to look up.
1241
1242 @returns True if the service is running, False otherwise.
1243 """
Richard Barnettee204dc52017-09-26 11:02:25 -07001244 return 'start/running' in self.run('status %s' % service_name,
1245 ignore_status=True).stdout
Simran Basiec564392014-08-25 16:48:09 -07001246
Tom Hughese9552342018-12-18 14:29:25 -08001247 def upstart_stop(self, service_name):
1248 """Stops an upstart job if it's running.
1249
1250 @param service_name: Service to stop
1251
1252 @returns True if service has been stopped or was already stopped
1253 False otherwise.
1254 """
1255 if not self.upstart_status(service_name):
1256 return True
1257
1258 result = self.run('stop %s' % service_name, ignore_status=True)
1259 if result.exit_status != 0:
1260 return False
1261 return True
1262
1263 def upstart_restart(self, service_name):
1264 """Restarts (or starts) an upstart job.
1265
1266 @param service_name: Service to start/restart
1267
1268 @returns True if service has been started/restarted, False otherwise.
1269 """
1270 cmd = 'start'
1271 if self.upstart_status(service_name):
1272 cmd = 'restart'
1273 cmd = cmd + ' %s' % service_name
1274 result = self.run(cmd)
1275 if result.exit_status != 0:
1276 return False
1277 return True
Simran Basiec564392014-08-25 16:48:09 -07001278
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001279 def verify_software(self):
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001280 """Verify working software on a Chrome OS system.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001281
Richard Barnetteb2bc13c2013-01-08 17:32:51 -08001282 Tests for the following conditions:
1283 1. All conditions tested by the parent version of this
1284 function.
1285 2. Sufficient space in /mnt/stateful_partition.
Fang Deng6b05f5b2013-03-20 13:42:11 -07001286 3. Sufficient space in /mnt/stateful_partition/encrypted.
1287 4. update_engine answers a simple status request over DBus.
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001288
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001289 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001290 super(CrosHost, self).verify_software()
Dan Shib8540a52015-07-16 14:18:23 -07001291 default_kilo_inodes_required = CONFIG.get_config_value(
1292 'SERVER', 'kilo_inodes_required', type=int, default=100)
1293 board = self.get_board().replace(ds_constants.BOARD_PREFIX, '')
1294 kilo_inodes_required = CONFIG.get_config_value(
1295 'SERVER', 'kilo_inodes_required_%s' % board,
1296 type=int, default=default_kilo_inodes_required)
1297 self.check_inodes('/mnt/stateful_partition', kilo_inodes_required)
J. Richard Barnette45e93de2012-04-11 17:24:15 -07001298 self.check_diskspace(
1299 '/mnt/stateful_partition',
Dan Shib8540a52015-07-16 14:18:23 -07001300 CONFIG.get_config_value(
Fang Deng6b05f5b2013-03-20 13:42:11 -07001301 'SERVER', 'gb_diskspace_required', type=float,
1302 default=20.0))
Gaurav Shahe448af82014-06-19 15:18:59 -07001303 encrypted_stateful_path = '/mnt/stateful_partition/encrypted'
1304 # Not all targets build with encrypted stateful support.
1305 if self.path_exists(encrypted_stateful_path):
1306 self.check_diskspace(
1307 encrypted_stateful_path,
Dan Shib8540a52015-07-16 14:18:23 -07001308 CONFIG.get_config_value(
Gaurav Shahe448af82014-06-19 15:18:59 -07001309 'SERVER', 'gb_encrypted_diskspace_required', type=float,
1310 default=0.1))
beepsc87ff602013-07-31 21:53:00 -07001311
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001312 self.wait_for_system_services()
Prashanth B5d0a0512014-04-25 12:26:08 -07001313
beepsc87ff602013-07-31 21:53:00 -07001314 # Factory images don't run update engine,
1315 # goofy controls dbus on these DUTs.
1316 if not self._is_factory_image():
1317 self.run('update_engine_client --status')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001318
Dan Shi549fb822015-03-24 18:01:11 -07001319 self.verify_cros_version_label()
1320
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001321
Justin TerAvest3b0c5ba2017-11-07 09:59:11 -07001322 @retry.retry(error.AutoservError, timeout_min=5, delay_sec=10)
1323 def wait_for_system_services(self):
1324 """Waits for system-services to be running.
1325
1326 Sometimes, update_engine will take a while to update firmware, so we
1327 should give this some time to finish. See crbug.com/765686#c38 for
1328 details.
1329 """
1330 if not self.upstart_status('system-services'):
1331 raise error.AutoservError('Chrome failed to reach login. '
1332 'System services not running.')
1333
1334
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001335 def verify(self):
Pradeep Sawlaniaa013fa2017-11-09 06:34:18 -08001336 """Verify Chrome OS system is in good state."""
Richard Barnetteabbdc252018-07-26 16:57:42 -07001337 message = 'Beginning verify for host %s board %s model %s'
1338 info = self.host_info_store.get()
1339 message %= (self.hostname, info.board, info.model)
1340 self.record('INFO', None, None, message)
J. Richard Barnettea7e7fdc2016-02-12 12:35:36 -08001341 self._repair_strategy.verify(self)
1342
1343
Fang Deng96667ca2013-08-01 17:46:18 -07001344 def make_ssh_command(self, user='root', port=22, opts='', hosts_file=None,
Dean Liaoe3e75f62017-11-14 10:36:43 +08001345 connect_timeout=None, alive_interval=None,
1346 alive_count_max=None, connection_attempts=None):
Fang Deng96667ca2013-08-01 17:46:18 -07001347 """Override default make_ssh_command to use options tuned for Chrome OS.
1348
1349 Tuning changes:
1350 - ConnectTimeout=30; maximum of 30 seconds allowed for an SSH
1351 connection failure. Consistency with remote_access.sh.
1352
Samuel Tan2ce155b2015-06-23 18:24:38 -07001353 - ServerAliveInterval=900; which causes SSH to ping connection every
1354 900 seconds. In conjunction with ServerAliveCountMax ensures
1355 that if the connection dies, Autotest will bail out.
Fang Deng96667ca2013-08-01 17:46:18 -07001356 Originally tried 60 secs, but saw frequent job ABORTS where
Samuel Tan2ce155b2015-06-23 18:24:38 -07001357 the test completed successfully. Later increased from 180 seconds to
1358 900 seconds to account for tests where the DUT is suspended for
1359 longer periods of time.
Fang Deng96667ca2013-08-01 17:46:18 -07001360
1361 - ServerAliveCountMax=3; consistency with remote_access.sh.
1362
1363 - ConnectAttempts=4; reduce flakiness in connection errors;
1364 consistency with remote_access.sh.
1365
1366 - UserKnownHostsFile=/dev/null; we don't care about the keys.
1367 Host keys change with every new installation, don't waste
1368 memory/space saving them.
1369
1370 - SSH protocol forced to 2; needed for ServerAliveInterval.
1371
1372 @param user User name to use for the ssh connection.
1373 @param port Port on the target host to use for ssh connection.
1374 @param opts Additional options to the ssh command.
1375 @param hosts_file Ignored.
1376 @param connect_timeout Ignored.
1377 @param alive_interval Ignored.
Dean Liaoe3e75f62017-11-14 10:36:43 +08001378 @param alive_count_max Ignored.
1379 @param connection_attempts Ignored.
Fang Deng96667ca2013-08-01 17:46:18 -07001380 """
Dean Liaoe3e75f62017-11-14 10:36:43 +08001381 options = ' '.join([opts, '-o Protocol=2'])
1382 return super(CrosHost, self).make_ssh_command(
1383 user=user, port=port, opts=options, hosts_file='/dev/null',
1384 connect_timeout=30, alive_interval=900, alive_count_max=3,
1385 connection_attempts=4)
1386
1387
Jason Abeleb6f924f2013-11-13 16:01:54 -08001388 def syslog(self, message, tag='autotest'):
1389 """Logs a message to syslog on host.
1390
1391 @param message String message to log into syslog
1392 @param tag String tag prefix for syslog
1393
1394 """
1395 self.run('logger -t "%s" "%s"' % (tag, message))
1396
1397
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001398 def _ping_check_status(self, status):
1399 """Ping the host once, and return whether it has a given status.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001400
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001401 @param status Check the ping status against this value.
1402 @return True iff `status` and the result of ping are the same
1403 (i.e. both True or both False).
1404
1405 """
1406 ping_val = utils.ping(self.hostname, tries=1, deadline=1)
1407 return not (status ^ (ping_val == 0))
1408
1409 def _ping_wait_for_status(self, status, timeout):
1410 """Wait for the host to have a given status (UP or DOWN).
1411
1412 Status is checked by polling. Polling will not last longer
1413 than the number of seconds in `timeout`. The polling
1414 interval will be long enough that only approximately
1415 _PING_WAIT_COUNT polling cycles will be executed, subject
1416 to a maximum interval of about one minute.
1417
1418 @param status Waiting will stop immediately if `ping` of the
1419 host returns this status.
1420 @param timeout Poll for at most this many seconds.
1421 @return True iff the host status from `ping` matched the
1422 requested status at the time of return.
1423
1424 """
1425 # _ping_check_status() takes about 1 second, hence the
1426 # "- 1" in the formula below.
Nathan Ciobanu38480a32016-10-25 15:26:45 -07001427 # FIXME: if the ping command errors then _ping_check_status()
1428 # returns instantly. If timeout is also smaller than twice
1429 # _PING_WAIT_COUNT then the while loop below forks many
1430 # thousands of ping commands (see /tmp/test_that_results_XXXXX/
1431 # /results-1-logging_YYY.ZZZ/debug/autoserv.DEBUG) and hogs one
1432 # CPU core for 60 seconds.
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001433 poll_interval = min(int(timeout / self._PING_WAIT_COUNT), 60) - 1
1434 end_time = time.time() + timeout
1435 while time.time() <= end_time:
1436 if self._ping_check_status(status):
1437 return True
1438 if poll_interval > 0:
1439 time.sleep(poll_interval)
1440
1441 # The last thing we did was sleep(poll_interval), so it may
1442 # have been too long since the last `ping`. Check one more
1443 # time, just to be sure.
1444 return self._ping_check_status(status)
1445
1446 def ping_wait_up(self, timeout):
1447 """Wait for the host to respond to `ping`.
1448
1449 N.B. This method is not a reliable substitute for
1450 `wait_up()`, because a host that responds to ping will not
1451 necessarily respond to ssh. This method should only be used
1452 if the target DUT can be considered functional even if it
1453 can't be reached via ssh.
1454
1455 @param timeout Minimum time to allow before declaring the
1456 host to be non-responsive.
1457 @return True iff the host answered to ping before the timeout.
1458
1459 """
1460 return self._ping_wait_for_status(self._PING_STATUS_UP, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001461
Andrew Bresticker678c0c72013-01-22 10:44:09 -08001462 def ping_wait_down(self, timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001463 """Wait until the host no longer responds to `ping`.
1464
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001465 This function can be used as a slightly faster version of
1466 `wait_down()`, by avoiding potentially long ssh timeouts.
1467
1468 @param timeout Minimum time to allow for the host to become
1469 non-responsive.
1470 @return True iff the host quit answering ping before the
1471 timeout.
1472
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001473 """
J. Richard Barnetteb6de7e32013-02-14 13:28:04 -08001474 return self._ping_wait_for_status(self._PING_STATUS_DOWN, timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001475
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001476 def test_wait_for_sleep(self, sleep_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001477 """Wait for the client to enter low-power sleep mode.
1478
1479 The test for "is asleep" can't distinguish a system that is
1480 powered off; to confirm that the unit was asleep, it is
1481 necessary to force resume, and then call
1482 `test_wait_for_resume()`.
1483
1484 This function is expected to be called from a test as part
1485 of a sequence like the following:
1486
1487 ~~~~~~~~
1488 boot_id = host.get_boot_id()
1489 # trigger sleep on the host
1490 host.test_wait_for_sleep()
1491 # trigger resume on the host
1492 host.test_wait_for_resume(boot_id)
1493 ~~~~~~~~
1494
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001495 @param sleep_timeout time limit in seconds to allow the host sleep.
1496
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001497 @exception TestFail The host did not go to sleep within
1498 the allowed time.
1499 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001500 if sleep_timeout is None:
1501 sleep_timeout = self.SLEEP_TIMEOUT
1502
1503 if not self.ping_wait_down(timeout=sleep_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001504 raise error.TestFail(
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001505 'client failed to sleep after %d seconds' % sleep_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001506
1507
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001508 def test_wait_for_resume(self, old_boot_id, resume_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001509 """Wait for the client to resume from low-power sleep mode.
1510
1511 The `old_boot_id` parameter should be the value from
1512 `get_boot_id()` obtained prior to entering sleep mode. A
1513 `TestFail` exception is raised if the boot id changes.
1514
1515 See @ref test_wait_for_sleep for more on this function's
1516 usage.
1517
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001518 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001519 target host went to sleep.
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001520 @param resume_timeout time limit in seconds to allow the host up.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001521
1522 @exception TestFail The host did not respond within the
1523 allowed time.
1524 @exception TestFail The host responded, but the boot id test
1525 indicated a reboot rather than a sleep
1526 cycle.
1527 """
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001528 if resume_timeout is None:
1529 resume_timeout = self.RESUME_TIMEOUT
1530
1531 if not self.wait_up(timeout=resume_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001532 raise error.TestFail(
1533 'client failed to resume from sleep after %d seconds' %
Tom Wai-Hong Tamfced4f62014-04-17 10:56:30 +08001534 resume_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001535 else:
1536 new_boot_id = self.get_boot_id()
1537 if new_boot_id != old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001538 logging.error('client rebooted (old boot %s, new boot %s)',
1539 old_boot_id, new_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001540 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001541 'client rebooted, but sleep was expected')
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001542
1543
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001544 def test_wait_for_shutdown(self, shutdown_timeout=None):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001545 """Wait for the client to shut down.
1546
1547 The test for "has shut down" can't distinguish a system that
1548 is merely asleep; to confirm that the unit was down, it is
1549 necessary to force boot, and then call test_wait_for_boot().
1550
1551 This function is expected to be called from a test as part
1552 of a sequence like the following:
1553
1554 ~~~~~~~~
1555 boot_id = host.get_boot_id()
1556 # trigger shutdown on the host
1557 host.test_wait_for_shutdown()
1558 # trigger boot on the host
1559 host.test_wait_for_boot(boot_id)
1560 ~~~~~~~~
1561
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001562 @param shutdown_timeout time limit in seconds to allow the host down.
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001563 @exception TestFail The host did not shut down within the
1564 allowed time.
1565 """
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001566 if shutdown_timeout is None:
1567 shutdown_timeout = self.SHUTDOWN_TIMEOUT
1568
1569 if not self.ping_wait_down(timeout=shutdown_timeout):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001570 raise error.TestFail(
1571 'client failed to shut down after %d seconds' %
Tom Wai-Hong Tamfe005c22014-12-03 09:25:44 +08001572 shutdown_timeout)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001573
1574
1575 def test_wait_for_boot(self, old_boot_id=None):
1576 """Wait for the client to boot from cold power.
1577
1578 The `old_boot_id` parameter should be the value from
1579 `get_boot_id()` obtained prior to shutting down. A
1580 `TestFail` exception is raised if the boot id does not
1581 change. The boot id test is omitted if `old_boot_id` is not
1582 specified.
1583
1584 See @ref test_wait_for_shutdown for more on this function's
1585 usage.
1586
J. Richard Barnette7214e0b2013-02-06 15:20:49 -08001587 @param old_boot_id A boot id value obtained before the
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001588 shut down.
1589
1590 @exception TestFail The host did not respond within the
1591 allowed time.
1592 @exception TestFail The host responded, but the boot id test
1593 indicated that there was no reboot.
1594 """
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001595 if not self.wait_up(timeout=self.REBOOT_TIMEOUT):
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001596 raise error.TestFail(
1597 'client failed to reboot after %d seconds' %
J. Richard Barnetteeb69d722012-06-18 17:29:44 -07001598 self.REBOOT_TIMEOUT)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001599 elif old_boot_id:
1600 if self.get_boot_id() == old_boot_id:
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001601 logging.error('client not rebooted (boot %s)',
1602 old_boot_id)
J. Richard Barnette134ec2c2012-04-25 12:59:37 -07001603 raise error.TestFail(
Tom Wai-Hong Tam01792682015-01-06 08:00:46 +08001604 'client is back up, but did not reboot')
Simran Basid5e5e272012-09-24 15:23:59 -07001605
1606
1607 @staticmethod
1608 def check_for_rpm_support(hostname):
1609 """For a given hostname, return whether or not it is powered by an RPM.
1610
Simran Basi1df55112013-09-06 11:25:09 -07001611 @param hostname: hostname to check for rpm support.
1612
Simran Basid5e5e272012-09-24 15:23:59 -07001613 @return None if this host does not follows the defined naming format
1614 for RPM powered DUT's in the lab. If it does follow the format,
1615 it returns a regular expression MatchObject instead.
1616 """
Fang Dengbaff9082015-01-06 13:46:15 -08001617 return re.match(CrosHost._RPM_HOSTNAME_REGEX, hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001618
1619
1620 def has_power(self):
1621 """For this host, return whether or not it is powered by an RPM.
1622
1623 @return True if this host is in the CROS lab and follows the defined
1624 naming format.
1625 """
Fang Deng0ca40e22013-08-27 17:47:44 -07001626 return CrosHost.check_for_rpm_support(self.hostname)
Simran Basid5e5e272012-09-24 15:23:59 -07001627
1628
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001629 def _set_power(self, state, power_method):
Garry Wang5e5538a2019-04-08 15:36:18 -07001630 """Sets the power to the host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001631
1632 @param state Specifies which power state to set to DUT
1633 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001634 use. By default "RPM" or "CCD" will be used based
1635 on servo type. Valid values from
1636 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001637
1638 """
1639 ACCEPTABLE_STATES = ['ON', 'OFF']
1640
Garry Wang5e5538a2019-04-08 15:36:18 -07001641 if not power_method:
1642 power_method = self.get_default_power_method()
1643
1644 state = state.upper()
1645 if state not in ACCEPTABLE_STATES:
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001646 raise error.TestError('State must be one of: %s.'
1647 % (ACCEPTABLE_STATES,))
1648
1649 if power_method == self.POWER_CONTROL_SERVO:
1650 logging.info('Setting servo port J10 to %s', state)
1651 self.servo.set('prtctl3_pwren', state.lower())
1652 time.sleep(self._USB_POWER_TIMEOUT)
1653 elif power_method == self.POWER_CONTROL_MANUAL:
1654 logging.info('You have %d seconds to set the AC power to %s.',
1655 self._POWER_CYCLE_TIMEOUT, state)
1656 time.sleep(self._POWER_CYCLE_TIMEOUT)
Garry Wang5e5538a2019-04-08 15:36:18 -07001657 elif power_method == self.POWER_CONTROL_CCD:
1658 servo_role = 'src' if state == 'ON' else 'snk'
1659 logging.info('servo ccd power pass through detected,'
1660 ' changing servo_role to %s.', servo_role)
1661 self.servo.set_servo_v4_role(servo_role)
1662 if not self.ping_wait_up(timeout=self._CHANGE_SERVO_ROLE_TIMEOUT):
Garry Wang94bf9de2019-06-10 17:23:37 -07001663 # Make sure we don't leave DUT with no power(servo_role=snk)
1664 # when DUT is not pingable, as we raise a exception here
1665 # that may break a power cycle in the middle.
1666 self.servo.set_servo_v4_role('src')
Garry Wang5e5538a2019-04-08 15:36:18 -07001667 raise error.AutoservError(
1668 'DUT failed to regain network connection after %d seconds.'
1669 % self._CHANGE_SERVO_ROLE_TIMEOUT)
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001670 else:
1671 if not self.has_power():
1672 raise error.TestFail('DUT does not have RPM connected.')
Garry Wangad4d4fd2019-01-30 17:00:38 -08001673 self._add_rpm_changed_tag()
Garry Wang5e5538a2019-04-08 15:36:18 -07001674 rpm_client.set_power(self, state, timeout_mins=5)
Simran Basid5e5e272012-09-24 15:23:59 -07001675
1676
Garry Wang5e5538a2019-04-08 15:36:18 -07001677 def power_off(self, power_method=None):
1678 """Turn off power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001679
1680 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001681 use. By default "RPM" or "CCD" will be used based
1682 on servo type. Valid values from
1683 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001684
1685 """
1686 self._set_power('OFF', power_method)
Simran Basid5e5e272012-09-24 15:23:59 -07001687
1688
Garry Wang5e5538a2019-04-08 15:36:18 -07001689 def power_on(self, power_method=None):
1690 """Turn on power to this host via RPM, CCD, Servo or manual.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001691
1692 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001693 use. By default "RPM" or "CCD" will be used based
1694 on servo type. Valid values from
1695 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001696
1697 """
1698 self._set_power('ON', power_method)
1699
1700
Garry Wang5e5538a2019-04-08 15:36:18 -07001701 def power_cycle(self, power_method=None):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001702 """Cycle power to this host by turning it OFF, then ON.
1703
1704 @param power_method Specifies which method of power control to
Garry Wang5e5538a2019-04-08 15:36:18 -07001705 use. By default "RPM" or "CCD" will be used based
1706 on servo type. Valid values from
1707 POWER_CONTROL_VALID_ARGS, or None to use default.
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001708
1709 """
Garry Wang5e5538a2019-04-08 15:36:18 -07001710 if not power_method:
1711 power_method = self.get_default_power_method()
1712
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001713 if power_method in (self.POWER_CONTROL_SERVO,
Garry Wang5e5538a2019-04-08 15:36:18 -07001714 self.POWER_CONTROL_MANUAL,
1715 self.POWER_CONTROL_CCD):
Ismail Noorbasha07fdb612013-02-14 14:13:31 -08001716 self.power_off(power_method=power_method)
1717 time.sleep(self._POWER_CYCLE_TIMEOUT)
1718 self.power_on(power_method=power_method)
1719 else:
Garry Wangad4d4fd2019-01-30 17:00:38 -08001720 self._add_rpm_changed_tag()
1721 rpm_client.set_power(self, 'CYCLE')
Simran Basic6f1f7a2012-10-16 10:47:46 -07001722
1723
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001724 def get_platform_from_fwid(self):
1725 """Determine the platform from the crossystem fwid.
1726
1727 @returns a string representing this host's platform.
1728 """
1729 # Look at the firmware for non-unibuild cases or if mosys fails.
1730 crossystem = utils.Crossystem(self)
1731 crossystem.init()
1732 # Extract fwid value and use the leading part as the platform id.
1733 # fwid generally follow the format of {platform}.{firmware version}
1734 # Example: Alex.X.YYY.Z or Google_Alex.X.YYY.Z
1735 platform = crossystem.fwid().split('.')[0].lower()
1736 # Newer platforms start with 'Google_' while the older ones do not.
1737 return platform.replace('google_', '')
1738
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001739
Simran Basic6f1f7a2012-10-16 10:47:46 -07001740 def get_platform(self):
1741 """Determine the correct platform label for this host.
1742
1743 @returns a string representing this host's platform.
1744 """
C Shapiroed87c6f2018-04-19 09:13:58 -06001745 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1746 run_method=self.run)
C Shapiroed87c6f2018-04-19 09:13:58 -06001747 platform = ''
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001748 if release_info.get('CHROMEOS_RELEASE_UNIBUILD') == '1':
1749 platform = self.get_platform_from_mosys()
Mary Ruthvende14a8b2019-08-23 12:43:52 -07001750 return platform if platform else self.get_platform_from_fwid()
Simran Basic6f1f7a2012-10-16 10:47:46 -07001751
1752
Puthikorn Voravootivat0f7c3d82019-11-27 13:48:39 -08001753 def get_platform_from_mosys(self):
1754 """Get the host platform from mosys command.
1755
1756 @returns a string representing this host's platform.
1757 """
1758 cmd = 'mosys platform model'
1759 result = self.run(command=cmd, ignore_status=True)
1760 return result.stdout.strip() if result.exit_status == 0 else ''
1761
1762
Hung-ying Tyanb1328032014-04-01 14:18:54 +08001763 def get_architecture(self):
1764 """Determine the correct architecture label for this host.
1765
1766 @returns a string representing this host's architecture.
1767 """
1768 crossystem = utils.Crossystem(self)
1769 crossystem.init()
1770 return crossystem.arch()
1771
1772
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001773 def get_chrome_version(self):
1774 """Gets the Chrome version number and milestone as strings.
1775
1776 Invokes "chrome --version" to get the version number and milestone.
1777
1778 @return A tuple (chrome_ver, milestone) where "chrome_ver" is the
1779 current Chrome version number as a string (in the form "W.X.Y.Z")
1780 and "milestone" is the first component of the version number
1781 (the "W" from "W.X.Y.Z"). If the version number cannot be parsed
1782 in the "W.X.Y.Z" format, the "chrome_ver" will be the full output
1783 of "chrome --version" and the milestone will be the empty string.
1784
1785 """
MK Ryu35d661e2014-09-25 17:44:10 -07001786 version_string = self.run(client_constants.CHROME_VERSION_COMMAND).stdout
Luis Lozano40b7d0d2014-01-17 15:12:06 -08001787 return utils.parse_chrome_version(version_string)
1788
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001789
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001790 def get_ec_version(self):
1791 """Get the ec version as strings.
1792
1793 @returns a string representing this host's ec version.
1794 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001795 command = 'mosys ec info -s fw_version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001796 result = self.run(command, ignore_status=True)
1797 if result.exit_status != 0:
1798 return ''
1799 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001800
1801
1802 def get_firmware_version(self):
1803 """Get the firmware version as strings.
1804
1805 @returns a string representing this host's firmware version.
1806 """
1807 crossystem = utils.Crossystem(self)
1808 crossystem.init()
1809 return crossystem.fwid()
1810
1811
1812 def get_hardware_revision(self):
1813 """Get the hardware revision as strings.
1814
1815 @returns a string representing this host's hardware revision.
1816 """
Puthikorn Voravootivat65ad7672018-01-30 14:00:01 -08001817 command = 'mosys platform version'
Puthikorn Voravootivat720640d2018-02-16 17:32:38 -08001818 result = self.run(command, ignore_status=True)
1819 if result.exit_status != 0:
1820 return ''
1821 return result.stdout.strip()
Puthikorn Voravootivatfd132172017-11-16 18:24:38 -08001822
1823
1824 def get_kernel_version(self):
1825 """Get the kernel version as strings.
1826
1827 @returns a string representing this host's kernel version.
1828 """
1829 return self.run('uname -r').stdout.strip()
1830
1831
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08001832 def get_cpu_name(self):
1833 """Get the cpu name as strings.
1834
1835 @returns a string representing this host's cpu name.
1836 """
1837
1838 # Try get cpu name from device tree first
1839 if self.path_exists('/proc/device-tree/compatible'):
1840 command = ' | '.join(
1841 ["sed -e 's/\\x0/\\n/g' /proc/device-tree/compatible",
1842 'tail -1'])
1843 return self.run(command).stdout.strip().replace(',', ' ')
1844
1845 # Get cpu name from uname -p
1846 command = 'uname -p'
1847 ret = self.run(command).stdout.strip()
1848
1849 # 'uname -p' return variant of unknown or amd64 or x86_64 or i686
1850 # Try get cpu name from /proc/cpuinfo instead
1851 if re.match("unknown|amd64|[ix][0-9]?86(_64)?", ret, re.IGNORECASE):
1852 command = "grep model.name /proc/cpuinfo | cut -f 2 -d: | head -1"
1853 self = self.run(command).stdout.strip()
1854
1855 # Remove bloat from CPU name, for example
1856 # Intel(R) Core(TM) i5-7Y57 CPU @ 1.20GHz -> Intel Core i5-7Y57
1857 # Intel(R) Xeon(R) CPU E5-2690 v4 @ 2.60GHz -> Intel Xeon E5-2690 v4
1858 # AMD A10-7850K APU with Radeon(TM) R7 Graphics -> AMD A10-7850K
1859 # AMD GX-212JC SOC with Radeon(TM) R2E Graphics -> AMD GX-212JC
1860 trim_re = r' (@|processor|apu|soc|radeon).*|\(.*?\)| cpu'
1861 return re.sub(trim_re, '', ret, flags=re.IGNORECASE)
1862
1863
1864 def get_screen_resolution(self):
1865 """Get the screen(s) resolution as strings.
1866 In case of more than 1 monitor, return resolution for each monitor
1867 separate with plus sign.
1868
1869 @returns a string representing this host's screen(s) resolution.
1870 """
1871 command = 'for f in /sys/class/drm/*/*/modes; do head -1 $f; done'
1872 ret = self.run(command, ignore_status=True)
1873 # We might have Chromebox without a screen
1874 if ret.exit_status != 0:
1875 return ''
1876 return ret.stdout.strip().replace('\n', '+')
1877
1878
1879 def get_mem_total_gb(self):
1880 """Get total memory available in the system in GiB (2^20).
1881
1882 @returns an integer representing total memory
1883 """
1884 mem_total_kb = self.read_from_meminfo('MemTotal')
1885 kb_in_gb = float(2 ** 20)
1886 return int(round(mem_total_kb / kb_in_gb))
1887
1888
1889 def get_disk_size_gb(self):
1890 """Get size of disk in GB (10^9)
1891
1892 @returns an integer representing size of disk, 0 in Error Case
1893 """
1894 command = 'grep $(rootdev -s -d | cut -f3 -d/)$ /proc/partitions'
1895 result = self.run(command, ignore_status=True)
1896 if result.exit_status != 0:
1897 return 0
1898 _, _, block, _ = re.split(r' +', result.stdout.strip())
1899 byte_per_block = 1024.0
1900 disk_kb_in_gb = 1e9
1901 return int(int(block) * byte_per_block / disk_kb_in_gb + 0.5)
1902
1903
1904 def get_battery_size(self):
1905 """Get size of battery in Watt-hour via sysfs
1906
1907 This method assumes that battery support voltage_min_design and
1908 charge_full_design sysfs.
1909
1910 @returns a float representing Battery size, 0 if error.
1911 """
1912 # sysfs report data in micro scale
1913 battery_scale = 1e6
1914
1915 command = 'cat /sys/class/power_supply/*/voltage_min_design'
1916 result = self.run(command, ignore_status=True)
1917 if result.exit_status != 0:
1918 return 0
1919 voltage = float(result.stdout.strip()) / battery_scale
1920
1921 command = 'cat /sys/class/power_supply/*/charge_full_design'
1922 result = self.run(command, ignore_status=True)
1923 if result.exit_status != 0:
1924 return 0
1925 amphereHour = float(result.stdout.strip()) / battery_scale
1926
1927 return voltage * amphereHour
1928
1929
1930 def get_low_battery_shutdown_percent(self):
1931 """Get the percent-based low-battery shutdown threshold.
1932
1933 @returns a float representing low-battery shutdown percent, 0 if error.
1934 """
1935 ret = 0.0
1936 try:
1937 command = 'check_powerd_config --low_battery_shutdown_percent'
1938 ret = float(self.run(command).stdout)
1939 except error.CmdError:
1940 logging.debug("Can't run %s", command)
1941 except ValueError:
1942 logging.debug("Didn't get number from %s", command)
1943
1944 return ret
1945
1946
Puthikorn Voravootivat09c83d72018-08-10 15:58:32 -07001947 def has_hammer(self):
1948 """Check whether DUT has hammer device or not.
1949
1950 @returns boolean whether device has hammer or not
1951 """
1952 command = 'grep Hammer /sys/bus/usb/devices/*/product'
1953 return self.run(command, ignore_status=True).exit_status == 0
1954
1955
Niranjan Kumar34618872017-05-31 12:57:09 -07001956 def is_chrome_switch_present(self, switch):
David Haddock3ce538e2017-06-22 13:37:05 -07001957 """Returns True if the specified switch was provided to Chrome.
1958
1959 @param switch The chrome switch to search for.
1960 """
Niranjan Kumar34618872017-05-31 12:57:09 -07001961
Niranjan Kumar5f23fe92017-06-22 15:18:55 -07001962 command = 'pgrep -x -f -c "/opt/google/chrome/chrome.*%s.*"' % switch
1963 return self.run(command, ignore_status=True).exit_status == 0
Niranjan Kumar34618872017-05-31 12:57:09 -07001964
1965
1966 def oobe_triggers_update(self):
1967 """Returns True if this host has an OOBE flow during which
1968 it will perform an update check and perhaps an update.
1969 One example of such a flow is Hands-Off Zero-Touch Enrollment.
1970 As more such flows are developed, code handling them needs
1971 to be added here.
1972
1973 @return Boolean indicating whether this host's OOBE triggers an update.
1974 """
1975 return self.is_chrome_switch_present(
1976 '--enterprise-enable-zero-touch-enrollment=hands-off')
1977
1978
Kevin Chenga2619dc2016-03-28 11:42:08 -07001979 # TODO(kevcheng): change this to just return the board without the
1980 # 'board:' prefix and fix up all the callers. Also look into removing the
1981 # need for this method.
Simran Basic6f1f7a2012-10-16 10:47:46 -07001982 def get_board(self):
1983 """Determine the correct board label for this host.
1984
1985 @returns a string representing this host's board.
1986 """
1987 release_info = utils.parse_cmd_output('cat /etc/lsb-release',
1988 run_method=self.run)
J. Richard Barnetted2af5852016-02-05 15:03:10 -08001989 return (ds_constants.BOARD_PREFIX +
1990 release_info['CHROMEOS_RELEASE_BOARD'])
Simran Basic6f1f7a2012-10-16 10:47:46 -07001991
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07001992 def get_channel(self):
1993 """Determine the correct channel label for this host.
Simran Basic6f1f7a2012-10-16 10:47:46 -07001994
Po-Hsien Wang4618adf2017-10-10 14:40:21 -07001995 @returns: a string represeting this host's build channel.
1996 (stable, dev, beta). None on fail.
1997 """
1998 return lsbrelease_utils.get_chromeos_channel(
1999 lsb_release_content=self._get_lsb_release_content())
Kevin Chenga328da62016-03-31 10:49:04 -07002000
Kevin Chenga328da62016-03-31 10:49:04 -07002001 def get_power_supply(self):
2002 """
2003 Determine what type of power supply the host has
2004
2005 @returns a string representing this host's power supply.
2006 'power:battery' when the device has a battery intended for
2007 extended use
2008 'power:AC_primary' when the device has a battery not intended
2009 for extended use (for moving the machine, etc)
2010 'power:AC_only' when the device has no battery at all.
2011 """
2012 psu = self.run(command='mosys psu type', ignore_status=True)
2013 if psu.exit_status:
2014 # The psu command for mosys is not included for all platforms. The
2015 # assumption is that the device will have a battery if the command
2016 # is not found.
2017 return 'power:battery'
2018
2019 psu_str = psu.stdout.strip()
2020 if psu_str == 'unknown':
2021 return None
2022
2023 return 'power:%s' % psu_str
2024
2025
Puthikorn Voravootivat54aa53c2018-03-01 19:00:25 -08002026 def has_battery(self):
2027 """Determine if DUT has a battery.
2028
2029 Returns:
2030 Boolean, False if known not to have battery, True otherwise.
2031 """
2032 rv = True
2033 power_supply = self.get_power_supply()
2034 if power_supply == 'power:battery':
2035 _NO_BATTERY_BOARD_TYPE = ['CHROMEBOX', 'CHROMEBIT', 'CHROMEBASE']
2036 board_type = self.get_board_type()
2037 if board_type in _NO_BATTERY_BOARD_TYPE:
2038 logging.warn('Do NOT believe type %s has battery. '
2039 'See debug for mosys details', board_type)
2040 psu = self.system_output('mosys -vvvv psu type',
2041 ignore_status=True)
2042 logging.debug(psu)
2043 rv = False
2044 elif power_supply == 'power:AC_only':
2045 rv = False
2046
2047 return rv
2048
2049
Kevin Chenga328da62016-03-31 10:49:04 -07002050 def get_servo(self):
2051 """Determine if the host has a servo attached.
2052
2053 If the host has a working servo attached, it should have a servo label.
2054
2055 @return: string 'servo' if the host has servo attached. Otherwise,
2056 returns None.
2057 """
2058 return 'servo' if self._servo_host else None
2059
2060
Kevin Chenga328da62016-03-31 10:49:04 -07002061 def has_internal_display(self):
2062 """Determine if the device under test is equipped with an internal
2063 display.
2064
2065 @return: 'internal_display' if one is present; None otherwise.
2066 """
2067 from autotest_lib.client.cros.graphics import graphics_utils
2068 from autotest_lib.client.common_lib import utils as common_utils
2069
2070 def __system_output(cmd):
2071 return self.run(cmd).stdout
2072
2073 def __read_file(remote_path):
2074 return self.run('cat %s' % remote_path).stdout
2075
2076 # Hijack the necessary client functions so that we can take advantage
2077 # of the client lib here.
2078 # FIXME: find a less hacky way than this
2079 original_system_output = utils.system_output
2080 original_read_file = common_utils.read_file
2081 utils.system_output = __system_output
2082 common_utils.read_file = __read_file
2083 try:
2084 return ('internal_display' if graphics_utils.has_internal_display()
2085 else None)
2086 finally:
2087 utils.system_output = original_system_output
2088 common_utils.read_file = original_read_file
2089
2090
Dan Shi85276d42014-04-08 22:11:45 -07002091 def is_boot_from_usb(self):
2092 """Check if DUT is boot from USB.
2093
2094 @return: True if DUT is boot from usb.
2095 """
2096 device = self.run('rootdev -s -d').stdout.strip()
2097 removable = int(self.run('cat /sys/block/%s/removable' %
2098 os.path.basename(device)).stdout.strip())
2099 return removable == 1
Helen Zhang17dae2b2014-11-11 09:25:52 -08002100
2101
2102 def read_from_meminfo(self, key):
Dan Shi49ca0932014-11-14 11:22:27 -08002103 """Return the memory info from /proc/meminfo
Helen Zhang17dae2b2014-11-11 09:25:52 -08002104
2105 @param key: meminfo requested
2106
2107 @return the memory value as a string
2108
2109 """
Helen Zhang17dae2b2014-11-11 09:25:52 -08002110 meminfo = self.run('grep %s /proc/meminfo' % key).stdout.strip()
2111 logging.debug('%s', meminfo)
2112 return int(re.search(r'\d+', meminfo).group(0))
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002113
2114
Rohit Makasana98e696f2016-06-03 18:48:10 -07002115 def get_cpu_arch(self):
2116 """Returns CPU arch of the device.
2117
2118 @return CPU architecture of the DUT.
2119 """
Allen Li2c32d6b2017-02-03 15:28:10 -08002120 # Add CPUs by following logic in client/bin/utils.py.
Rohit Makasana98e696f2016-06-03 18:48:10 -07002121 if self.run("grep '^flags.*:.* lm .*' /proc/cpuinfo",
2122 ignore_status=True).stdout:
2123 return 'x86_64'
2124 if self.run("grep -Ei 'ARM|CPU implementer' /proc/cpuinfo",
2125 ignore_status=True).stdout:
2126 return 'arm'
2127 return 'i386'
2128
2129
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002130 def get_board_type(self):
2131 """
2132 Get the DUT's device type from /etc/lsb-release.
Danny Chan471a8d12015-08-18 14:57:41 -07002133 DEVICETYPE can be one of CHROMEBOX, CHROMEBASE, CHROMEBOOK or more.
2134
2135 @return value of DEVICETYPE param from lsb-release.
Rohit Makasana8a4923c2015-08-13 17:04:26 -07002136 """
Danny Chan471a8d12015-08-18 14:57:41 -07002137 device_type = self.run('grep DEVICETYPE /etc/lsb-release',
2138 ignore_status=True).stdout
2139 if device_type:
Kalin Stoyanov524310b2015-08-21 16:24:04 -07002140 return device_type.split('=')[-1].strip()
Danny Chan471a8d12015-08-18 14:57:41 -07002141 return ''
Gilad Arnolda76bef02015-09-29 13:55:15 -07002142
2143
Rohit Makasanadf0a3a32017-06-30 13:55:18 -07002144 def get_arc_version(self):
2145 """Return ARC version installed on the DUT.
2146
2147 @returns ARC version as string if the CrOS build has ARC, else None.
2148 """
2149 arc_version = self.run('grep CHROMEOS_ARC_VERSION /etc/lsb-release',
2150 ignore_status=True).stdout
2151 if arc_version:
2152 return arc_version.split('=')[-1].strip()
2153 return None
2154
2155
Gilad Arnolda76bef02015-09-29 13:55:15 -07002156 def get_os_type(self):
2157 return 'cros'
Simran Basia5522a32015-10-06 11:01:24 -07002158
2159
Kevin Chenga2619dc2016-03-28 11:42:08 -07002160 def get_labels(self):
Kevin Chengf8660142016-08-12 10:17:41 -07002161 """Return the detected labels on the host."""
Kevin Chenga2619dc2016-03-28 11:42:08 -07002162 return self.labels.get_labels(self)
Garry Wang5e5538a2019-04-08 15:36:18 -07002163
2164
2165 def get_default_power_method(self):
2166 """
2167 Get the default power method for power_on/off/cycle() methods.
2168 @return POWER_CONTROL_RPM or POWER_CONTROL_CCD
2169 """
2170 if not self._default_power_method:
Garry Wang1a004aa2019-05-16 22:56:51 -07002171 self._default_power_method = self.POWER_CONTROL_RPM
Ruben Rodriguez Buchillon3eeeab32019-10-02 15:29:58 -07002172 if self.servo and self.servo.supports_built_in_pd_control():
2173 self._default_power_method = self.POWER_CONTROL_CCD
2174 else:
2175 logging.debug('Either servo is unitialized or the servo '
2176 'setup does not support pd controls. Falling '
2177 'back to default RPM method.')
Garry Wang5e5538a2019-04-08 15:36:18 -07002178 return self._default_power_method
Puthikorn Voravootivat4a054792019-12-13 16:44:17 -08002179
2180
2181 def find_usb_devices(self, idVendor, idProduct):
2182 """
2183 Get usb device sysfs name for specific device.
2184
2185 @param idVendor Vendor ID to search in sysfs directory.
2186 @param idProduct Product ID to search in sysfs directory.
2187
2188 @return Usb node names in /sys/bus/usb/drivers/usb/ that match.
2189 """
2190 # Look for matching file and cut at position 7 to get dir name.
2191 grep_cmd = 'grep {} /sys/bus/usb/drivers/usb/*/{} | cut -f 7 -d /'
2192
2193 vendor_cmd = grep_cmd.format(idVendor, 'idVendor')
2194 product_cmd = grep_cmd.format(idProduct, 'idProduct')
2195
2196 # Use uniq -d to print duplicate line from both command
2197 cmd = 'sort <({}) <({}) | uniq -d'.format(vendor_cmd, product_cmd)
2198
2199 return self.run(cmd, ignore_status=True).stdout.strip().split('\n')
2200
2201
2202 def bind_usb_device(self, usb_node):
2203 """
2204 Bind usb device
2205
2206 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2207 """
2208 cmd = 'echo {} > /sys/bus/usb/drivers/usb/bind'.format(usb_node)
2209 self.run(cmd, ignore_status=True)
2210
2211
2212 def unbind_usb_device(self, usb_node):
2213 """
2214 Unbind usb device
2215
2216 @param usb_node Node name in /sys/bus/usb/drivers/usb/
2217 """
2218 cmd = 'echo {} > /sys/bus/usb/drivers/usb/unbind'.format(usb_node)
2219 self.run(cmd, ignore_status=True)
2220
2221
2222 def get_wlan_ip(self):
2223 """
2224 Get ip address of wlan interface.
2225
2226 @return ip address of wlan or empty string if wlan is not connected.
2227 """
2228 cmds = [
2229 'iw dev', # List wlan physical device
2230 'grep Interface', # Grep only interface name
2231 'cut -f 2 -d" "', # Cut the name part
2232 'xargs ifconfig', # Feed it to ifconfig to get ip
2233 'grep -oE "inet [0-9.]+"', # Grep only ipv4
2234 'cut -f 2 -d " "' # Cut the ip part
2235 ]
2236 return self.run(' | '.join(cmds), ignore_status=True).stdout.strip()
Puthikorn Voravootivatcd0dc9e2020-01-22 14:22:22 -08002237
2238 def connect_to_wifi(self, ssid, passphrase=None, security=None):
2239 """
2240 Connect to wifi network
2241
2242 @param ssid SSID of the wifi network.
2243 @param passphrase Passphrase of the wifi network. None if not existed.
2244 @param security Security of the wifi network. Default to "psk" if
2245 passphase is given without security. Possible values
2246 are "none", "psk", "802_1x".
2247
2248 @return True if succeed, False if not.
2249 """
2250 cmd = '/usr/local/autotest/cros/scripts/wifi connect ' + ssid
2251 if passphrase:
2252 cmd += ' ' + passphrase
2253 if security:
2254 cmd += ' ' + security
2255 return self.run(cmd, ignore_status=True).exit_status == 0