1#!/usr/bin/env python 2# Copyright 2014 The Chromium OS Authors. All rights reserved. 3# Use of this source code is governed by a BSD-style license that can be 4# found in the LICENSE file. 5 6"""This module provides audio test data.""" 7 8import logging 9import os 10import subprocess 11 12from autotest_lib.client.cros.audio import audio_data 13from autotest_lib.client.cros.audio import sox_utils 14 15 16class AudioTestDataException(Exception): 17 """Exception for audio test data.""" 18 pass 19 20 21class AudioTestData(object): 22 """Class to represent audio test data.""" 23 def __init__(self, data_format=None, path=None, frequencies=None, 24 duration_secs=None): 25 """ 26 Initializes an audio test file. 27 28 @param data_format: A dict containing data format including 29 file_type, sample_format, channel, and rate. 30 file_type: file type e.g. 'raw' or 'wav'. 31 sample_format: One of the keys in 32 audio_data.SAMPLE_FORMAT. 33 channel: number of channels. 34 rate: sampling rate. 35 @param path: The path to the file. 36 @param frequencies: A list containing the frequency of each channel in 37 this file. Only applicable to data of sine tone. 38 @param duration_secs: Duration of test file in seconds. 39 40 @raises: AudioTestDataException if the path does not exist. 41 42 """ 43 self.data_format = data_format 44 if not os.path.exists(path): 45 raise AudioTestDataException('Can not find path %s' % path) 46 self.path = path 47 self.frequencies = frequencies 48 self.duration_secs = duration_secs 49 50 51 def get_binary(self): 52 """The binary of test data. 53 54 @returns: The binary of test data. 55 56 """ 57 with open(self.path, 'rb') as f: 58 return f.read() 59 60 61 def convert(self, data_format, volume_scale): 62 """Converts the data format and returns a new AudioTestData object. 63 64 Converts the source file at self.path to a new data format. 65 The destination file path is self.path with a suffix. E.g. 66 original_path = '/tmp/test.raw' 67 data_format = dict(file_type='raw', sample_format='S32_LE', 68 channel=2, rate=48000) 69 new_path = '/tmp/test_raw_48000_S32_LE_2.raw' 70 71 This method returns a new AudioTestData object so the original object is 72 not changed. 73 74 @param data_format: A dict containing new data format. 75 @param volume_scale: A float for volume scale used in sox command. 76 E.g. 1.0 is the same. 0.5 to scale volume by 77 half. -1.0 to invert the data. 78 79 @returns: A new AudioTestData object with converted format and new path. 80 81 """ 82 original_path_without_ext, _ = os.path.splitext(self.path) 83 new_ext = '.' + data_format['file_type'] 84 # New path will be the composition of original name, new data format, 85 # and new file type as extension. 86 new_path = (original_path_without_ext + '_' + 87 '_'.join(str(x) for x in data_format.values()) + new_ext) 88 89 logging.debug('src data_format: %s', self.data_format) 90 logging.debug('dst data_format: %s', data_format) 91 92 # If source file has header, use that header. 93 if self.data_format['file_type'] != 'raw': 94 use_src_header = True 95 channels_src = None 96 rate_src = None 97 bits_src = None 98 else: 99 use_src_header = False 100 channels_src = self.data_format['channel'] 101 rate_src = self.data_format['rate'] 102 bits_src = audio_data.SAMPLE_FORMATS[ 103 self.data_format['sample_format']]['size_bytes'] * 8 104 105 # If dst file type is not raw, write file format into header of dst 106 # file. 107 use_dst_header = data_format['file_type'] != 'raw' 108 109 sox_utils.convert_format( 110 path_src=self.path, 111 channels_src=channels_src, 112 rate_src=rate_src, 113 bits_src=bits_src, 114 path_dst=new_path, 115 channels_dst=data_format['channel'], 116 rate_dst=data_format['rate'], 117 bits_dst=audio_data.SAMPLE_FORMATS[ 118 data_format['sample_format']]['size_bytes'] * 8, 119 volume_scale=volume_scale, 120 use_src_header=use_src_header, 121 use_dst_header=use_dst_header) 122 123 new_test_data = AudioTestData(path=new_path, 124 data_format=data_format) 125 126 return new_test_data 127 128 129 def delete(self): 130 """Deletes the file at self.path.""" 131 os.unlink(self.path) 132 133 134class FakeTestData(object): 135 def __init__(self, frequencies, url=None, duration_secs=None): 136 """A fake test data which contains properties but no real data. 137 138 This is useful when we need to pass an AudioTestData object into a test 139 or audio_test_utils.check_recorded_frequency. 140 141 @param frequencies: A list containing the frequency of each channel in 142 this file. Only applicable to data of sine tone. 143 @param url: The URL to the test file. 144 @param duration_secs: The duration of the file in seconds. 145 146 """ 147 self.frequencies = frequencies 148 self.url = url 149 self.duration_secs = duration_secs 150 151 152class AudioTestDataGenerateOnDemand(AudioTestData): 153 """AudioTestData that generates real data on demand.""" 154 def __init__(self, data_format=None, path=None, frequencies=None, 155 duration_secs=None): 156 """ 157 Initializes an audio test file that generate file on demand. 158 159 @param data_format: A dict containing data format including 160 file_type, sample_format, channel, and rate. 161 file_type: file type e.g. 'raw' or 'wav'. 162 sample_format: One of the keys in 163 audio_data.SAMPLE_FORMAT. 164 channel: number of channels. 165 rate: sampling rate. 166 @param path: The path to the file. 167 @param frequencies: A list containing the frequency of each channel in 168 this file. Only applicable to data of sine tone. 169 @param duration_secs: Duration of test file in seconds. 170 171 """ 172 self.data_format = data_format 173 self.path = path 174 self.frequencies = frequencies 175 self.duration_secs = duration_secs 176 177 178 def generate_file(self): 179 """Generates the data with specified format and frequencies.""" 180 sample_format = audio_data.SAMPLE_FORMATS[self.data_format['sample_format']] 181 bits = sample_format['size_bytes'] * 8 182 183 command = sox_utils.generate_sine_tone_cmd( 184 filename=self.path, 185 channels=self.data_format['channel'], 186 bits=bits, 187 rate=self.data_format['rate'], 188 duration=self.duration_secs, 189 frequencies=self.frequencies, 190 raw=(self.data_format['file_type'] == 'raw')) 191 192 subprocess.check_call(command) 193 194 195AUDIO_PATH = os.path.join(os.path.dirname(__file__)) 196 197""" 198This test data contains frequency sweep from 20Hz to 20000Hz in two channels. 199Left channel sweeps from 20Hz to 20000Hz, while right channel sweeps from 20020000Hz to 20Hz. The sweep duration is 2 seconds. The begin and end of the file 201is padded with 0.4 seconds of silence. The file is two-channel raw data with 202each sample being a signed 16-bit integer in little-endian with sampling rate 20348000 samples/sec. 204""" 205SWEEP_TEST_FILE = AudioTestData( 206 path=os.path.join(AUDIO_PATH, 'pad_sweep_pad_16.raw'), 207 data_format=dict(file_type='raw', 208 sample_format='S16_LE', 209 channel=2, 210 rate=48000)) 211 212""" 213This test data contains fixed frequency sine wave in two channels. 214Left channel is 2KHz, while right channel is 1KHz. The duration is 6 seconds. 215The file format is two-channel raw data with each sample being a signed 21616-bit integer in little-endian with sampling rate 48000 samples/sec. 217""" 218FREQUENCY_TEST_FILE = AudioTestData( 219 path=os.path.join(AUDIO_PATH, 'fix_2k_1k_16.raw'), 220 data_format=dict(file_type='raw', 221 sample_format='S16_LE', 222 channel=2, 223 rate=48000), 224 frequencies=[2000, 1000]) 225 226""" 227This test data contains fixed frequency sine wave in two channels. 228Left and right channel are both 440Hz. The duration is 10 seconds. 229The file format is two-channel raw data with each sample being a signed 23016-bit integer in little-endian with sampling rate 48000 samples/sec. 231The volume is 0.1. The small volume is to avoid distortion when played 232on Chameleon. 233""" 234SIMPLE_FREQUENCY_TEST_FILE = AudioTestData( 235 path=os.path.join(AUDIO_PATH, 'fix_440_16.raw'), 236 data_format=dict(file_type='raw', 237 sample_format='S16_LE', 238 channel=2, 239 rate=48000), 240 frequencies=[440, 440]) 241 242 243""" 244This test data contains fixed frequency sine wave in two channels. 245Left and right channel are both 660Hz. The duration is 60 seconds. 246The file format is two-channel wav data with each sample being a signed 24716-bit integer in little-endian with sampling rate 48000 samples/sec. 248The volume is 1.0. 249""" 250SIMPLE_FREQUENCY_LOUD_WAVE_FILE = AudioTestDataGenerateOnDemand( 251 path=os.path.join(AUDIO_PATH, 'fix_660_16.wav'), 252 data_format=dict(file_type='wav', 253 sample_format='S16_LE', 254 channel=2, 255 rate=48000), 256 duration_secs=60, 257 frequencies=[660, 660]) 258 259 260""" 261This test data contains fixed frequency sine wave in one channel. 262Left channel is 440Hz. The duration is 10 seconds. 263The file format is two-channel raw data with each sample being a signed 26416-bit integer in little-endian with sampling rate 48000 samples/sec. 265The volume is 0.5. 266""" 267LEFT_CHANNEL_TEST_FILE = AudioTestDataGenerateOnDemand( 268 path=os.path.join(AUDIO_PATH, 'left_440_half.raw'), 269 data_format=dict(file_type='raw', 270 sample_format='S16_LE', 271 channel=2, 272 rate=48000), 273 duration_secs=10, 274 frequencies=[440, 0]) 275 276""" 277This test data contains fixed frequency sine wave in one channel. 278Right channel is 440Hz. The duration is 10 seconds. 279The file format is two-channel raw data with each sample being a signed 28016-bit integer in little-endian with sampling rate 48000 samples/sec. 281The volume is 0.5. 282""" 283RIGHT_CHANNEL_TEST_FILE = AudioTestDataGenerateOnDemand( 284 path=os.path.join(AUDIO_PATH, 'right_440_half.raw'), 285 data_format=dict(file_type='raw', 286 sample_format='S16_LE', 287 channel=2, 288 rate=48000), 289 duration_secs=10, 290 frequencies=[0, 440]) 291 292""" 293This test data contains fixed frequency sine wave in two channels. 294Left and right channel are both 440Hz. The duration is 10 seconds. 295The file format is two-channel raw data with each sample being a signed 29616-bit integer in little-endian with sampling rate 48000 samples/sec. 297The volume is 0.5. The larger volume is needed to test internal 298speaker of Cros device because the microphone of Chameleon is not sensitive 299enough. 300""" 301SIMPLE_FREQUENCY_SPEAKER_TEST_FILE = AudioTestData( 302 path=os.path.join(AUDIO_PATH, 'fix_440_16_half.raw'), 303 data_format=dict(file_type='raw', 304 sample_format='S16_LE', 305 channel=2, 306 rate=48000), 307 frequencies=[440, 440]) 308 309""" 310Media test verification for 256Hz frequency (headphone audio). 311""" 312MEDIA_HEADPHONE_TEST_FILE = FakeTestData(frequencies=[256, 256]) 313 314""" 315Media test verification for 512Hz frequency (onboard speakers). 316""" 317MEDIA_SPEAKER_TEST_FILE = FakeTestData(frequencies=[512, 512]) 318 319""" 320Test file for 10 min playback for headphone. Left frequency is 1350Hz, right 321frequency is 870 Hz, and amplitude is 0.85. 322""" 323HEADPHONE_10MIN_TEST_FILE = FakeTestData( 324 frequencies=[1350, 870], 325 url=('http://commondatastorage.googleapis.com/chromiumos-test-assets-' 326 'public/audio_test/chameleon/Headphone/L1350_R870_A085_10min.wav'), 327 duration_secs=600) 328