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