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 hashlib, logging, os, time 6 7 8from autotest_lib.client.bin import test 9from autotest_lib.client.common_lib import error, file_utils 10from autotest_lib.client.common_lib.cros import chrome 11from autotest_lib.client.cros import service_stopper 12from autotest_lib.client.cros.power import power_status, power_utils 13 14_DOWNLOAD_BASE = ('http://commondatastorage.googleapis.com/' 15 'chromiumos-test-assets-public/audio_power/') 16 17# Minimum battery charge percentage to run the test 18BATTERY_INITIAL_CHARGED_MIN = 10 19 20# Measurement duration in seconds. 21MEASUREMENT_DURATION = 150 22 23POWER_DESCRIPTION = 'avg_energy_rate_' 24 25# Time to exclude from calculation after playing audio [seconds]. 26STABILIZATION_DURATION = 10 27 28 29class audio_PlaybackPower(test.test): 30 """Captures power usage for audio playback.""" 31 32 version = 1 33 34 35 def initialize(self): 36 self._service_stopper = None 37 self._backlight = None 38 39 def run_power_test(self, audio_type): 40 """ 41 Captures power usage and reports it to the perf dashboard. 42 43 @param audio_type: audio format label to attach with perf keyval. 44 """ 45 46 self._backlight = power_utils.Backlight() 47 self._backlight.set_default() 48 49 self._service_stopper = service_stopper.ServiceStopper( 50 service_stopper.ServiceStopper.POWER_DRAW_SERVICES) 51 self._service_stopper.stop_services() 52 53 self._power_status = power_status.get_status() 54 # Verify that we are running on battery and the battery is sufficiently 55 # charged. 56 self._power_status.assert_battery_state(BATTERY_INITIAL_CHARGED_MIN) 57 58 measurements = [power_status.SystemPower( 59 self._power_status.battery_path)] 60 61 def get_power(): 62 power_logger = power_status.PowerLogger(measurements) 63 power_logger.start() 64 time.sleep(STABILIZATION_DURATION) 65 start_time = time.time() 66 time.sleep(MEASUREMENT_DURATION) 67 power_logger.checkpoint('result', start_time) 68 keyval = power_logger.calc() 69 logging.info('Power output %s', keyval) 70 return keyval['result_' + measurements[0].domain + '_pwr'] 71 72 energy_rate = get_power() 73 perf_keyval = {} 74 perf_keyval[POWER_DESCRIPTION + audio_type] = energy_rate 75 self.output_perf_value(description=POWER_DESCRIPTION + audio_type, 76 value=energy_rate, units='W', 77 higher_is_better=False) 78 self.write_perf_keyval(perf_keyval) 79 80 81 def run_once(self, test_file, checksum): 82 local_path = os.path.join(self.bindir, '%s' % test_file) 83 file_utils.download_file(_DOWNLOAD_BASE + test_file, local_path) 84 logging.info('Downloaded file: %s. Expected checksum: %s', 85 local_path, checksum) 86 with open(local_path, 'r') as r: 87 md5sum = hashlib.md5(r.read()).hexdigest() 88 if md5sum != checksum: 89 raise error.TestError('unmatched md5 sum: %s' % md5sum) 90 with chrome.Chrome(init_network_controller=True) as cr: 91 cr.browser.platform.SetHTTPServerDirectories(self.bindir) 92 url = cr.browser.platform.http_server.UrlOf(local_path) 93 self.play_audio(cr.browser.tabs[0], url) 94 self.run_power_test(url.split('.')[-1]) 95 96 97 def play_audio(self, tab, url): 98 """Navigates to an audio file over http and plays it in loop. 99 100 @param tab: tab to open an audio stream. 101 @param url: audio/video test url. 102 """ 103 tab.Navigate(url) 104 tab.ExecuteJavaScript( 105 "document.getElementsByTagName('video')[0].loop=true") 106 tab.ExecuteJavaScript( 107 "document.getElementsByTagName('video')[0].volume=1") 108 109 110 def cleanup(self): 111 # cleanup() is run by common_lib/test.py. 112 if self._backlight: 113 self._backlight.restore() 114 if self._service_stopper: 115 self._service_stopper.restore_services() 116 117 super(audio_PlaybackPower, self).cleanup() 118