153a5a1b3Sopenharmony_ci/***
253a5a1b3Sopenharmony_ci  This file is part of PulseAudio.
353a5a1b3Sopenharmony_ci
453a5a1b3Sopenharmony_ci  Copyright 2008-2013 João Paulo Rechi Vita
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
853a5a1b3Sopenharmony_ci  published by the Free Software Foundation; either version 2.1 of the
953a5a1b3Sopenharmony_ci  License, 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
1753a5a1b3Sopenharmony_ci  License 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 <pulsecore/core.h>
2553a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h>
2653a5a1b3Sopenharmony_ci#include <pulsecore/macro.h>
2753a5a1b3Sopenharmony_ci#include <pulsecore/module.h>
2853a5a1b3Sopenharmony_ci#include <pulsecore/modargs.h>
2953a5a1b3Sopenharmony_ci#include <pulsecore/shared.h>
3053a5a1b3Sopenharmony_ci
3153a5a1b3Sopenharmony_ci#include "bluez5-util.h"
3253a5a1b3Sopenharmony_ci
3353a5a1b3Sopenharmony_ciPA_MODULE_AUTHOR("João Paulo Rechi Vita");
3453a5a1b3Sopenharmony_ciPA_MODULE_DESCRIPTION("Detect available BlueZ 5 Bluetooth audio devices and load BlueZ 5 Bluetooth audio drivers");
3553a5a1b3Sopenharmony_ciPA_MODULE_VERSION(PACKAGE_VERSION);
3653a5a1b3Sopenharmony_ciPA_MODULE_LOAD_ONCE(true);
3753a5a1b3Sopenharmony_ciPA_MODULE_USAGE(
3853a5a1b3Sopenharmony_ci    "headset=ofono|native|auto"
3953a5a1b3Sopenharmony_ci    "autodetect_mtu=<boolean>"
4053a5a1b3Sopenharmony_ci    "enable_msbc=<boolean, enable mSBC support in native and oFono backends, default is true>"
4153a5a1b3Sopenharmony_ci    "output_rate_refresh_interval_ms=<interval between attempts to improve output rate in milliseconds>"
4253a5a1b3Sopenharmony_ci    "enable_native_hsp_hs=<boolean, enable HSP support in native backend>"
4353a5a1b3Sopenharmony_ci    "enable_native_hfp_hf=<boolean, enable HFP support in native backend>"
4453a5a1b3Sopenharmony_ci    "avrcp_absolute_volume=<synchronize volume with peer, true by default>"
4553a5a1b3Sopenharmony_ci);
4653a5a1b3Sopenharmony_ci
4753a5a1b3Sopenharmony_cistatic const char* const valid_modargs[] = {
4853a5a1b3Sopenharmony_ci    "headset",
4953a5a1b3Sopenharmony_ci    "autodetect_mtu",
5053a5a1b3Sopenharmony_ci    "enable_msbc",
5153a5a1b3Sopenharmony_ci    "output_rate_refresh_interval_ms",
5253a5a1b3Sopenharmony_ci    "enable_native_hsp_hs",
5353a5a1b3Sopenharmony_ci    "enable_native_hfp_hf",
5453a5a1b3Sopenharmony_ci    "avrcp_absolute_volume",
5553a5a1b3Sopenharmony_ci    NULL
5653a5a1b3Sopenharmony_ci};
5753a5a1b3Sopenharmony_ci
5853a5a1b3Sopenharmony_cistruct userdata {
5953a5a1b3Sopenharmony_ci    pa_module *module;
6053a5a1b3Sopenharmony_ci    pa_core *core;
6153a5a1b3Sopenharmony_ci    pa_hashmap *loaded_device_paths;
6253a5a1b3Sopenharmony_ci    pa_hook_slot *device_connection_changed_slot;
6353a5a1b3Sopenharmony_ci    pa_bluetooth_discovery *discovery;
6453a5a1b3Sopenharmony_ci    bool autodetect_mtu;
6553a5a1b3Sopenharmony_ci    bool avrcp_absolute_volume;
6653a5a1b3Sopenharmony_ci    uint32_t output_rate_refresh_interval_ms;
6753a5a1b3Sopenharmony_ci};
6853a5a1b3Sopenharmony_ci
6953a5a1b3Sopenharmony_cistatic pa_hook_result_t device_connection_changed_cb(pa_bluetooth_discovery *y, const pa_bluetooth_device *d, struct userdata *u) {
7053a5a1b3Sopenharmony_ci    bool module_loaded;
7153a5a1b3Sopenharmony_ci
7253a5a1b3Sopenharmony_ci    pa_assert(d);
7353a5a1b3Sopenharmony_ci    pa_assert(u);
7453a5a1b3Sopenharmony_ci
7553a5a1b3Sopenharmony_ci    module_loaded = pa_hashmap_get(u->loaded_device_paths, d->path) ? true : false;
7653a5a1b3Sopenharmony_ci
7753a5a1b3Sopenharmony_ci    /* When changing A2DP codec there is no transport connected, ensure that no module is unloaded */
7853a5a1b3Sopenharmony_ci    if (module_loaded && !pa_bluetooth_device_any_transport_connected(d) &&
7953a5a1b3Sopenharmony_ci            !d->codec_switching_in_progress) {
8053a5a1b3Sopenharmony_ci        /* disconnection, the module unloads itself */
8153a5a1b3Sopenharmony_ci        pa_log_debug("Unregistering module for %s", d->path);
8253a5a1b3Sopenharmony_ci        pa_hashmap_remove(u->loaded_device_paths, d->path);
8353a5a1b3Sopenharmony_ci        return PA_HOOK_OK;
8453a5a1b3Sopenharmony_ci    }
8553a5a1b3Sopenharmony_ci
8653a5a1b3Sopenharmony_ci    if (!module_loaded && pa_bluetooth_device_any_transport_connected(d)) {
8753a5a1b3Sopenharmony_ci        /* a new device has been connected */
8853a5a1b3Sopenharmony_ci        pa_module *m;
8953a5a1b3Sopenharmony_ci        char *args = pa_sprintf_malloc("path=%s autodetect_mtu=%i output_rate_refresh_interval_ms=%u"
9053a5a1b3Sopenharmony_ci                                       " avrcp_absolute_volume=%i",
9153a5a1b3Sopenharmony_ci                                       d->path,
9253a5a1b3Sopenharmony_ci                                       (int)u->autodetect_mtu,
9353a5a1b3Sopenharmony_ci                                       u->output_rate_refresh_interval_ms,
9453a5a1b3Sopenharmony_ci                                       (int)u->avrcp_absolute_volume);
9553a5a1b3Sopenharmony_ci
9653a5a1b3Sopenharmony_ci        pa_log_debug("Loading module-bluez5-device %s", args);
9753a5a1b3Sopenharmony_ci        pa_module_load(&m, u->module->core, "module-bluez5-device", args);
9853a5a1b3Sopenharmony_ci        pa_xfree(args);
9953a5a1b3Sopenharmony_ci
10053a5a1b3Sopenharmony_ci        if (m)
10153a5a1b3Sopenharmony_ci            /* No need to duplicate the path here since the device object will
10253a5a1b3Sopenharmony_ci             * exist for the whole hashmap entry lifespan */
10353a5a1b3Sopenharmony_ci            pa_hashmap_put(u->loaded_device_paths, d->path, d->path);
10453a5a1b3Sopenharmony_ci        else
10553a5a1b3Sopenharmony_ci            pa_log_warn("Failed to load module for device %s", d->path);
10653a5a1b3Sopenharmony_ci
10753a5a1b3Sopenharmony_ci        return PA_HOOK_OK;
10853a5a1b3Sopenharmony_ci    }
10953a5a1b3Sopenharmony_ci
11053a5a1b3Sopenharmony_ci    return PA_HOOK_OK;
11153a5a1b3Sopenharmony_ci}
11253a5a1b3Sopenharmony_ci
11353a5a1b3Sopenharmony_ci#ifdef HAVE_BLUEZ_5_NATIVE_HEADSET
11453a5a1b3Sopenharmony_ciconst char *default_headset_backend = "native";
11553a5a1b3Sopenharmony_ci#else
11653a5a1b3Sopenharmony_ciconst char *default_headset_backend = "ofono";
11753a5a1b3Sopenharmony_ci#endif
11853a5a1b3Sopenharmony_ci
11953a5a1b3Sopenharmony_ciint pa__init(pa_module *m) {
12053a5a1b3Sopenharmony_ci    struct userdata *u;
12153a5a1b3Sopenharmony_ci    pa_modargs *ma;
12253a5a1b3Sopenharmony_ci    const char *headset_str;
12353a5a1b3Sopenharmony_ci    int headset_backend;
12453a5a1b3Sopenharmony_ci    bool autodetect_mtu;
12553a5a1b3Sopenharmony_ci    bool enable_msbc;
12653a5a1b3Sopenharmony_ci    bool avrcp_absolute_volume;
12753a5a1b3Sopenharmony_ci    uint32_t output_rate_refresh_interval_ms;
12853a5a1b3Sopenharmony_ci    bool enable_native_hsp_hs;
12953a5a1b3Sopenharmony_ci    bool enable_native_hfp_hf;
13053a5a1b3Sopenharmony_ci
13153a5a1b3Sopenharmony_ci    pa_assert(m);
13253a5a1b3Sopenharmony_ci
13353a5a1b3Sopenharmony_ci    if (!(ma = pa_modargs_new(m->argument, valid_modargs))) {
13453a5a1b3Sopenharmony_ci        pa_log("failed to parse module arguments.");
13553a5a1b3Sopenharmony_ci        goto fail;
13653a5a1b3Sopenharmony_ci    }
13753a5a1b3Sopenharmony_ci
13853a5a1b3Sopenharmony_ci    pa_assert_se(headset_str = pa_modargs_get_value(ma, "headset", default_headset_backend));
13953a5a1b3Sopenharmony_ci    if (pa_streq(headset_str, "ofono"))
14053a5a1b3Sopenharmony_ci        headset_backend = HEADSET_BACKEND_OFONO;
14153a5a1b3Sopenharmony_ci    else if (pa_streq(headset_str, "native"))
14253a5a1b3Sopenharmony_ci        headset_backend = HEADSET_BACKEND_NATIVE;
14353a5a1b3Sopenharmony_ci    else if (pa_streq(headset_str, "auto"))
14453a5a1b3Sopenharmony_ci        headset_backend = HEADSET_BACKEND_AUTO;
14553a5a1b3Sopenharmony_ci    else {
14653a5a1b3Sopenharmony_ci        pa_log("headset parameter must be either ofono, native or auto (found %s)", headset_str);
14753a5a1b3Sopenharmony_ci        goto fail;
14853a5a1b3Sopenharmony_ci    }
14953a5a1b3Sopenharmony_ci
15053a5a1b3Sopenharmony_ci    /* default value if no module parameter */
15153a5a1b3Sopenharmony_ci    enable_native_hfp_hf = (headset_backend == HEADSET_BACKEND_NATIVE);
15253a5a1b3Sopenharmony_ci
15353a5a1b3Sopenharmony_ci    autodetect_mtu = false;
15453a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_boolean(ma, "autodetect_mtu", &autodetect_mtu) < 0) {
15553a5a1b3Sopenharmony_ci        pa_log("Invalid boolean value for autodetect_mtu parameter");
15653a5a1b3Sopenharmony_ci    }
15753a5a1b3Sopenharmony_ci    enable_msbc = true;
15853a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_boolean(ma, "enable_msbc", &enable_msbc) < 0) {
15953a5a1b3Sopenharmony_ci        pa_log("Invalid boolean value for enable_msbc parameter");
16053a5a1b3Sopenharmony_ci    }
16153a5a1b3Sopenharmony_ci    enable_native_hfp_hf = true;
16253a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_boolean(ma, "enable_native_hfp_hf", &enable_native_hfp_hf) < 0) {
16353a5a1b3Sopenharmony_ci        pa_log("enable_native_hfp_hf must be true or false");
16453a5a1b3Sopenharmony_ci        goto fail;
16553a5a1b3Sopenharmony_ci    }
16653a5a1b3Sopenharmony_ci    enable_native_hsp_hs = !enable_native_hfp_hf;
16753a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_boolean(ma, "enable_native_hsp_hs", &enable_native_hsp_hs) < 0) {
16853a5a1b3Sopenharmony_ci        pa_log("enable_native_hsp_hs must be true or false");
16953a5a1b3Sopenharmony_ci        goto fail;
17053a5a1b3Sopenharmony_ci    }
17153a5a1b3Sopenharmony_ci
17253a5a1b3Sopenharmony_ci    avrcp_absolute_volume = true;
17353a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_boolean(ma, "avrcp_absolute_volume", &avrcp_absolute_volume) < 0) {
17453a5a1b3Sopenharmony_ci        pa_log("avrcp_absolute_volume must be true or false");
17553a5a1b3Sopenharmony_ci        goto fail;
17653a5a1b3Sopenharmony_ci    }
17753a5a1b3Sopenharmony_ci
17853a5a1b3Sopenharmony_ci    output_rate_refresh_interval_ms = DEFAULT_OUTPUT_RATE_REFRESH_INTERVAL_MS;
17953a5a1b3Sopenharmony_ci    if (pa_modargs_get_value_u32(ma, "output_rate_refresh_interval_ms", &output_rate_refresh_interval_ms) < 0) {
18053a5a1b3Sopenharmony_ci        pa_log("Invalid value for output_rate_refresh_interval parameter.");
18153a5a1b3Sopenharmony_ci        goto fail;
18253a5a1b3Sopenharmony_ci    }
18353a5a1b3Sopenharmony_ci
18453a5a1b3Sopenharmony_ci    m->userdata = u = pa_xnew0(struct userdata, 1);
18553a5a1b3Sopenharmony_ci    u->module = m;
18653a5a1b3Sopenharmony_ci    u->core = m->core;
18753a5a1b3Sopenharmony_ci    u->autodetect_mtu = autodetect_mtu;
18853a5a1b3Sopenharmony_ci    u->avrcp_absolute_volume = avrcp_absolute_volume;
18953a5a1b3Sopenharmony_ci    u->output_rate_refresh_interval_ms = output_rate_refresh_interval_ms;
19053a5a1b3Sopenharmony_ci    u->loaded_device_paths = pa_hashmap_new(pa_idxset_string_hash_func, pa_idxset_string_compare_func);
19153a5a1b3Sopenharmony_ci
19253a5a1b3Sopenharmony_ci    if (!(u->discovery = pa_bluetooth_discovery_get(u->core, headset_backend, enable_native_hsp_hs, enable_native_hfp_hf, enable_msbc)))
19353a5a1b3Sopenharmony_ci        goto fail;
19453a5a1b3Sopenharmony_ci
19553a5a1b3Sopenharmony_ci    u->device_connection_changed_slot =
19653a5a1b3Sopenharmony_ci        pa_hook_connect(pa_bluetooth_discovery_hook(u->discovery, PA_BLUETOOTH_HOOK_DEVICE_CONNECTION_CHANGED),
19753a5a1b3Sopenharmony_ci                        PA_HOOK_NORMAL, (pa_hook_cb_t) device_connection_changed_cb, u);
19853a5a1b3Sopenharmony_ci
19953a5a1b3Sopenharmony_ci    pa_modargs_free(ma);
20053a5a1b3Sopenharmony_ci    return 0;
20153a5a1b3Sopenharmony_ci
20253a5a1b3Sopenharmony_cifail:
20353a5a1b3Sopenharmony_ci    if (ma)
20453a5a1b3Sopenharmony_ci        pa_modargs_free(ma);
20553a5a1b3Sopenharmony_ci    pa__done(m);
20653a5a1b3Sopenharmony_ci    return -1;
20753a5a1b3Sopenharmony_ci}
20853a5a1b3Sopenharmony_ci
20953a5a1b3Sopenharmony_civoid pa__done(pa_module *m) {
21053a5a1b3Sopenharmony_ci    struct userdata *u;
21153a5a1b3Sopenharmony_ci
21253a5a1b3Sopenharmony_ci    pa_assert(m);
21353a5a1b3Sopenharmony_ci
21453a5a1b3Sopenharmony_ci    if (!(u = m->userdata))
21553a5a1b3Sopenharmony_ci        return;
21653a5a1b3Sopenharmony_ci
21753a5a1b3Sopenharmony_ci    if (u->device_connection_changed_slot)
21853a5a1b3Sopenharmony_ci        pa_hook_slot_free(u->device_connection_changed_slot);
21953a5a1b3Sopenharmony_ci
22053a5a1b3Sopenharmony_ci    if (u->loaded_device_paths)
22153a5a1b3Sopenharmony_ci        pa_hashmap_free(u->loaded_device_paths);
22253a5a1b3Sopenharmony_ci
22353a5a1b3Sopenharmony_ci    if (u->discovery)
22453a5a1b3Sopenharmony_ci        pa_bluetooth_discovery_unref(u->discovery);
22553a5a1b3Sopenharmony_ci
22653a5a1b3Sopenharmony_ci    pa_xfree(u);
22753a5a1b3Sopenharmony_ci}
228