1# Copyright 2015 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 os 7import time 8 9from autotest_lib.client.bin import test 10from autotest_lib.client.bin import utils 11from autotest_lib.client.common_lib.cros import chrome 12from autotest_lib.client.common_lib import error 13from autotest_lib.client.cros.audio import cras_utils 14 15_CRASH_PATH = '/var/spool/crash' 16_JS = """ 17var c = new AudioContext(); 18var o = c.createOscillator(); 19o.connect(c.destination); o.start(); 20""" 21 22class audio_CrasSanity(test.test): 23 """Verifies cras sanity using its status, active streams and crashes""" 24 25 version = 1 26 _check = {'crashes_on_boot': False, 27 'stream_activation': False, 28 'cras_status': False, 29 'crashes_at_end': False 30 } 31 32 def run_once(self): 33 # Check for existing cras crashes which might occur during UI bring up. 34 # TODO: (rohitbm) check if we need to reboot the DUT before the test 35 # start to verify cras crashes during boot. 36 existing_crash_reports = self.collect_cras_crash() 37 if len(existing_crash_reports) == 0: 38 self._check['crashes_on_boot'] = True 39 40 # Capturing cras pid before startig the test. 41 cras_pid_1 = utils.get_oldest_pid_by_name('/usr/bin/cras') 42 43 with chrome.Chrome(init_network_controller=True) as self._cr: 44 # Push the 1st stream 45 self.push_new_stream(self._cr.browser.tabs.New()) 46 47 # Capturing cras pid before opening a new set of audio streams. 48 cras_pid_2 = utils.get_oldest_pid_by_name('/usr/bin/cras') 49 50 # Push the 2nd stream 51 self.push_new_stream(self._cr.browser.tabs.New()) 52 53 # Let's play audio for sometime to ensure that 54 # long playback is good. 55 time.sleep(10) 56 57 total_tests = 2 58 active_streams = cras_utils.get_active_stream_count() 59 logging.debug( 60 'Number of active streams after opening all tabs: %d.', 61 active_streams) 62 if active_streams >= total_tests: 63 self._check['stream_activation'] = True 64 65 # Capturing cras pid after opening all audio/video streams. 66 cras_pid_3 = utils.get_oldest_pid_by_name('/usr/bin/cras') 67 68 # Close all open audio streams. 69 while total_tests > 0: 70 self._cr.browser.tabs[total_tests].Close() 71 total_tests -= 1 72 time.sleep(1) 73 active_streams = cras_utils.get_active_stream_count() 74 logging.debug( 75 'Number of active streams after closing all tabs: %d.', 76 active_streams) 77 78 # Capturing cras pid after closing all audio/stream streams. 79 cras_pid_4 = utils.get_oldest_pid_by_name('/usr/bin/cras') 80 81 if cras_pid_1 == cras_pid_2 == cras_pid_3 == cras_pid_4: 82 self._check['cras_status'] = True 83 84 new_crash_reports = self.collect_cras_crash() 85 new_reports = list(set(new_crash_reports) - 86 set(existing_crash_reports)) 87 if len(new_reports) == 0: 88 self._check['crashes_at_end'] = True 89 90 err_msg = '' 91 if self._check.values().count(False) > 0: 92 if not self._check['crashes_on_boot']: 93 err_msg = ('1. Found cras crashes on boot: %s.\n' 94 % existing_crash_reports) 95 if not self._check['stream_activation']: 96 err_msg += ('2. CRAS stream count is not matching with ' 97 'number of streams.\n') 98 if not self._check['cras_status']: 99 err_msg += ('CRAS PID changed during the test. CRAS might be ' 100 'crashing while adding/removing streams.\n') 101 if not self._check['crashes_at_end']: 102 err_msg += ('Found cras crashes at the end of the test : %s.' % 103 new_reports) 104 raise error.TestError(err_msg) 105 106 107 def push_new_stream(self, tab): 108 """Starts next audio stream from self._streams list. 109 110 @param tab: tab to open an audio stream. 111 """ 112 tab.Activate() 113 tab.Navigate("file:///") 114 tab.ExecuteJavaScript(_JS) 115 time.sleep(1) # Adding a delay so cras can update the active count. 116 117 118 def collect_cras_crash(self): 119 """Check for cras crashes. 120 121 @return a list of cras crash reports found. 122 """ 123 124 crash_reports = [] 125 if not os.path.isdir(_CRASH_PATH): 126 logging.debug('No cras crash detected!') 127 else: 128 cras_reports = os.listdir(_CRASH_PATH) 129 crash_reports = [report for report in cras_reports 130 if report.startswith('cras')] 131 return crash_reports 132