1# Copyright (c) 2012 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, re, time
6
7from autotest_lib.server import test
8from autotest_lib.server.cros import stress
9from autotest_lib.server.cros.servo import servo
10from autotest_lib.client.common_lib import error
11
12_WAIT_DELAY = 5
13
14class platform_ExternalUSBBootStress(test.test):
15    """Uses servo to repeatedly connect/remove USB devices during boot."""
16    version = 1
17
18    def run_once(self, host, reboots):
19        reboots = int(reboots)
20        self.client = host
21        # The servo hubs come up as diffs in connected components.  These
22        # should be ignored for this test.  It is a list so when servo next
23        # is available it may have a differnet hub which can be appended.
24        servo_hardware_list = ['Standard Microsystems Corp.']
25
26
27        def strip_lsusb_output(lsusb_output):
28            items = lsusb_output.split('\n')
29            named_list = []
30            unnamed_device_count = 0
31            for item in items:
32                columns = item.split(' ')
33                if len(columns) == 6 or len(' '.join(columns[6:]).strip()) == 0:
34                    logging.info('Unnamed device located, adding generic name.')
35                    name = 'Unnamed device %d' % unnamed_device_count
36                    unnamed_device_count += 1
37                else:
38                    name = ' '.join(columns[6:]).strip()
39                if name not in servo_hardware_list:
40                    named_list.append(name)
41            return named_list
42
43
44        def set_hub_power(on=True, check_host_detection=False):
45            reset = 'off'
46            if not on:
47                reset = 'on'
48            host.servo.set('dut_hub1_rst1', reset)
49            if check_host_detection:
50                time.sleep(_WAIT_DELAY)
51                return strip_lsusb_output(host.run('lsusb').stdout.strip())
52
53
54        def stress_hotplug():
55            # Devices need some time to come up and to be recognized.  However
56            # this is a stress test so we want to move reasonably fast.
57            time.sleep(2)
58            removed = set_hub_power(False)
59            time.sleep(1)
60            connected = set_hub_power()
61
62
63        host.servo.switch_usbkey('dut')
64        host.servo.set('usb_mux_sel3', 'dut_sees_usbkey')
65
66        # There are some mice that need the data and power connection to both
67        # be removed, otherwise they won't come back up.  This means that the
68        # external devices should only use the usb connections labeled:
69        # USB_KEY and DUT_HUB1_USB.
70        connected = set_hub_power(check_host_detection=True)
71        off_list = set_hub_power(on=False, check_host_detection=True)
72        diff_list = set(connected).difference(set(off_list))
73        if len(diff_list) == 0:
74            raise error.TestError('No connected devices were detected.  Make '
75                                  'sure the devices are connected to USB_KEY '
76                                  'and DUT_HUB1_USB on the servo board.')
77        logging.info('Connected devices list: %s' % diff_list)
78        set_hub_power(True)
79
80        lsb_release = host.run('cat /etc/lsb-release').stdout.split('\n')
81        unsupported_gbb_boards = ['x86-mario', 'x86-alex', 'x86-zgb']
82        skip_gbb = False
83        for line in lsb_release:
84            m = re.match(r'^CHROMEOS_RELEASE_BOARD=(.+)$', line)
85            if m and m.group(1) in unsupported_gbb_boards:
86                skip_gbb = True
87                break
88
89        logging.info('Rebooting the device %d time(s)' % reboots)
90        for i in xrange(reboots):
91            # We want fast boot past the dev screen
92            if not skip_gbb:
93                host.run('/usr/share/vboot/bin/set_gbb_flags.sh 0x01')
94            stressor = stress.ControlledStressor(stress_hotplug)
95            logging.info('Reboot iteration %d of %d' % (i + 1, reboots))
96            if skip_gbb:
97                # For devices that do not support gbb we have servo
98                # accelerate booting through dev mode.
99                host.servo.get_power_state_controller().reset()
100                host.servo.power_short_press()
101                time.sleep(servo.Servo.BOOT_DELAY)
102                host.servo.ctrl_d()
103                stressor.start()
104                host.wait_up(timeout=120)
105            else:
106                stressor.start()
107                self.client.reboot()
108            logging.info('Reboot complete, shutting down stressor.')
109            stressor.stop()
110            connected_now = set_hub_power(check_host_detection=True)
111            diff_now = set(connected_now).difference(set(off_list))
112            if diff_list != diff_now:
113                raise error.TestFail('The list of connected items does not '
114                                      'match the master list.\nMaster: %s\n'
115                                      'Current: %s' %
116                                      (diff_list, diff_now))
117            logging.info('Connected devices for iteration %d: %s' %
118                         (i, diff_now))
119