Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 1 | # 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 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 5 | """This module provides cras audio utilities.""" |
| 6 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 7 | import logging |
| 8 | import re |
| 9 | |
Owen Lin | 9d19b27 | 2013-11-28 12:13:24 +0800 | [diff] [blame] | 10 | from autotest_lib.client.cros.audio import cmd_utils |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 11 | |
| 12 | _CRAS_TEST_CLIENT = '/usr/bin/cras_test_client' |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 13 | |
| 14 | def playback(*args, **kargs): |
| 15 | """A helper function to execute the playback_cmd.""" |
Owen Lin | 9d19b27 | 2013-11-28 12:13:24 +0800 | [diff] [blame] | 16 | cmd_utils.execute(playback_cmd(*args, **kargs)) |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 17 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 18 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 19 | def capture(*args, **kargs): |
| 20 | """A helper function to execute the capture_cmd.""" |
Owen Lin | 9d19b27 | 2013-11-28 12:13:24 +0800 | [diff] [blame] | 21 | cmd_utils.execute(capture_cmd(*args, **kargs)) |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 22 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 23 | |
Owen Lin | 2f7e7f4 | 2013-12-25 17:25:43 +0800 | [diff] [blame] | 24 | def playback_cmd(playback_file, block_size=None, duration=None, |
Owen Lin | 2013e46 | 2013-12-05 17:54:42 +0800 | [diff] [blame] | 25 | channels=2, rate=48000): |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 26 | """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 Lin | 2f7e7f4 | 2013-12-25 17:25:43 +0800 | [diff] [blame] | 30 | @param block_size: the number of frames per callback(dictates latency). |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 31 | @param duration: seconds to playback |
| 32 | @param rate: the sampling rate |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 33 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 34 | """ |
| 35 | args = [_CRAS_TEST_CLIENT] |
| 36 | args += ['--playback_file', playback_file] |
Owen Lin | 2f7e7f4 | 2013-12-25 17:25:43 +0800 | [diff] [blame] | 37 | if block_size is not None: |
| 38 | args += ['--block_size', str(block_size)] |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 39 | if duration is not None: |
| 40 | args += ['--duration', str(duration)] |
Owen Lin | 2013e46 | 2013-12-05 17:54:42 +0800 | [diff] [blame] | 41 | args += ['--num_channels', str(channels)] |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 42 | args += ['--rate', str(rate)] |
| 43 | return args |
| 44 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 45 | |
Owen Lin | a8129c6 | 2013-11-26 16:51:46 +0800 | [diff] [blame] | 46 | def capture_cmd( |
Owen Lin | 2f7e7f4 | 2013-12-25 17:25:43 +0800 | [diff] [blame] | 47 | capture_file, block_size=None, duration=10, channels=1, rate=48000): |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 48 | """Gets a command to capture the audio into the file with given settings. |
| 49 | |
| 50 | @param capture_file: the name of file the audio to be stored in. |
Owen Lin | 2f7e7f4 | 2013-12-25 17:25:43 +0800 | [diff] [blame] | 51 | @param block_size: the number of frames per callback(dictates latency). |
Cheng-Yi Chiang | eaf5329 | 2015-01-12 20:53:56 +0800 | [diff] [blame] | 52 | @param duration: seconds to record. If it is None, duration is not set, |
| 53 | and command will keep capturing audio until it is |
| 54 | terminated. |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 55 | @param rate: the sampling rate. |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 56 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 57 | """ |
| 58 | args = [_CRAS_TEST_CLIENT] |
| 59 | args += ['--capture_file', capture_file] |
Owen Lin | 2f7e7f4 | 2013-12-25 17:25:43 +0800 | [diff] [blame] | 60 | if block_size is not None: |
| 61 | args += ['--block_size', str(block_size)] |
Cheng-Yi Chiang | eaf5329 | 2015-01-12 20:53:56 +0800 | [diff] [blame] | 62 | if duration is not None: |
| 63 | args += ['--duration', str(duration)] |
Owen Lin | a8129c6 | 2013-11-26 16:51:46 +0800 | [diff] [blame] | 64 | args += ['--num_channels', str(channels)] |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 65 | args += ['--rate', str(rate)] |
| 66 | return args |
| 67 | |
Owen Lin | dae7a0d | 2013-12-05 13:34:06 +0800 | [diff] [blame] | 68 | |
| 69 | def loopback(*args, **kargs): |
| 70 | """A helper function to execute loopback_cmd.""" |
| 71 | cmd_utils.execute(loopback_cmd(*args, **kargs)) |
| 72 | |
| 73 | |
Owen Lin | 2013e46 | 2013-12-05 17:54:42 +0800 | [diff] [blame] | 74 | def loopback_cmd(output_file, duration=10, channels=2, rate=48000): |
Owen Lin | dae7a0d | 2013-12-05 13:34:06 +0800 | [diff] [blame] | 75 | """Gets a command to record the loopback. |
| 76 | |
| 77 | @param output_file: The name of the file the loopback to be stored in. |
| 78 | @param channels: The number of channels of the recorded audio. |
| 79 | @param duration: seconds to record. |
| 80 | @param rate: the sampling rate. |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 81 | |
Owen Lin | dae7a0d | 2013-12-05 13:34:06 +0800 | [diff] [blame] | 82 | """ |
| 83 | args = [_CRAS_TEST_CLIENT] |
| 84 | args += ['--loopback_file', output_file] |
| 85 | args += ['--duration_seconds', str(duration)] |
| 86 | args += ['--num_channels', str(channels)] |
| 87 | args += ['--rate', str(rate)] |
| 88 | return args |
| 89 | |
| 90 | |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 91 | def get_cras_nodes_cmd(): |
| 92 | """Gets a command to query the nodes from Cras. |
| 93 | |
| 94 | @returns: The command to query nodes information from Cras using dbus-send. |
| 95 | |
| 96 | """ |
| 97 | return ('dbus-send --system --type=method_call --print-reply ' |
| 98 | '--dest=org.chromium.cras /org/chromium/cras ' |
| 99 | 'org.chromium.cras.Control.GetNodes') |
| 100 | |
| 101 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 102 | def set_system_volume(volume): |
| 103 | """Set the system volume. |
| 104 | |
| 105 | @param volume: the system output vlume to be set(0 - 100). |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 106 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 107 | """ |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 108 | get_cras_control_interface().SetOutputVolume(volume) |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 109 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 110 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 111 | def set_node_volume(node_id, volume): |
| 112 | """Set the volume of the given output node. |
| 113 | |
| 114 | @param node_id: the id of the output node to be set the volume. |
| 115 | @param volume: the volume to be set(0-100). |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 116 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 117 | """ |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 118 | get_cras_control_interface().SetOutputNodeVolume(node_id, volume) |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 119 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 120 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 121 | def set_capture_gain(gain): |
| 122 | """Set the system capture gain. |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 123 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 124 | @param gain the capture gain in db*100 (100 = 1dB) |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 125 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 126 | """ |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 127 | get_cras_control_interface().SetInputGain(gain) |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 128 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 129 | |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 130 | def get_cras_control_interface(): |
| 131 | """Gets Cras DBus control interface. |
| 132 | |
| 133 | @returns: A dBus.Interface object with Cras Control interface. |
| 134 | |
| 135 | @raises: ImportError if this is not called on Cros device. |
| 136 | |
| 137 | """ |
| 138 | try: |
| 139 | import dbus |
| 140 | except ImportError, e: |
| 141 | logging.exception( |
| 142 | 'Can not import dbus: %s. This method should only be ' |
| 143 | 'called on Cros device.', e) |
| 144 | raise |
| 145 | bus = dbus.SystemBus() |
| 146 | cras_object = bus.get_object('org.chromium.cras', '/org/chromium/cras') |
| 147 | return dbus.Interface(cras_object, 'org.chromium.cras.Control') |
| 148 | |
| 149 | |
| 150 | def get_cras_nodes(): |
| 151 | """Gets nodes information from Cras. |
| 152 | |
| 153 | @returns: A dict containing information of each node. |
| 154 | |
| 155 | """ |
| 156 | return get_cras_control_interface().GetNodes() |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 157 | |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 158 | |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 159 | def get_selected_nodes(): |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 160 | """Gets selected output nodes and input nodes. |
Owen Lin | ca365f8 | 2013-11-08 16:52:28 +0800 | [diff] [blame] | 161 | |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 162 | @returns: A tuple (output_nodes, input_nodes) where each |
| 163 | field is a list of selected node IDs returned from Cras DBus API. |
| 164 | Note that there may be multiple output/input nodes being selected |
| 165 | at the same time. |
| 166 | |
| 167 | """ |
| 168 | output_nodes = [] |
| 169 | input_nodes = [] |
| 170 | nodes = get_cras_nodes() |
| 171 | for node in nodes: |
| 172 | if node['Active']: |
| 173 | if node['IsInput']: |
| 174 | input_nodes.append(node['Id']) |
| 175 | else: |
| 176 | output_nodes.append(node['Id']) |
| 177 | return (output_nodes, input_nodes) |
Owen Lin | 7ab45a2 | 2013-11-19 17:26:33 +0800 | [diff] [blame] | 178 | |
Cheng-Yi Chiang | c902f2e | 2015-03-09 10:45:44 +0800 | [diff] [blame] | 179 | |
| 180 | def set_selected_output_node_volume(volume): |
| 181 | """Sets the selected output node volume. |
| 182 | |
| 183 | @param volume: the volume to be set (0-100). |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 184 | |
Cheng-Yi Chiang | c902f2e | 2015-03-09 10:45:44 +0800 | [diff] [blame] | 185 | """ |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 186 | selected_output_node_ids, _ = get_selected_nodes() |
| 187 | for node_id in selected_output_node_ids: |
| 188 | set_node_volume(node_id, volume) |
Cheng-Yi Chiang | c902f2e | 2015-03-09 10:45:44 +0800 | [diff] [blame] | 189 | |
| 190 | |
Owen Lin | 7ab45a2 | 2013-11-19 17:26:33 +0800 | [diff] [blame] | 191 | def get_active_stream_count(): |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 192 | """Gets the number of active streams. |
| 193 | |
| 194 | @returns: The number of active streams. |
| 195 | |
| 196 | """ |
| 197 | return int(get_cras_control_interface().GetNumberOfActiveStreams()) |
Owen Lin | 5605086 | 2013-12-09 11:42:51 +0800 | [diff] [blame] | 198 | |
| 199 | |
| 200 | def set_system_mute(is_mute): |
| 201 | """Sets the system mute switch. |
| 202 | |
| 203 | @param is_mute: Set True to mute the system playback. |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 204 | |
Owen Lin | 5605086 | 2013-12-09 11:42:51 +0800 | [diff] [blame] | 205 | """ |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 206 | get_cras_control_interface().SetOutputMute(is_mute) |
Owen Lin | 5605086 | 2013-12-09 11:42:51 +0800 | [diff] [blame] | 207 | |
| 208 | |
| 209 | def set_capture_mute(is_mute): |
| 210 | """Sets the capture mute switch. |
| 211 | |
| 212 | @param is_mute: Set True to mute the capture. |
Cheng-Yi Chiang | b0ec904 | 2015-03-10 15:45:18 +0800 | [diff] [blame] | 213 | |
Owen Lin | 5605086 | 2013-12-09 11:42:51 +0800 | [diff] [blame] | 214 | """ |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 215 | get_cras_control_interface().SetInputMute(is_mute) |
Cheng-Yi Chiang | f4104ff | 2014-12-23 19:39:01 +0800 | [diff] [blame] | 216 | |
| 217 | |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 218 | def node_type_is_plugged(node_type, nodes_info): |
Cheng-Yi Chiang | f4104ff | 2014-12-23 19:39:01 +0800 | [diff] [blame] | 219 | """Determine if there is any node of node_type plugged. |
| 220 | |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 221 | This method is used in has_loopback_dongle in cros_host, where |
| 222 | the call is executed on autotest server. Use get_cras_nodes instead if |
| 223 | the call can be executed on Cros device. |
Cheng-Yi Chiang | f4104ff | 2014-12-23 19:39:01 +0800 | [diff] [blame] | 224 | |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 225 | Since Cras only reports the plugged node in GetNodes, we can |
| 226 | parse the return value to see if there is any node with the given type. |
| 227 | For example, if INTERNAL_MIC is of intereset, the pattern we are |
| 228 | looking for is: |
Cheng-Yi Chiang | f4104ff | 2014-12-23 19:39:01 +0800 | [diff] [blame] | 229 | |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 230 | dict entry( |
| 231 | string "Type" |
| 232 | variant string "INTERNAL_MIC" |
| 233 | ) |
Cheng-Yi Chiang | f4104ff | 2014-12-23 19:39:01 +0800 | [diff] [blame] | 234 | |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 235 | @param node_type: A str representing node type defined in CRAS_NODE_TYPES. |
| 236 | @param nodes_info: A str containing output of command get_nodes_cmd. |
Cheng-Yi Chiang | f4104ff | 2014-12-23 19:39:01 +0800 | [diff] [blame] | 237 | |
| 238 | @returns: True if there is any node of node_type plugged. False otherwise. |
| 239 | |
Cheng-Yi Chiang | f4104ff | 2014-12-23 19:39:01 +0800 | [diff] [blame] | 240 | """ |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 241 | match = re.search(r'string "Type"\s+variant\s+string "%s"' % node_type, |
| 242 | nodes_info) |
| 243 | return True if match else False |
Cheng-Yi Chiang | ea5a71f | 2015-03-19 21:01:24 +0800 | [diff] [blame] | 244 | |
| 245 | |
Cheng-Yi Chiang | 8de7811 | 2015-05-27 14:47:08 +0800 | [diff] [blame] | 246 | # Cras node types reported from Cras DBus control API. |
Cheng-Yi Chiang | ea5a71f | 2015-03-19 21:01:24 +0800 | [diff] [blame] | 247 | CRAS_OUTPUT_NODE_TYPES = ['HEADPHONE', 'INTERNAL_SPEAKER', 'HDMI', 'USB', |
| 248 | 'BLUETOOTH'] |
| 249 | CRAS_INPUT_NODE_TYPES = ['MIC', 'INTERNAL_MIC', 'USB', 'BLUETOOTH'] |
| 250 | CRAS_NODE_TYPES = CRAS_OUTPUT_NODE_TYPES + CRAS_INPUT_NODE_TYPES |
| 251 | |
| 252 | |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 253 | def get_selected_node_types(): |
| 254 | """Returns the pair of active output node types and input node types. |
Cheng-Yi Chiang | ea5a71f | 2015-03-19 21:01:24 +0800 | [diff] [blame] | 255 | |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 256 | @returns: A tuple (output_node_types, input_node_types) where each |
| 257 | field is a list of selected node types defined in CRAS_NODE_TYPES. |
Cheng-Yi Chiang | ea5a71f | 2015-03-19 21:01:24 +0800 | [diff] [blame] | 258 | |
| 259 | """ |
Cheng-Yi Chiang | 79b9ada | 2015-05-27 14:46:56 +0800 | [diff] [blame^] | 260 | output_node_types = [] |
| 261 | input_node_types = [] |
| 262 | nodes = get_cras_nodes() |
| 263 | for node in nodes: |
| 264 | if node['Active']: |
| 265 | node_type = str(node['Type']) |
| 266 | if node_type not in CRAS_NODE_TYPES: |
| 267 | raise RuntimeError( |
| 268 | 'node type %s is not valid' % node_type) |
| 269 | if node['IsInput']: |
| 270 | input_node_types.append(node_type) |
| 271 | else: |
| 272 | output_node_types.append(node_type) |
| 273 | return (output_node_types, input_node_types) |