1# Copyright 2016 The Chromium OS Authors. All rights reserved. 2# Use of this source code is governed by a BSD-style license that can be 3# found in the LICENSE file. 4 5import logging 6import re 7 8from collections import defaultdict 9 10from autotest_lib.client.common_lib import error 11from autotest_lib.server.cros.faft.firmware_test import FirmwareTest 12 13class firmware_PDProtocol(FirmwareTest): 14 """ 15 Servo based USB PD protocol test. 16 17 A charger must be connected to the DUT for this test. 18 19 This test checks that when an appropriate zinger charger is connected that 20 the PD is properly negotiated in dev mode and when booted from a test image 21 through recovery that the PD is not negotiated. 22 23 Example: 24 PD Successfully negotiated 25 - ectool usbpdpower should output Charger PD 26 27 PD not negotiated 28 - ectool usbpdpower should not output Charger PD 29 30 """ 31 version = 1 32 33 NEGOTIATED_PATTERN = 'Charger PD' 34 PD_NOT_SUPPORTED_PATTERN = 'INVALID_COMMAND' 35 36 ECTOOL_CMD_DICT = defaultdict(lambda: 'ectool usbpdpower') 37 38 def initialize(self, host, cmdline_args): 39 super(firmware_PDProtocol, self).initialize(host, cmdline_args) 40 41 self.ECTOOL_CMD_DICT['samus'] = 'ectool --dev=1 usbpdpower' 42 43 self.current_board = self.servo.get_board(); 44 45 self.check_if_pd_supported() 46 self.assert_test_image_in_usb_disk() 47 self.switcher.setup_mode('dev') 48 self.setup_usbkey(usbkey=True, host=False) 49 50 self.original_dev_boot_usb = self.faft_client.system.get_dev_boot_usb() 51 logging.info('Original dev_boot_usb value: %s', 52 str(self.original_dev_boot_usb)) 53 54 def cleanup(self): 55 self.ensure_internal_device_boot() 56 super(firmware_PDProtocol, self).cleanup() 57 58 def check_if_pd_supported(self): 59 """ Checks if the DUT responds to ectool usbpdpower and skips the test 60 if it isn't supported on the device. 61 """ 62 output = self.run_command(self.ECTOOL_CMD_DICT[self.current_board]) 63 64 if (not output or 65 self.check_ec_output(output, self.PD_NOT_SUPPORTED_PATTERN)): 66 raise error.TestNAError("PD not supported skipping test.") 67 68 def ensure_internal_device_boot(self): 69 """Ensure internal device boot; if not, reboot into it. 70 71 If not, it may be a test failure during step 2 or 3, try to reboot 72 and press Ctrl-D to internal device boot. 73 """ 74 if self.faft_client.system.is_removable_device_boot(): 75 logging.info('Reboot into internal disk...') 76 self.faft_client.system.set_dev_boot_usb(self.original_dev_boot_usb) 77 self.switcher.mode_aware_reboot() 78 79 self.check_state((self.checkers.dev_boot_usb_checker, 80 False, 81 'Did not boot from internal disk.')) 82 83 def boot_to_recovery(self): 84 """Boot device into recovery mode.""" 85 logging.info('Reboot into Recovery...') 86 self.assert_test_image_in_usb_disk() 87 self.switcher.reboot_to_mode(to_mode='rec') 88 89 self.check_state((self.checkers.crossystem_checker, 90 {'mainfw_type': 'recovery'})) 91 92 def run_command(self, command): 93 """Runs the specified command and returns the output 94 as a list of strings. 95 96 @param command: The command to run on the DUT 97 @return A list of strings of the command output 98 """ 99 logging.info('Command to run: %s', command) 100 101 output = self.faft_client.system.run_shell_command_get_output(command) 102 103 logging.info('Command output: %s', output) 104 105 return output 106 107 def check_ec_output(self, output, pattern): 108 """Checks if any line in the output matches the given pattern. 109 110 @param output: A list of strings containg the output to search 111 @param pattern: The regex to search the output for 112 113 @return True upon first match found or False 114 """ 115 logging.info('Checking %s for %s.', output, pattern) 116 117 for line in output: 118 if bool(re.search(pattern, line)): 119 return True 120 121 return False 122 123 124 def run_once(self): 125 self.ensure_internal_device_boot() 126 output = self.run_command(self.ECTOOL_CMD_DICT[self.current_board]) 127 128 if not self.check_ec_output(output, self.NEGOTIATED_PATTERN): 129 raise error.TestFail( 130 'ectool usbpdpower output %s did not match %s', 131 (output, self.NEGOTIATED_PATTERN)) 132 133 134 self.boot_to_recovery() 135 output = self.run_command(self.ECTOOL_CMD_DICT[self.current_board]) 136 137 if self.check_ec_output(output, self.NEGOTIATED_PATTERN): 138 raise error.TestFail( 139 'ectool usbpdpower output %s matched %s', 140 (output, self.NEGOTIATED_PATTERN)) 141 142