OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 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 | |
5 #include "media/audio/mac/aggregate_device_manager.h" | |
6 | |
7 #include <CoreAudio/AudioHardware.h> | |
8 #include <string> | |
9 | |
10 #include "base/mac/mac_logging.h" | |
11 #include "base/mac/scoped_cftyperef.h" | |
12 #include "media/audio/audio_parameters.h" | |
13 #include "media/audio/mac/audio_manager_mac.h" | |
14 | |
15 using base::mac::ScopedCFTypeRef; | |
16 | |
17 namespace media { | |
18 | |
19 AggregateDeviceManager::AggregateDeviceManager() | |
20 : plugin_id_(kAudioObjectUnknown), | |
21 input_device_(kAudioDeviceUnknown), | |
22 output_device_(kAudioDeviceUnknown), | |
23 aggregate_device_(kAudioObjectUnknown) { | |
24 } | |
25 | |
26 AggregateDeviceManager::~AggregateDeviceManager() { | |
27 DestroyAggregateDevice(); | |
28 } | |
29 | |
30 AudioDeviceID AggregateDeviceManager::GetDefaultAggregateDevice() { | |
31 // Use a lazily created aggregate device if it's already available | |
32 // and still appropriate. | |
33 if (aggregate_device_ != kAudioObjectUnknown) { | |
34 // TODO(crogers): handle default device changes for synchronized I/O. | |
35 // For now, we check to make sure the default devices haven't changed | |
36 // since we lazily created the aggregate device. | |
37 AudioDeviceID current_input_device; | |
38 AudioDeviceID current_output_device; | |
39 AudioManagerMac::GetDefaultInputDevice(¤t_input_device); | |
40 AudioManagerMac::GetDefaultOutputDevice(¤t_output_device); | |
41 | |
42 if (current_input_device == input_device_ && | |
43 current_output_device == output_device_) | |
44 return aggregate_device_; | |
45 | |
46 // For now, once lazily created don't attempt to create another | |
47 // aggregate device. | |
DaleCurtis
2013/04/17 01:01:38
Should you destroy the old one at this point?
Chris Rogers
2013/04/17 20:42:31
I thought about this, but it would not be safe bec
| |
48 return kAudioDeviceUnknown; | |
49 } | |
50 | |
51 AudioManagerMac::GetDefaultInputDevice(&input_device_); | |
52 AudioManagerMac::GetDefaultOutputDevice(&output_device_); | |
53 | |
54 // Only create an aggregrate device if the clock domains match. | |
55 UInt32 input_clockdomain = GetClockDomain(input_device_); | |
56 UInt32 output_clockdomain = GetClockDomain(output_device_); | |
57 VLOG(1) << "input_clockdomain: " << input_clockdomain; | |
DaleCurtis
2013/04/17 01:01:38
DVLOG? Here and all other VLOG.
Chris Rogers
2013/04/17 20:42:31
Done.
| |
58 VLOG(1) << "output_clockdomain: " << output_clockdomain; | |
59 | |
60 if (input_clockdomain == 0 || input_clockdomain != output_clockdomain) | |
61 return kAudioDeviceUnknown; | |
62 | |
63 OSStatus result = CreateAggregateDevice( | |
64 input_device_, | |
65 output_device_, | |
66 &aggregate_device_); | |
67 if (result != noErr) | |
68 DestroyAggregateDevice(); | |
69 | |
70 return aggregate_device_; | |
71 } | |
72 | |
73 CFStringRef AggregateDeviceManager::GetDeviceUID(AudioDeviceID id) { | |
74 static const AudioObjectPropertyAddress kDeviceUIDAddress = { | |
75 kAudioDevicePropertyDeviceUID, | |
76 kAudioObjectPropertyScopeGlobal, | |
77 kAudioObjectPropertyElementMaster | |
78 }; | |
79 | |
80 // As stated in the CoreAudio header (AudioHardwareBase.h), | |
81 // the caller is responsible for releasing the device_UID. | |
82 CFStringRef device_UID; | |
83 UInt32 size = sizeof(device_UID); | |
84 OSStatus result = AudioObjectGetPropertyData( | |
85 id, | |
86 &kDeviceUIDAddress, | |
87 0, | |
88 0, | |
89 &size, | |
90 &device_UID); | |
91 | |
92 return (result == noErr) ? device_UID : NULL; | |
93 } | |
94 | |
95 OSStatus AggregateDeviceManager::GetDeviceName( | |
no longer working on chromium
2013/04/17 09:16:22
nit, make it void since we don't care the return v
Chris Rogers
2013/04/17 20:42:31
Done.
| |
96 AudioDeviceID id, char* name, UInt32 size) { | |
97 static const AudioObjectPropertyAddress kDeviceNameAddress = { | |
98 kAudioDevicePropertyDeviceName, | |
99 kAudioObjectPropertyScopeGlobal, | |
100 kAudioObjectPropertyElementMaster | |
101 }; | |
102 | |
103 OSStatus result = AudioObjectGetPropertyData( | |
104 id, | |
105 &kDeviceNameAddress, | |
106 0, | |
107 0, | |
108 &size, | |
109 name); | |
110 | |
111 return result; | |
112 } | |
113 | |
114 UInt32 AggregateDeviceManager::GetClockDomain(AudioDeviceID device_id) { | |
115 static const AudioObjectPropertyAddress kClockDomainAddress = { | |
116 kAudioDevicePropertyClockDomain, | |
117 kAudioObjectPropertyScopeGlobal, | |
118 kAudioObjectPropertyElementMaster | |
119 }; | |
120 | |
121 UInt32 clockdomain = 0; | |
122 UInt32 size = sizeof(UInt32); | |
123 OSStatus result = AudioObjectGetPropertyData( | |
124 device_id, | |
125 &kClockDomainAddress, | |
126 0, | |
127 0, | |
128 &size, | |
129 &clockdomain); | |
130 | |
131 return (result == noErr) ? clockdomain : 0; | |
132 } | |
133 | |
134 OSStatus AggregateDeviceManager::GetPluginID(AudioObjectID* id) { | |
135 DCHECK(id); | |
136 | |
137 // Get the audio hardware plugin. | |
138 CFStringRef bundle_name = CFSTR("com.apple.audio.CoreAudio"); | |
139 | |
140 AudioValueTranslation plugin_translation; | |
141 plugin_translation.mInputData = &bundle_name; | |
142 plugin_translation.mInputDataSize = sizeof(bundle_name); | |
143 plugin_translation.mOutputData = id; | |
144 plugin_translation.mOutputDataSize = sizeof(*id); | |
145 | |
146 static const AudioObjectPropertyAddress kPlugInAddress = { | |
147 kAudioHardwarePropertyPlugInForBundleID, | |
148 kAudioObjectPropertyScopeGlobal, | |
149 kAudioObjectPropertyElementMaster | |
150 }; | |
151 | |
152 UInt32 size = sizeof(plugin_translation); | |
153 OSStatus result = AudioObjectGetPropertyData( | |
154 kAudioObjectSystemObject, | |
155 &kPlugInAddress, | |
156 0, | |
157 0, | |
158 &size, | |
159 &plugin_translation); | |
160 | |
161 VLOG(1) << "CoreAudio plugin ID: " << *id; | |
162 | |
163 return result; | |
164 } | |
165 | |
166 CFMutableDictionaryRef | |
167 AggregateDeviceManager::CreateAggregateDeviceDictionary( | |
168 AudioDeviceID input_id, | |
169 AudioDeviceID output_id) { | |
170 CFMutableDictionaryRef aggregate_device_dict = CFDictionaryCreateMutable( | |
171 NULL, | |
172 0, | |
173 &kCFTypeDictionaryKeyCallBacks, | |
174 &kCFTypeDictionaryValueCallBacks); | |
175 if (!aggregate_device_dict) | |
176 return NULL; | |
177 | |
178 CFStringRef kAggregateDeviceName = | |
no longer working on chromium
2013/04/17 09:16:22
nit, should it be aggregate_device_name here, acco
Chris Rogers
2013/04/17 20:42:31
changed to "const"
| |
179 CFSTR("ChromeAggregateAudioDevice"); | |
180 CFStringRef kAggregateDeviceUID = | |
181 CFSTR("com.google.chrome.AggregateAudioDevice"); | |
182 | |
183 // Add name and UID of the device to the dictionary. | |
184 CFDictionaryAddValue( | |
185 aggregate_device_dict, | |
186 CFSTR(kAudioAggregateDeviceNameKey), | |
187 kAggregateDeviceName); | |
188 CFDictionaryAddValue( | |
189 aggregate_device_dict, | |
190 CFSTR(kAudioAggregateDeviceUIDKey), | |
191 kAggregateDeviceUID); | |
192 | |
193 // Add a "private aggregate key" to the dictionary. | |
194 // The 1 value means that the created aggregate device will | |
195 // only be accessible from the process that created it, and | |
196 // won't be visible to outside processes. | |
197 int value = 1; | |
198 CFNumberRef kAggregateDeviceNumber = CFNumberCreate( | |
no longer working on chromium
2013/04/17 09:16:22
is it a leak here? I guess you might need to call
Chris Rogers
2013/04/17 20:42:31
Yes, good catch - Fixed
| |
199 NULL, | |
200 kCFNumberIntType, | |
201 &value); | |
202 CFDictionaryAddValue( | |
203 aggregate_device_dict, | |
204 CFSTR(kAudioAggregateDeviceIsPrivateKey), | |
205 kAggregateDeviceNumber); | |
206 | |
207 return aggregate_device_dict; | |
208 } | |
209 | |
210 CFMutableArrayRef | |
211 AggregateDeviceManager::CreateSubDeviceArray( | |
212 CFStringRef input_device_UID, CFStringRef output_device_UID) { | |
213 CFMutableArrayRef sub_devices_array = CFArrayCreateMutable( | |
214 NULL, | |
215 0, | |
216 &kCFTypeArrayCallBacks); | |
217 | |
218 CFArrayAppendValue(sub_devices_array, input_device_UID); | |
219 CFArrayAppendValue(sub_devices_array, output_device_UID); | |
220 | |
221 return sub_devices_array; | |
222 } | |
223 | |
224 OSStatus AggregateDeviceManager::CreateAggregateDevice( | |
225 AudioDeviceID input_id, | |
226 AudioDeviceID output_id, | |
227 AudioDeviceID* aggregate_device) { | |
228 DCHECK(aggregate_device); | |
229 | |
230 const size_t kMaxDeviceNameLength = 256; | |
231 char input_device_name_[kMaxDeviceNameLength]; | |
DaleCurtis
2013/04/17 01:01:38
dynamically allocate instead?
Chris Rogers
2013/04/17 20:42:31
Done.
| |
232 GetDeviceName( | |
233 input_id, | |
234 input_device_name_, | |
235 sizeof(input_device_name_)); | |
236 VLOG(1) << "Input device: \n" << input_device_name_; | |
237 | |
238 char output_device_name_[kMaxDeviceNameLength]; | |
DaleCurtis
2013/04/17 01:01:38
Ditto.
Chris Rogers
2013/04/17 20:42:31
Done.
| |
239 GetDeviceName( | |
240 output_id, | |
241 output_device_name_, | |
242 sizeof(output_device_name_)); | |
243 VLOG(1) << "Output device: \n" << output_device_name_; | |
244 | |
245 OSStatus result = GetPluginID(&plugin_id_); | |
246 if (result != noErr) | |
247 return result; | |
248 | |
249 // Create a dictionary for the aggregate device. | |
250 ScopedCFTypeRef<CFMutableDictionaryRef> aggregate_device_dict( | |
251 CreateAggregateDeviceDictionary(input_id, output_id)); | |
252 if (aggregate_device_dict == NULL) | |
DaleCurtis
2013/04/17 01:01:38
!aggregate_device_dict
Chris Rogers
2013/04/17 20:42:31
Done.
| |
253 return -1; | |
254 | |
255 // Create the aggregate device. | |
256 static const AudioObjectPropertyAddress kCreateAggregateDeviceAddress = { | |
257 kAudioPlugInCreateAggregateDevice, | |
258 kAudioObjectPropertyScopeGlobal, | |
259 kAudioObjectPropertyElementMaster | |
260 }; | |
261 | |
262 UInt32 size = sizeof(*aggregate_device); | |
263 result = AudioObjectGetPropertyData( | |
264 plugin_id_, | |
265 &kCreateAggregateDeviceAddress, | |
266 sizeof(aggregate_device_dict), | |
267 &aggregate_device_dict, | |
268 &size, | |
269 aggregate_device); | |
270 if (result != noErr) { | |
271 DLOG(ERROR) << "Error creating aggregate audio device!"; | |
272 return result; | |
273 } | |
274 | |
275 // Set the sub-devices for the aggregate device. | |
276 // In this case we use two: the input and output devices. | |
277 | |
278 ScopedCFTypeRef<CFStringRef> input_device_UID(GetDeviceUID(input_id)); | |
279 ScopedCFTypeRef<CFStringRef> output_device_UID(GetDeviceUID(output_id)); | |
280 if (input_device_UID == NULL || output_device_UID == NULL) { | |
DaleCurtis
2013/04/17 01:01:38
use ! instead.
Chris Rogers
2013/04/17 20:42:31
Done.
| |
281 DLOG(ERROR) << "Error getting audio device UID strings."; | |
282 return -1; | |
283 } | |
284 | |
285 ScopedCFTypeRef<CFMutableArrayRef> sub_devices_array( | |
286 CreateSubDeviceArray(input_device_UID, output_device_UID)); | |
287 if (sub_devices_array == NULL) { | |
288 DLOG(ERROR) << "Error creating sub-devices array."; | |
289 return -1; | |
290 } | |
291 | |
292 static const AudioObjectPropertyAddress kSetSubDevicesAddress = { | |
293 kAudioAggregateDevicePropertyFullSubDeviceList, | |
294 kAudioObjectPropertyScopeGlobal, | |
295 kAudioObjectPropertyElementMaster | |
296 }; | |
297 | |
298 size = sizeof(CFMutableArrayRef); | |
299 result = AudioObjectSetPropertyData( | |
300 *aggregate_device, | |
301 &kSetSubDevicesAddress, | |
302 0, | |
303 NULL, | |
304 size, | |
305 &sub_devices_array); | |
306 if (result != noErr) { | |
307 DLOG(ERROR) << "Error setting aggregate audio device sub-devices!"; | |
308 return result; | |
309 } | |
310 | |
311 // Use the input device as the master device. | |
312 static const AudioObjectPropertyAddress kSetMasterDeviceAddress = { | |
313 kAudioAggregateDevicePropertyMasterSubDevice, | |
314 kAudioObjectPropertyScopeGlobal, | |
315 kAudioObjectPropertyElementMaster | |
316 }; | |
317 | |
318 size = sizeof(CFStringRef); | |
319 result = AudioObjectSetPropertyData( | |
320 *aggregate_device, | |
321 &kSetMasterDeviceAddress, | |
322 0, | |
323 NULL, | |
324 size, | |
325 &input_device_UID); | |
326 if (result != noErr) { | |
327 DLOG(ERROR) << "Error setting aggregate audio device master device!"; | |
328 return result; | |
329 } | |
330 | |
331 VLOG(1) << "New aggregate device: " << *aggregate_device; | |
332 return noErr; | |
333 } | |
334 | |
335 void AggregateDeviceManager::DestroyAggregateDevice() { | |
336 if (aggregate_device_ == kAudioObjectUnknown) | |
337 return; | |
338 | |
339 static const AudioObjectPropertyAddress kDestroyAddress = { | |
340 kAudioPlugInDestroyAggregateDevice, | |
341 kAudioObjectPropertyScopeGlobal, | |
342 kAudioObjectPropertyElementMaster | |
343 }; | |
344 | |
345 UInt32 size = sizeof(aggregate_device_); | |
346 OSStatus result = AudioObjectGetPropertyData( | |
347 plugin_id_, | |
348 &kDestroyAddress, | |
349 0, | |
350 NULL, | |
351 &size, | |
352 &aggregate_device_); | |
353 if (result != noErr) { | |
354 DLOG(ERROR) << "Error destroying aggregate audio device!"; | |
355 return; | |
356 } | |
357 | |
358 aggregate_device_ = kAudioObjectUnknown; | |
359 } | |
360 | |
361 } // namespace media | |
OLD | NEW |