blob: 33ca9b90b54b67dc56e749bb2089202f6614a15e [file] [log] [blame]
Owen Linca365f82013-11-08 16:52:28 +08001# Copyright (c) 2013 The Chromium 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 re
7
8from autotest_lib.client.bin import utils
Owen Lin9d19b272013-11-28 12:13:24 +08009from autotest_lib.client.cros.audio import cmd_utils
Owen Linca365f82013-11-08 16:52:28 +080010
11_CRAS_TEST_CLIENT = '/usr/bin/cras_test_client'
12_RE_SELECTED_OUTPUT_NODE = re.compile('Selected Output Node: (.*)')
13_RE_SELECTED_INPUT_NODE = re.compile('Selected Input Node: (.*)')
Owen Lin7ab45a22013-11-19 17:26:33 +080014_RE_NUM_ACTIVE_STREAM = re.compile('Num active streams: (.*)')
Owen Linca365f82013-11-08 16:52:28 +080015
16def playback(*args, **kargs):
17 """A helper function to execute the playback_cmd."""
Owen Lin9d19b272013-11-28 12:13:24 +080018 cmd_utils.execute(playback_cmd(*args, **kargs))
Owen Linca365f82013-11-08 16:52:28 +080019
20def capture(*args, **kargs):
21 """A helper function to execute the capture_cmd."""
Owen Lin9d19b272013-11-28 12:13:24 +080022 cmd_utils.execute(capture_cmd(*args, **kargs))
Owen Linca365f82013-11-08 16:52:28 +080023
Owen Lin2f7e7f42013-12-25 17:25:43 +080024def playback_cmd(playback_file, block_size=None, duration=None,
Owen Lin2013e462013-12-05 17:54:42 +080025 channels=2, rate=48000):
Owen Linca365f82013-11-08 16:52:28 +080026 """Gets a command to playback a file with given settings.
27
28 @param playback_file: the name of the file to play. '-' indicates to
29 playback raw audio from the stdin.
Owen Lin2f7e7f42013-12-25 17:25:43 +080030 @param block_size: the number of frames per callback(dictates latency).
Owen Linca365f82013-11-08 16:52:28 +080031 @param duration: seconds to playback
32 @param rate: the sampling rate
33 """
34 args = [_CRAS_TEST_CLIENT]
35 args += ['--playback_file', playback_file]
Owen Lin2f7e7f42013-12-25 17:25:43 +080036 if block_size is not None:
37 args += ['--block_size', str(block_size)]
Owen Linca365f82013-11-08 16:52:28 +080038 if duration is not None:
39 args += ['--duration', str(duration)]
Owen Lin2013e462013-12-05 17:54:42 +080040 args += ['--num_channels', str(channels)]
Owen Linca365f82013-11-08 16:52:28 +080041 args += ['--rate', str(rate)]
42 return args
43
Owen Lina8129c62013-11-26 16:51:46 +080044def capture_cmd(
Owen Lin2f7e7f42013-12-25 17:25:43 +080045 capture_file, block_size=None, duration=10, channels=1, rate=48000):
Owen Linca365f82013-11-08 16:52:28 +080046 """Gets a command to capture the audio into the file with given settings.
47
48 @param capture_file: the name of file the audio to be stored in.
Owen Lin2f7e7f42013-12-25 17:25:43 +080049 @param block_size: the number of frames per callback(dictates latency).
Cheng-Yi Chiangeaf53292015-01-12 20:53:56 +080050 @param duration: seconds to record. If it is None, duration is not set,
51 and command will keep capturing audio until it is
52 terminated.
Owen Linca365f82013-11-08 16:52:28 +080053 @param rate: the sampling rate.
54 """
55 args = [_CRAS_TEST_CLIENT]
56 args += ['--capture_file', capture_file]
Owen Lin2f7e7f42013-12-25 17:25:43 +080057 if block_size is not None:
58 args += ['--block_size', str(block_size)]
Cheng-Yi Chiangeaf53292015-01-12 20:53:56 +080059 if duration is not None:
60 args += ['--duration', str(duration)]
Owen Lina8129c62013-11-26 16:51:46 +080061 args += ['--num_channels', str(channels)]
Owen Linca365f82013-11-08 16:52:28 +080062 args += ['--rate', str(rate)]
63 return args
64
Owen Lindae7a0d2013-12-05 13:34:06 +080065
66def loopback(*args, **kargs):
67 """A helper function to execute loopback_cmd."""
68 cmd_utils.execute(loopback_cmd(*args, **kargs))
69
70
Owen Lin2013e462013-12-05 17:54:42 +080071def loopback_cmd(output_file, duration=10, channels=2, rate=48000):
Owen Lindae7a0d2013-12-05 13:34:06 +080072 """Gets a command to record the loopback.
73
74 @param output_file: The name of the file the loopback to be stored in.
75 @param channels: The number of channels of the recorded audio.
76 @param duration: seconds to record.
77 @param rate: the sampling rate.
78 """
79 args = [_CRAS_TEST_CLIENT]
80 args += ['--loopback_file', output_file]
81 args += ['--duration_seconds', str(duration)]
82 args += ['--num_channels', str(channels)]
83 args += ['--rate', str(rate)]
84 return args
85
86
Owen Linca365f82013-11-08 16:52:28 +080087def set_system_volume(volume):
88 """Set the system volume.
89
90 @param volume: the system output vlume to be set(0 - 100).
91 """
92 args = [_CRAS_TEST_CLIENT]
93 args += ['--volume', str(volume)]
Owen Lin9d19b272013-11-28 12:13:24 +080094 cmd_utils.execute(args)
Owen Linca365f82013-11-08 16:52:28 +080095
96def set_node_volume(node_id, volume):
97 """Set the volume of the given output node.
98
99 @param node_id: the id of the output node to be set the volume.
100 @param volume: the volume to be set(0-100).
101 """
102 args = [_CRAS_TEST_CLIENT]
103 args += ['--set_node_volume', '%s:%d' % (node_id, volume)]
Owen Lin9d19b272013-11-28 12:13:24 +0800104 cmd_utils.execute(args)
Owen Linca365f82013-11-08 16:52:28 +0800105
106def set_capture_gain(gain):
107 """Set the system capture gain.
108 @param gain the capture gain in db*100 (100 = 1dB)
109 """
110 args = [_CRAS_TEST_CLIENT]
111 args += ['--capture_gain', str(gain)]
Owen Lin9d19b272013-11-28 12:13:24 +0800112 cmd_utils.execute(args)
Owen Linca365f82013-11-08 16:52:28 +0800113
114def dump_server_info():
115 """Gets the CRAS's server information."""
Owen Lin0d65e8a2013-11-28 14:29:54 +0800116 args = [_CRAS_TEST_CLIENT, '--dump_server_info']
Owen Linad6610a2013-12-13 11:20:48 +0800117 return cmd_utils.execute(args, stdout=cmd_utils.PIPE)
Owen Linca365f82013-11-08 16:52:28 +0800118
119def get_selected_nodes():
120 """Returns the pair of active output node and input node."""
121 server_info = dump_server_info()
122 output_match = _RE_SELECTED_OUTPUT_NODE.search(server_info)
123 input_match = _RE_SELECTED_INPUT_NODE.search(server_info)
124 if not output_match or not input_match:
125 logging.error(server_info)
126 raise RuntimeError('No match for the pattern')
127
128 return (output_match.group(1).strip(), input_match.group(1).strip())
Owen Lin7ab45a22013-11-19 17:26:33 +0800129
130def get_active_stream_count():
131 """Gets the number of active streams."""
132 server_info = dump_server_info()
133 match = _RE_NUM_ACTIVE_STREAM.search(server_info)
134 if not match:
135 logging.error(server_info)
136 raise RuntimeException('Cannot find matched pattern')
137 return int(match.group(1))
Owen Lin56050862013-12-09 11:42:51 +0800138
139
140def set_system_mute(is_mute):
141 """Sets the system mute switch.
142
143 @param is_mute: Set True to mute the system playback.
144 """
145 args = [_CRAS_TEST_CLIENT, '--mute', '1' if is_mute else '0']
146 cmd_utils.execute(args)
147
148
149def set_capture_mute(is_mute):
150 """Sets the capture mute switch.
151
152 @param is_mute: Set True to mute the capture.
153 """
154 args = [_CRAS_TEST_CLIENT, '--capture_mute', '1' if is_mute else '0']
155 cmd_utils.execute(args)
Cheng-Yi Chiangf4104ff2014-12-23 19:39:01 +0800156
157
158def node_type_is_plugged(node_type, server_info=None):
159 """Determine if there is any node of node_type plugged.
160
161 Parses server info to get the plugged state of certain node type.
162 The server info of interest is in this format:
163
164 ...(other info)...
165
166 ID Vol Plugged L/R swapped Time Type Name
167 3:0 75 yes no 1419323058 HEADPHONE *Headphone
168 4:0 0 yes no 1419323059 MIC *Mic Jack
169
170 ...(other info)...
171
172
173 @param node_type: A str representing node type. e.g. 'HEADPHONE' or
174 'MIC'.
175 @param server_info: A str containing server info. None to call
176 dump_server_info in this function.
177
178 @returns: True if there is any node of node_type plugged. False otherwise.
179
180 @raises: ValueError: if cras server info format is not as expected.
181 """
182 # The label line
183 # ID Vol Plugged L/R swapped Time Type Name
184 _MIN_LEN_LABELS = 8
185 _INDEX_LABEL_PLUGGED = 2
186 _INDEX_LABEL_TYPE = 6
187
188 # The value line
189 # 3:0 75 yes no 1419323058 HEADPHONE *Headphone
190 _MIN_LEN_VALUES = 7
191 _INDEX_VALUE_PLUGGED = 2
192 _INDEX_VALUE_TYPE = 5
193
194 if not server_info:
195 server_info = dump_server_info()
196 state = False
197 for line in server_info.splitlines():
198 line_split = line.split()
199 # Checks if a label line follows format.
200 if 'Plugged' in line_split and 'Type' in line_split:
201 if (len(line_split) < _MIN_LEN_LABELS or
202 line_split[_INDEX_LABEL_PLUGGED] != 'Plugged' or
203 line_split[_INDEX_LABEL_TYPE] != 'Type'):
204 raise ValueError('cras server info format is not as '
205 'expected')
206 if len(line_split) < _MIN_LEN_VALUES:
207 continue
208 # Checks a value line of interest.
209 # There might be other nodes of node_type, so keep searching if
210 # this node is not plugged.
211 if (line_split[_INDEX_VALUE_TYPE] == node_type and
212 line_split[_INDEX_VALUE_PLUGGED] == 'yes'):
213 state = True
214 return state