153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2009,2010 Daniel Mack <daniel@caiaq.de> 553a5a1b3Sopenharmony_ci 653a5a1b3Sopenharmony_ci PulseAudio is free software; you can redistribute it and/or modify 753a5a1b3Sopenharmony_ci it under the terms of the GNU Lesser General Public License as published 853a5a1b3Sopenharmony_ci by the Free Software Foundation; either version 2.1 of the License, 953a5a1b3Sopenharmony_ci or (at your option) any later version. 1053a5a1b3Sopenharmony_ci 1153a5a1b3Sopenharmony_ci PulseAudio is distributed in the hope that it will be useful, but 1253a5a1b3Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 1353a5a1b3Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 1453a5a1b3Sopenharmony_ci General Public License for more details. 1553a5a1b3Sopenharmony_ci 1653a5a1b3Sopenharmony_ci You should have received a copy of the GNU Lesser General Public License 1753a5a1b3Sopenharmony_ci along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 1853a5a1b3Sopenharmony_ci***/ 1953a5a1b3Sopenharmony_ci 2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H 2153a5a1b3Sopenharmony_ci#include <config.h> 2253a5a1b3Sopenharmony_ci#endif 2353a5a1b3Sopenharmony_ci 2453a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h> 2553a5a1b3Sopenharmony_ci 2653a5a1b3Sopenharmony_ci#include <pulsecore/module.h> 2753a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h> 2853a5a1b3Sopenharmony_ci#include <pulsecore/modargs.h> 2953a5a1b3Sopenharmony_ci#include <pulsecore/log.h> 3053a5a1b3Sopenharmony_ci#include <pulsecore/llist.h> 3153a5a1b3Sopenharmony_ci 3253a5a1b3Sopenharmony_ci#include <CoreAudio/CoreAudio.h> 3353a5a1b3Sopenharmony_ci 3453a5a1b3Sopenharmony_ci#define DEVICE_MODULE_NAME "module-coreaudio-device" 3553a5a1b3Sopenharmony_ci 3653a5a1b3Sopenharmony_ciPA_MODULE_AUTHOR("Daniel Mack"); 3753a5a1b3Sopenharmony_ciPA_MODULE_DESCRIPTION("CoreAudio device detection"); 3853a5a1b3Sopenharmony_ciPA_MODULE_VERSION(PACKAGE_VERSION); 3953a5a1b3Sopenharmony_ciPA_MODULE_LOAD_ONCE(true); 4053a5a1b3Sopenharmony_ciPA_MODULE_USAGE("ioproc_frames=<passed on to module-coreaudio-device> " 4153a5a1b3Sopenharmony_ci "record=<enable source?> " 4253a5a1b3Sopenharmony_ci "playback=<enable sink?> "); 4353a5a1b3Sopenharmony_ci 4453a5a1b3Sopenharmony_cistatic const char* const valid_modargs[] = { 4553a5a1b3Sopenharmony_ci "ioproc_frames", 4653a5a1b3Sopenharmony_ci "record", 4753a5a1b3Sopenharmony_ci "playback", 4853a5a1b3Sopenharmony_ci NULL 4953a5a1b3Sopenharmony_ci}; 5053a5a1b3Sopenharmony_ci 5153a5a1b3Sopenharmony_citypedef struct ca_device ca_device; 5253a5a1b3Sopenharmony_ci 5353a5a1b3Sopenharmony_cistruct ca_device { 5453a5a1b3Sopenharmony_ci AudioObjectID id; 5553a5a1b3Sopenharmony_ci unsigned int module_index; 5653a5a1b3Sopenharmony_ci PA_LLIST_FIELDS(ca_device); 5753a5a1b3Sopenharmony_ci}; 5853a5a1b3Sopenharmony_ci 5953a5a1b3Sopenharmony_cistruct userdata { 6053a5a1b3Sopenharmony_ci int detect_fds[2]; 6153a5a1b3Sopenharmony_ci pa_io_event *detect_io; 6253a5a1b3Sopenharmony_ci unsigned int ioproc_frames; 6353a5a1b3Sopenharmony_ci bool record; 6453a5a1b3Sopenharmony_ci bool playback; 6553a5a1b3Sopenharmony_ci PA_LLIST_HEAD(ca_device, devices); 6653a5a1b3Sopenharmony_ci}; 6753a5a1b3Sopenharmony_ci 6853a5a1b3Sopenharmony_cistatic int ca_device_added(struct pa_module *m, AudioObjectID id) { 6953a5a1b3Sopenharmony_ci AudioObjectPropertyAddress property_address; 7053a5a1b3Sopenharmony_ci OSStatus err; 7153a5a1b3Sopenharmony_ci pa_module *mod; 7253a5a1b3Sopenharmony_ci struct userdata *u; 7353a5a1b3Sopenharmony_ci struct ca_device *dev; 7453a5a1b3Sopenharmony_ci char *args, tmp[64]; 7553a5a1b3Sopenharmony_ci UInt32 size; 7653a5a1b3Sopenharmony_ci 7753a5a1b3Sopenharmony_ci pa_assert(m); 7853a5a1b3Sopenharmony_ci pa_assert_se(u = m->userdata); 7953a5a1b3Sopenharmony_ci 8053a5a1b3Sopenharmony_ci /* To prevent generating a black hole that will suck us in, 8153a5a1b3Sopenharmony_ci don't create sources/sinks for PulseAudio virtual devices */ 8253a5a1b3Sopenharmony_ci 8353a5a1b3Sopenharmony_ci property_address.mSelector = kAudioDevicePropertyDeviceManufacturer; 8453a5a1b3Sopenharmony_ci property_address.mScope = kAudioObjectPropertyScopeGlobal; 8553a5a1b3Sopenharmony_ci property_address.mElement = kAudioObjectPropertyElementMaster; 8653a5a1b3Sopenharmony_ci 8753a5a1b3Sopenharmony_ci size = sizeof(tmp); 8853a5a1b3Sopenharmony_ci err = AudioObjectGetPropertyData(id, &property_address, 0, NULL, &size, tmp); 8953a5a1b3Sopenharmony_ci 9053a5a1b3Sopenharmony_ci if (!err && pa_streq(tmp, "pulseaudio.org")) 9153a5a1b3Sopenharmony_ci return 0; 9253a5a1b3Sopenharmony_ci 9353a5a1b3Sopenharmony_ci if (u->ioproc_frames) 9453a5a1b3Sopenharmony_ci args = pa_sprintf_malloc("object_id=%d ioproc_frames=%d record=%d playback=%d", (int) id, u->ioproc_frames, (int) u->record, (int) u->playback); 9553a5a1b3Sopenharmony_ci else 9653a5a1b3Sopenharmony_ci args = pa_sprintf_malloc("object_id=%d record=%d playback=%d", (int) id, (int) u->record, (int) u->playback); 9753a5a1b3Sopenharmony_ci 9853a5a1b3Sopenharmony_ci pa_log_debug("Loading %s with arguments '%s'", DEVICE_MODULE_NAME, args); 9953a5a1b3Sopenharmony_ci pa_module_load(&mod, m->core, DEVICE_MODULE_NAME, args); 10053a5a1b3Sopenharmony_ci pa_xfree(args); 10153a5a1b3Sopenharmony_ci 10253a5a1b3Sopenharmony_ci if (!mod) { 10353a5a1b3Sopenharmony_ci pa_log_info("Failed to load module %s with arguments '%s'", DEVICE_MODULE_NAME, args); 10453a5a1b3Sopenharmony_ci return -1; 10553a5a1b3Sopenharmony_ci } 10653a5a1b3Sopenharmony_ci 10753a5a1b3Sopenharmony_ci dev = pa_xnew0(ca_device, 1); 10853a5a1b3Sopenharmony_ci dev->module_index = mod->index; 10953a5a1b3Sopenharmony_ci dev->id = id; 11053a5a1b3Sopenharmony_ci 11153a5a1b3Sopenharmony_ci PA_LLIST_INIT(ca_device, dev); 11253a5a1b3Sopenharmony_ci PA_LLIST_PREPEND(ca_device, u->devices, dev); 11353a5a1b3Sopenharmony_ci 11453a5a1b3Sopenharmony_ci return 0; 11553a5a1b3Sopenharmony_ci} 11653a5a1b3Sopenharmony_ci 11753a5a1b3Sopenharmony_cistatic int ca_update_device_list(struct pa_module *m) { 11853a5a1b3Sopenharmony_ci AudioObjectPropertyAddress property_address; 11953a5a1b3Sopenharmony_ci OSStatus err; 12053a5a1b3Sopenharmony_ci UInt32 i, size, num_devices; 12153a5a1b3Sopenharmony_ci AudioDeviceID *device_id; 12253a5a1b3Sopenharmony_ci struct ca_device *dev; 12353a5a1b3Sopenharmony_ci struct userdata *u; 12453a5a1b3Sopenharmony_ci 12553a5a1b3Sopenharmony_ci pa_assert(m); 12653a5a1b3Sopenharmony_ci pa_assert_se(u = m->userdata); 12753a5a1b3Sopenharmony_ci 12853a5a1b3Sopenharmony_ci property_address.mSelector = kAudioHardwarePropertyDevices; 12953a5a1b3Sopenharmony_ci property_address.mScope = kAudioObjectPropertyScopeGlobal; 13053a5a1b3Sopenharmony_ci property_address.mElement = kAudioObjectPropertyElementMaster; 13153a5a1b3Sopenharmony_ci 13253a5a1b3Sopenharmony_ci /* get the number of currently available audio devices */ 13353a5a1b3Sopenharmony_ci err = AudioObjectGetPropertyDataSize(kAudioObjectSystemObject, &property_address, 0, NULL, &size); 13453a5a1b3Sopenharmony_ci if (err) { 13553a5a1b3Sopenharmony_ci pa_log("Unable to get data size for kAudioHardwarePropertyDevices."); 13653a5a1b3Sopenharmony_ci return -1; 13753a5a1b3Sopenharmony_ci } 13853a5a1b3Sopenharmony_ci 13953a5a1b3Sopenharmony_ci num_devices = size / sizeof(AudioDeviceID); 14053a5a1b3Sopenharmony_ci device_id = pa_xnew(AudioDeviceID, num_devices); 14153a5a1b3Sopenharmony_ci 14253a5a1b3Sopenharmony_ci err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &property_address, 0, NULL, &size, device_id); 14353a5a1b3Sopenharmony_ci if (err) { 14453a5a1b3Sopenharmony_ci pa_log("Unable to get kAudioHardwarePropertyDevices."); 14553a5a1b3Sopenharmony_ci pa_xfree(device_id); 14653a5a1b3Sopenharmony_ci return -1; 14753a5a1b3Sopenharmony_ci } 14853a5a1b3Sopenharmony_ci 14953a5a1b3Sopenharmony_ci /* scan for devices which are reported but not in our cached list */ 15053a5a1b3Sopenharmony_ci for (i = 0; i < num_devices; i++) { 15153a5a1b3Sopenharmony_ci bool found = false; 15253a5a1b3Sopenharmony_ci 15353a5a1b3Sopenharmony_ci PA_LLIST_FOREACH(dev, u->devices) 15453a5a1b3Sopenharmony_ci if (dev->id == device_id[i]) { 15553a5a1b3Sopenharmony_ci found = true; 15653a5a1b3Sopenharmony_ci break; 15753a5a1b3Sopenharmony_ci } 15853a5a1b3Sopenharmony_ci 15953a5a1b3Sopenharmony_ci if (!found) 16053a5a1b3Sopenharmony_ci ca_device_added(m, device_id[i]); 16153a5a1b3Sopenharmony_ci } 16253a5a1b3Sopenharmony_ci 16353a5a1b3Sopenharmony_ci /* scan for devices which are in our cached list but are not reported */ 16453a5a1b3Sopenharmony_ciscan_removed: 16553a5a1b3Sopenharmony_ci 16653a5a1b3Sopenharmony_ci PA_LLIST_FOREACH(dev, u->devices) { 16753a5a1b3Sopenharmony_ci bool found = false; 16853a5a1b3Sopenharmony_ci 16953a5a1b3Sopenharmony_ci for (i = 0; i < num_devices; i++) 17053a5a1b3Sopenharmony_ci if (dev->id == device_id[i]) { 17153a5a1b3Sopenharmony_ci found = true; 17253a5a1b3Sopenharmony_ci break; 17353a5a1b3Sopenharmony_ci } 17453a5a1b3Sopenharmony_ci 17553a5a1b3Sopenharmony_ci if (!found) { 17653a5a1b3Sopenharmony_ci pa_log_debug("object id %d has been removed (module index %d) %p", (unsigned int) dev->id, dev->module_index, dev); 17753a5a1b3Sopenharmony_ci pa_module_unload_request_by_index(m->core, dev->module_index, true); 17853a5a1b3Sopenharmony_ci PA_LLIST_REMOVE(ca_device, u->devices, dev); 17953a5a1b3Sopenharmony_ci pa_xfree(dev); 18053a5a1b3Sopenharmony_ci /* the current list item pointer is not valid anymore, so start over. */ 18153a5a1b3Sopenharmony_ci goto scan_removed; 18253a5a1b3Sopenharmony_ci } 18353a5a1b3Sopenharmony_ci } 18453a5a1b3Sopenharmony_ci 18553a5a1b3Sopenharmony_ci pa_xfree(device_id); 18653a5a1b3Sopenharmony_ci return 0; 18753a5a1b3Sopenharmony_ci} 18853a5a1b3Sopenharmony_ci 18953a5a1b3Sopenharmony_cistatic OSStatus property_listener_proc(AudioObjectID objectID, UInt32 numberAddresses, 19053a5a1b3Sopenharmony_ci const AudioObjectPropertyAddress inAddresses[], 19153a5a1b3Sopenharmony_ci void *clientData) { 19253a5a1b3Sopenharmony_ci struct userdata *u = clientData; 19353a5a1b3Sopenharmony_ci char dummy = 1; 19453a5a1b3Sopenharmony_ci 19553a5a1b3Sopenharmony_ci pa_assert(u); 19653a5a1b3Sopenharmony_ci 19753a5a1b3Sopenharmony_ci /* dispatch module load/unload operations in main thread */ 19853a5a1b3Sopenharmony_ci write(u->detect_fds[1], &dummy, 1); 19953a5a1b3Sopenharmony_ci 20053a5a1b3Sopenharmony_ci return 0; 20153a5a1b3Sopenharmony_ci} 20253a5a1b3Sopenharmony_ci 20353a5a1b3Sopenharmony_cistatic void detect_handle(pa_mainloop_api *a, pa_io_event *e, int fd, pa_io_event_flags_t events, void *userdata) { 20453a5a1b3Sopenharmony_ci pa_module *m = userdata; 20553a5a1b3Sopenharmony_ci char dummy; 20653a5a1b3Sopenharmony_ci 20753a5a1b3Sopenharmony_ci pa_assert(m); 20853a5a1b3Sopenharmony_ci 20953a5a1b3Sopenharmony_ci read(fd, &dummy, 1); 21053a5a1b3Sopenharmony_ci ca_update_device_list(m); 21153a5a1b3Sopenharmony_ci} 21253a5a1b3Sopenharmony_ci 21353a5a1b3Sopenharmony_ciint pa__init(pa_module *m) { 21453a5a1b3Sopenharmony_ci struct userdata *u = pa_xnew0(struct userdata, 1); 21553a5a1b3Sopenharmony_ci AudioObjectPropertyAddress property_address; 21653a5a1b3Sopenharmony_ci pa_modargs *ma; 21753a5a1b3Sopenharmony_ci 21853a5a1b3Sopenharmony_ci pa_assert(m); 21953a5a1b3Sopenharmony_ci pa_assert(m->core); 22053a5a1b3Sopenharmony_ci 22153a5a1b3Sopenharmony_ci m->userdata = u; 22253a5a1b3Sopenharmony_ci 22353a5a1b3Sopenharmony_ci if (!(ma = pa_modargs_new(m->argument, valid_modargs))) { 22453a5a1b3Sopenharmony_ci pa_log("Failed to parse module arguments."); 22553a5a1b3Sopenharmony_ci goto fail; 22653a5a1b3Sopenharmony_ci } 22753a5a1b3Sopenharmony_ci 22853a5a1b3Sopenharmony_ci /* 22953a5a1b3Sopenharmony_ci * Set default value to true if not given as a modarg. 23053a5a1b3Sopenharmony_ci * In such a case, pa_modargs_get_value_boolean() will not touch the 23153a5a1b3Sopenharmony_ci * buffer. 23253a5a1b3Sopenharmony_ci */ 23353a5a1b3Sopenharmony_ci u->playback = u->record = true; 23453a5a1b3Sopenharmony_ci 23553a5a1b3Sopenharmony_ci if (pa_modargs_get_value_boolean(ma, "record", &u->record) < 0 || pa_modargs_get_value_boolean(ma, "playback", &u->playback) < 0) { 23653a5a1b3Sopenharmony_ci pa_log("record= and playback= expect boolean argument."); 23753a5a1b3Sopenharmony_ci goto fail; 23853a5a1b3Sopenharmony_ci } 23953a5a1b3Sopenharmony_ci 24053a5a1b3Sopenharmony_ci if (!u->playback && !u->record) { 24153a5a1b3Sopenharmony_ci pa_log("neither playback nor record enabled for device."); 24253a5a1b3Sopenharmony_ci goto fail; 24353a5a1b3Sopenharmony_ci } 24453a5a1b3Sopenharmony_ci 24553a5a1b3Sopenharmony_ci pa_modargs_get_value_u32(ma, "ioproc_frames", &u->ioproc_frames); 24653a5a1b3Sopenharmony_ci 24753a5a1b3Sopenharmony_ci property_address.mSelector = kAudioHardwarePropertyDevices; 24853a5a1b3Sopenharmony_ci property_address.mScope = kAudioObjectPropertyScopeGlobal; 24953a5a1b3Sopenharmony_ci property_address.mElement = kAudioObjectPropertyElementMaster; 25053a5a1b3Sopenharmony_ci 25153a5a1b3Sopenharmony_ci if (AudioObjectAddPropertyListener(kAudioObjectSystemObject, &property_address, property_listener_proc, u)) { 25253a5a1b3Sopenharmony_ci pa_log("AudioObjectAddPropertyListener() failed."); 25353a5a1b3Sopenharmony_ci goto fail; 25453a5a1b3Sopenharmony_ci } 25553a5a1b3Sopenharmony_ci 25653a5a1b3Sopenharmony_ci if (ca_update_device_list(m)) 25753a5a1b3Sopenharmony_ci goto fail; 25853a5a1b3Sopenharmony_ci 25953a5a1b3Sopenharmony_ci pa_assert_se(pipe(u->detect_fds) == 0); 26053a5a1b3Sopenharmony_ci pa_assert_se(u->detect_io = m->core->mainloop->io_new(m->core->mainloop, u->detect_fds[0], PA_IO_EVENT_INPUT, detect_handle, m)); 26153a5a1b3Sopenharmony_ci 26253a5a1b3Sopenharmony_ci return 0; 26353a5a1b3Sopenharmony_ci 26453a5a1b3Sopenharmony_cifail: 26553a5a1b3Sopenharmony_ci pa_xfree(u); 26653a5a1b3Sopenharmony_ci return -1; 26753a5a1b3Sopenharmony_ci} 26853a5a1b3Sopenharmony_ci 26953a5a1b3Sopenharmony_civoid pa__done(pa_module *m) { 27053a5a1b3Sopenharmony_ci struct userdata *u; 27153a5a1b3Sopenharmony_ci struct ca_device *dev; 27253a5a1b3Sopenharmony_ci AudioObjectPropertyAddress property_address; 27353a5a1b3Sopenharmony_ci 27453a5a1b3Sopenharmony_ci pa_assert(m); 27553a5a1b3Sopenharmony_ci pa_assert_se(u = m->userdata); 27653a5a1b3Sopenharmony_ci 27753a5a1b3Sopenharmony_ci dev = u->devices; 27853a5a1b3Sopenharmony_ci 27953a5a1b3Sopenharmony_ci property_address.mSelector = kAudioHardwarePropertyDevices; 28053a5a1b3Sopenharmony_ci property_address.mScope = kAudioObjectPropertyScopeGlobal; 28153a5a1b3Sopenharmony_ci property_address.mElement = kAudioObjectPropertyElementMaster; 28253a5a1b3Sopenharmony_ci 28353a5a1b3Sopenharmony_ci AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &property_address, property_listener_proc, u); 28453a5a1b3Sopenharmony_ci 28553a5a1b3Sopenharmony_ci while (dev) { 28653a5a1b3Sopenharmony_ci struct ca_device *next = dev->next; 28753a5a1b3Sopenharmony_ci 28853a5a1b3Sopenharmony_ci pa_module_unload_request_by_index(m->core, dev->module_index, true); 28953a5a1b3Sopenharmony_ci pa_xfree(dev); 29053a5a1b3Sopenharmony_ci 29153a5a1b3Sopenharmony_ci dev = next; 29253a5a1b3Sopenharmony_ci } 29353a5a1b3Sopenharmony_ci 29453a5a1b3Sopenharmony_ci if (u->detect_fds[0] >= 0) 29553a5a1b3Sopenharmony_ci close(u->detect_fds[0]); 29653a5a1b3Sopenharmony_ci 29753a5a1b3Sopenharmony_ci if (u->detect_fds[1] >= 0) 29853a5a1b3Sopenharmony_ci close(u->detect_fds[1]); 29953a5a1b3Sopenharmony_ci 30053a5a1b3Sopenharmony_ci if (u->detect_io) 30153a5a1b3Sopenharmony_ci m->core->mainloop->io_free(u->detect_io); 30253a5a1b3Sopenharmony_ci 30353a5a1b3Sopenharmony_ci pa_xfree(u); 30453a5a1b3Sopenharmony_ci} 305