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 os 7import tempfile 8 9from autotest_lib.client.common_lib import file_utils 10from autotest_lib.client.cros.chameleon import avsync_probe_utils 11from autotest_lib.server import test 12from autotest_lib.server.cros.multimedia import remote_facade_factory 13 14 15class audiovideo_AVSyncInternalDisplayAudioJack(test.test): 16 """Server side audio/video sync quality measurement. 17 18 This test measure the audio/video sync between internal display and audio 19 jack. 20 21 """ 22 version = 1 23 24 def run_once(self, host, video_url, capture_seconds, video_fps, 25 sound_interval_frames, perf_prefix): 26 """Running audio/video synchronization quality measurement 27 28 @param host: A host object representing the DUT. 29 @param video_url: The ULR of the test video. 30 @param capture_seconds: How long do we capture the data from 31 avsync_probe device. 32 @param video_fps: Video frames per second of the video. We need the 33 data to detect corrupted video frame. 34 @param sound_interval_frames: The period of sound (beep) in the number 35 of video frames. 36 @param perf_prefix: The prefix name of perf graph. 37 38 """ 39 factory = remote_facade_factory.RemoteFacadeFactory( 40 host, results_dir=self.resultsdir, no_chrome=True) 41 42 chameleon_board = host.chameleon 43 audio_facade = factory.create_audio_facade() 44 browser_facade = factory.create_browser_facade() 45 video_facade = factory.create_video_facade() 46 avsync_probe = chameleon_board.get_avsync_probe() 47 chameleon_board.setup_and_reset(self.outputdir) 48 49 browser_facade.start_default_chrome() 50 51 _, ext = os.path.splitext(video_url) 52 with tempfile.NamedTemporaryFile(prefix='playback_', suffix=ext) as f: 53 # The default permission is 0o600. 54 os.chmod(f.name, 0o644) 55 56 file_utils.download_file(video_url, f.name) 57 video_facade.prepare_playback(f.name) 58 59 audio_facade.set_chrome_active_volume(100) 60 video_facade.start_playback() 61 capture_data = avsync_probe.Capture(capture_seconds) 62 parser = avsync_probe_utils.AVSyncProbeDataParser( 63 self.resultsdir, capture_data, video_fps, sound_interval_frames) 64 65 logging.info('Video frame stats:') 66 logging.info('average: %f', parser.video_duration_average) 67 logging.info('standard deviation: %f', parser.video_duration_std) 68 logging.info('Sync stats:') 69 logging.info('average: %f', parser.sync_duration_average) 70 logging.info('standard deviation: %f', parser.sync_duration_std) 71 logging.info('Number of total frames: %d', 72 parser.cumulative_frame_count) 73 logging.info('Number of corrupted frames: %d', 74 parser.corrupted_frame_count) 75 logging.info('Number of dropoped frames: %d', 76 parser.dropped_frame_count) 77 logging.info('Number of dropoped frames by player: %d', 78 video_facade.dropped_frame_count()) 79 80 video_graph_name = '%s_video' % perf_prefix 81 audiovideo_graph_name = '%s_audiovideo' % perf_prefix 82 self.output_perf_value(description='Video frame duration average', 83 value=parser.video_duration_average, units='ms', 84 graph=video_graph_name) 85 self.output_perf_value(description='Video frame duration std', 86 value=parser.video_duration_std, 87 graph=video_graph_name) 88 self.output_perf_value(description='Corrupted video frames', 89 value=parser.corrupted_frame_count, 90 higher_is_better=False, graph=video_graph_name) 91 self.output_perf_value(description='Dropped video frames', 92 value=parser.dropped_frame_count, 93 higher_is_better=False, graph=video_graph_name) 94 self.output_perf_value(description='Dropped video frames by player', 95 value=video_facade.dropped_frame_count(), 96 higher_is_better=False, graph=video_graph_name) 97 98 self.output_perf_value(description='Audio/Video Sync duration average', 99 value=parser.sync_duration_average, units='ms', 100 higher_is_better=False, 101 graph=audiovideo_graph_name) 102 self.output_perf_value(description='Audio/Video Sync std', 103 value=parser.sync_duration_std, 104 higher_is_better=False, 105 graph=audiovideo_graph_name) 106