1# Copyright (c) 2011 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 time 7import sys 8 9from autotest_lib.client.common_lib import error 10from autotest_lib.server.cros.faft.firmware_test import FirmwareTest 11 12 13class firmware_FwScreenPressPower(FirmwareTest): 14 """ 15 Servo based power button triggered shutdown test during firmware screens. 16 17 This test requires a USB disk plugged-in, which contains a Chrome OS test 18 image (built by "build_image --test"). On runtime, this test triggers 19 firmware screens (developer, remove, insert, yuck, and to_norm screens), 20 and then presses the power button in order to power the machine down. 21 """ 22 version = 1 23 24 SHORT_SHUTDOWN_CONFIRMATION_PERIOD = 0.1 25 26 def wait_fw_screen_and_press_power(self): 27 """Wait for firmware warning screen and press power button.""" 28 time.sleep(self.faft_config.firmware_screen) 29 # While the firmware screen, the power button probing loop sleeps 30 # 0.25 second on every scan. Use the normal delay (1.2 second) for 31 # power press. 32 self.servo.power_normal_press() 33 34 def wait_longer_fw_screen_and_press_power(self): 35 """Wait for firmware screen without timeout and press power button.""" 36 time.sleep(self.faft_config.dev_screen_timeout) 37 self.wait_fw_screen_and_press_power() 38 39 def wait_second_screen_and_press_power(self): 40 """Wait and trigger a second screen and press power button.""" 41 self.switcher.trigger_dev_to_rec() 42 self.wait_longer_fw_screen_and_press_power() 43 44 def wait_yuck_screen_and_press_power(self): 45 """Insert corrupted USB for yuck screen and press power button.""" 46 # This USB stick will be removed in cleanup phase. 47 self.servo.switch_usbkey('dut') 48 time.sleep(self.faft_config.usb_plug) 49 self.wait_longer_fw_screen_and_press_power() 50 51 def initialize(self, host, cmdline_args): 52 super(firmware_FwScreenPressPower, self).initialize(host, cmdline_args) 53 self.assert_test_image_in_usb_disk() 54 self.switcher.setup_mode('dev') 55 self.servo.switch_usbkey('host') 56 usb_dev = self.servo.probe_host_usb_dev() 57 # Corrupt the kernel of USB stick. It is needed for triggering a 58 # yuck screen later. 59 self.corrupt_usb_kernel(usb_dev) 60 61 def cleanup(self): 62 self.servo.switch_usbkey('host') 63 usb_dev = self.servo.probe_host_usb_dev() 64 # Restore the kernel of USB stick which is corrupted on setup phase. 65 self.restore_usb_kernel(usb_dev) 66 super(firmware_FwScreenPressPower, self).cleanup() 67 68 def run_once(self): 69 if self.faft_config.fw_bypasser_type != 'ctrl_d_bypasser': 70 raise error.TestNAError("This test is only valid on devices with " 71 "screens.") 72 if not self.faft_config.has_powerbutton: 73 raise error.TestNAError("This test is only valid on devices with " 74 "power button.") 75 76 logging.info("Expected dev mode and reboot. " 77 "When the next DEVELOPER SCREEN shown, press power button " 78 "to make DUT shutdown.") 79 self.check_state((self.checkers.crossystem_checker, { 80 'devsw_boot': '1', 81 'mainfw_type': 'developer', 82 })) 83 self.switcher.simple_reboot() 84 self.run_shutdown_process(self.wait_fw_screen_and_press_power, 85 post_power_action=self.switcher.bypass_dev_mode) 86 self.switcher.wait_for_client() 87 88 logging.info("Reboot. When the developer screen shown, press " 89 "enter key to trigger either TO_NORM screen (new) or " 90 "RECOVERY INSERT screen (old). Then press power button to " 91 "make DUT shutdown.") 92 self.check_state((self.checkers.crossystem_checker, { 93 'devsw_boot': '1', 94 'mainfw_type': 'developer', 95 })) 96 self.switcher.simple_reboot() 97 self.run_shutdown_process(self.wait_second_screen_and_press_power, 98 post_power_action=self.switcher.bypass_dev_mode, 99 shutdown_timeout=self.SHORT_SHUTDOWN_CONFIRMATION_PERIOD) 100 self.switcher.wait_for_client() 101 102 logging.info("Request recovery boot. When the RECOVERY INSERT " 103 "screen shows, press power button to make DUT shutdown.") 104 self.check_state((self.checkers.crossystem_checker, { 105 'devsw_boot': '1', 106 'mainfw_type': 'developer', 107 })) 108 self.faft_client.system.request_recovery_boot() 109 self.switcher.simple_reboot() 110 self.run_shutdown_process(self.wait_longer_fw_screen_and_press_power, 111 post_power_action=self.switcher.bypass_dev_mode, 112 shutdown_timeout=self.SHORT_SHUTDOWN_CONFIRMATION_PERIOD) 113 self.switcher.wait_for_client() 114 115 logging.info("Request recovery boot again. When the recovery " 116 "insert screen shows, insert a corrupted USB and trigger " 117 "a YUCK SCREEN. Then press power button to " 118 "make DUT shutdown.") 119 self.check_state((self.checkers.crossystem_checker, { 120 'devsw_boot': '1', 121 'mainfw_type': 'developer', 122 })) 123 self.faft_client.system.request_recovery_boot() 124 self.switcher.simple_reboot() 125 self.run_shutdown_process(self.wait_yuck_screen_and_press_power, 126 post_power_action=self.switcher.bypass_dev_mode, 127 shutdown_timeout=self.SHORT_SHUTDOWN_CONFIRMATION_PERIOD) 128 self.switcher.wait_for_client() 129 130 logging.info("Switch back to normal mode.") 131 self.check_state((self.checkers.crossystem_checker, { 132 'devsw_boot': '1', 133 'mainfw_type': 'developer', 134 })) 135 self.switcher.reboot_to_mode(to_mode='normal') 136 137 logging.info("Expected normal mode and request recovery boot. " 138 "Because an USB stick is inserted, a RECOVERY REMOVE " 139 "screen shows. Press power button to make DUT shutdown.") 140 self.check_state((self.checkers.crossystem_checker, { 141 'devsw_boot': '0', 142 'mainfw_type': 'normal', 143 })) 144 self.faft_client.system.request_recovery_boot() 145 self.switcher.simple_reboot() 146 self.run_shutdown_process(self.wait_longer_fw_screen_and_press_power, 147 shutdown_timeout=self.SHORT_SHUTDOWN_CONFIRMATION_PERIOD) 148 self.switcher.wait_for_client() 149 150 logging.info("Check and done.") 151 self.check_state((self.checkers.crossystem_checker, { 152 'devsw_boot': '0', 153 'mainfw_type': 'normal', 154 })) 155