/*** This file is part of PulseAudio. Copyright 2008-2013 João Paulo Rechi Vita Copyright 2011-2013 BMW Car IT GmbH. Copyright 2018-2019 Pali Rohár PulseAudio is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. PulseAudio is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with PulseAudio; if not, see . ***/ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "a2dp-codecs.h" #include "a2dp-codec-util.h" #include "bluez5-util.h" PA_MODULE_AUTHOR("João Paulo Rechi Vita"); PA_MODULE_DESCRIPTION("BlueZ 5 Bluetooth audio sink and source"); PA_MODULE_VERSION(PACKAGE_VERSION); PA_MODULE_LOAD_ONCE(false); PA_MODULE_USAGE("path=" "autodetect_mtu="); #define FIXED_LATENCY_PLAYBACK_A2DP (25 * PA_USEC_PER_MSEC) #define FIXED_LATENCY_PLAYBACK_SCO (25 * PA_USEC_PER_MSEC) #define FIXED_LATENCY_RECORD_A2DP (25 * PA_USEC_PER_MSEC) #define FIXED_LATENCY_RECORD_SCO (25 * PA_USEC_PER_MSEC) #define HSP_MAX_GAIN 15 static const char* const valid_modargs[] = { "path", "autodetect_mtu", NULL }; enum { BLUETOOTH_MESSAGE_IO_THREAD_FAILED, BLUETOOTH_MESSAGE_STREAM_FD_HUP, BLUETOOTH_MESSAGE_SET_TRANSPORT_PLAYING, BLUETOOTH_MESSAGE_MAX }; enum { PA_SOURCE_MESSAGE_SETUP_STREAM = PA_SOURCE_MESSAGE_MAX, }; enum { PA_SINK_MESSAGE_SETUP_STREAM = PA_SINK_MESSAGE_MAX, }; typedef struct bluetooth_msg { pa_msgobject parent; pa_card *card; } bluetooth_msg; PA_DEFINE_PRIVATE_CLASS(bluetooth_msg, pa_msgobject); #define BLUETOOTH_MSG(o) (bluetooth_msg_cast(o)) struct userdata { pa_module *module; pa_core *core; pa_hook_slot *device_connection_changed_slot; pa_hook_slot *transport_state_changed_slot; pa_hook_slot *transport_speaker_gain_changed_slot; pa_hook_slot *transport_microphone_gain_changed_slot; pa_bluetooth_discovery *discovery; pa_bluetooth_device *device; pa_bluetooth_transport *transport; bool transport_acquired; bool stream_setup_done; pa_card *card; pa_sink *sink; pa_source *source; pa_bluetooth_profile_t profile; char *output_port_name; char *input_port_name; pa_thread *thread; pa_thread_mq thread_mq; pa_rtpoll *rtpoll; pa_rtpoll_item *rtpoll_item; bluetooth_msg *msg; int stream_fd; int stream_write_type; size_t read_link_mtu; size_t write_link_mtu; size_t read_block_size; size_t write_block_size; uint64_t read_index; uint64_t write_index; pa_usec_t started_at; pa_smoother *read_smoother; pa_memchunk write_memchunk; const pa_a2dp_codec *a2dp_codec; void *encoder_info; pa_sample_spec encoder_sample_spec; void *encoder_buffer; /* Codec transfer buffer */ size_t encoder_buffer_size; /* Size of the buffer */ void *decoder_info; pa_sample_spec decoder_sample_spec; void *decoder_buffer; /* Codec transfer buffer */ size_t decoder_buffer_size; /* Size of the buffer */ }; typedef enum pa_bluetooth_form_factor { PA_BLUETOOTH_FORM_FACTOR_UNKNOWN, PA_BLUETOOTH_FORM_FACTOR_HEADSET, PA_BLUETOOTH_FORM_FACTOR_HANDSFREE, PA_BLUETOOTH_FORM_FACTOR_MICROPHONE, PA_BLUETOOTH_FORM_FACTOR_SPEAKER, PA_BLUETOOTH_FORM_FACTOR_HEADPHONE, PA_BLUETOOTH_FORM_FACTOR_PORTABLE, PA_BLUETOOTH_FORM_FACTOR_CAR, PA_BLUETOOTH_FORM_FACTOR_HIFI, PA_BLUETOOTH_FORM_FACTOR_PHONE, } pa_bluetooth_form_factor_t; /* Run from main thread */ static pa_bluetooth_form_factor_t form_factor_from_class(uint32_t class_of_device) { unsigned major, minor; pa_bluetooth_form_factor_t r; static const pa_bluetooth_form_factor_t table[] = { [1] = PA_BLUETOOTH_FORM_FACTOR_HEADSET, [2] = PA_BLUETOOTH_FORM_FACTOR_HANDSFREE, [4] = PA_BLUETOOTH_FORM_FACTOR_MICROPHONE, [5] = PA_BLUETOOTH_FORM_FACTOR_SPEAKER, [6] = PA_BLUETOOTH_FORM_FACTOR_HEADPHONE, [7] = PA_BLUETOOTH_FORM_FACTOR_PORTABLE, [8] = PA_BLUETOOTH_FORM_FACTOR_CAR, [10] = PA_BLUETOOTH_FORM_FACTOR_HIFI }; /* * See Bluetooth Assigned Numbers: * https://www.bluetooth.org/Technical/AssignedNumbers/baseband.htm */ major = (class_of_device >> 8) & 0x1F; minor = (class_of_device >> 2) & 0x3F; switch (major) { case 2: return PA_BLUETOOTH_FORM_FACTOR_PHONE; case 4: break; default: pa_log_debug("Unknown Bluetooth major device class %u", major); return PA_BLUETOOTH_FORM_FACTOR_UNKNOWN; } r = minor < PA_ELEMENTSOF(table) ? table[minor] : PA_BLUETOOTH_FORM_FACTOR_UNKNOWN; if (!r) pa_log_debug("Unknown Bluetooth minor device class %u", minor); return r; } /* Run from main thread */ static const char *form_factor_to_string(pa_bluetooth_form_factor_t ff) { switch (ff) { case PA_BLUETOOTH_FORM_FACTOR_UNKNOWN: return "unknown"; case PA_BLUETOOTH_FORM_FACTOR_HEADSET: return "headset"; case PA_BLUETOOTH_FORM_FACTOR_HANDSFREE: return "hands-free"; case PA_BLUETOOTH_FORM_FACTOR_MICROPHONE: return "microphone"; case PA_BLUETOOTH_FORM_FACTOR_SPEAKER: return "speaker"; case PA_BLUETOOTH_FORM_FACTOR_HEADPHONE: return "headphone"; case PA_BLUETOOTH_FORM_FACTOR_PORTABLE: return "portable"; case PA_BLUETOOTH_FORM_FACTOR_CAR: return "car"; case PA_BLUETOOTH_FORM_FACTOR_HIFI: return "hifi"; case PA_BLUETOOTH_FORM_FACTOR_PHONE: return "phone"; } pa_assert_not_reached(); } /* Run from main thread */ static void connect_ports(struct userdata *u, void *new_data, pa_direction_t direction) { pa_device_port *port; if (direction == PA_DIRECTION_OUTPUT) { pa_sink_new_data *sink_new_data = new_data; pa_assert_se(port = pa_hashmap_get(u->card->ports, u->output_port_name)); pa_assert_se(pa_hashmap_put(sink_new_data->ports, port->name, port) >= 0); pa_device_port_ref(port); } else { pa_source_new_data *source_new_data = new_data; pa_assert_se(port = pa_hashmap_get(u->card->ports, u->input_port_name)); pa_assert_se(pa_hashmap_put(source_new_data->ports, port->name, port) >= 0); pa_device_port_ref(port); } } /* Run from IO thread */ static int sco_process_render(struct userdata *u) { ssize_t l; pa_memchunk memchunk; int saved_errno; pa_assert(u); pa_assert(u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT || u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY); pa_assert(u->sink); pa_sink_render_full(u->sink, u->write_block_size, &memchunk); pa_assert(memchunk.length == u->write_block_size); for (;;) { const void *p; /* Now write that data to the socket. The socket is of type * SEQPACKET, and we generated the data of the MTU size, so this * should just work. */ p = (const uint8_t *) pa_memblock_acquire_chunk(&memchunk); l = pa_write(u->stream_fd, p, memchunk.length, &u->stream_write_type); pa_memblock_release(memchunk.memblock); pa_assert(l != 0); if (l > 0) break; saved_errno = errno; if (saved_errno == EINTR) /* Retry right away if we got interrupted */ continue; pa_memblock_unref(memchunk.memblock); if (saved_errno == EAGAIN) { /* Hmm, apparently the socket was not writable, give up for now. * Because the data was already rendered, let's discard the block. */ pa_log_debug("Got EAGAIN on write() after POLLOUT, probably there is a temporary connection loss."); return 1; } pa_log_error("Failed to write data to SCO socket: %s", pa_cstrerror(saved_errno)); return -1; } pa_assert((size_t) l <= memchunk.length); if ((size_t) l != memchunk.length) { pa_log_error("Wrote memory block to socket only partially! %llu written, wanted to write %llu.", (unsigned long long) l, (unsigned long long) memchunk.length); pa_memblock_unref(memchunk.memblock); return -1; } u->write_index += (uint64_t) memchunk.length; pa_memblock_unref(memchunk.memblock); return 1; } /* Run from IO thread */ static int sco_process_push(struct userdata *u) { ssize_t l; pa_memchunk memchunk; struct cmsghdr *cm; struct msghdr m; bool found_tstamp = false; pa_usec_t tstamp = 0; pa_assert(u); pa_assert(u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT || u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY); pa_assert(u->source); pa_assert(u->read_smoother); memchunk.memblock = pa_memblock_new(u->core->mempool, u->read_block_size); memchunk.index = memchunk.length = 0; for (;;) { void *p; uint8_t aux[1024]; struct iovec iov; pa_zero(m); pa_zero(aux); pa_zero(iov); m.msg_iov = &iov; m.msg_iovlen = 1; m.msg_control = aux; m.msg_controllen = sizeof(aux); p = pa_memblock_acquire(memchunk.memblock); iov.iov_base = p; iov.iov_len = pa_memblock_get_length(memchunk.memblock); l = recvmsg(u->stream_fd, &m, 0); pa_memblock_release(memchunk.memblock); if (l > 0) break; if (l < 0 && errno == EINTR) /* Retry right away if we got interrupted */ continue; pa_memblock_unref(memchunk.memblock); if (l < 0 && errno == EAGAIN) /* Hmm, apparently the socket was not readable, give up for now. */ return 0; pa_log_error("Failed to read data from SCO socket: %s", l < 0 ? pa_cstrerror(errno) : "EOF"); return -1; } pa_assert((size_t) l <= pa_memblock_get_length(memchunk.memblock)); /* In some rare occasions, we might receive packets of a very strange * size. This could potentially be possible if the SCO packet was * received partially over-the-air, or more probably due to hardware * issues in our Bluetooth adapter. In these cases, in order to avoid * an assertion failure due to unaligned data, just discard the whole * packet */ if (!pa_frame_aligned(l, &u->decoder_sample_spec)) { pa_log_warn("SCO packet received of unaligned size: %zu", l); pa_memblock_unref(memchunk.memblock); return -1; } memchunk.length = (size_t) l; u->read_index += (uint64_t) l; for (cm = CMSG_FIRSTHDR(&m); cm; cm = CMSG_NXTHDR(&m, cm)) if (cm->cmsg_level == SOL_SOCKET && cm->cmsg_type == SO_TIMESTAMP) { struct timeval *tv = (struct timeval*) CMSG_DATA(cm); pa_rtclock_from_wallclock(tv); tstamp = pa_timeval_load(tv); found_tstamp = true; break; } if (!found_tstamp) { PA_ONCE_BEGIN { pa_log_warn("Couldn't find SO_TIMESTAMP data in auxiliary recvmsg() data!"); } PA_ONCE_END; tstamp = pa_rtclock_now(); } pa_smoother_put(u->read_smoother, tstamp, pa_bytes_to_usec(u->read_index, &u->decoder_sample_spec)); pa_smoother_resume(u->read_smoother, tstamp, true); pa_source_post(u->source, &memchunk); pa_memblock_unref(memchunk.memblock); return l; } /* Run from IO thread */ static void a2dp_prepare_encoder_buffer(struct userdata *u) { pa_assert(u); if (u->encoder_buffer_size < u->write_link_mtu) { pa_xfree(u->encoder_buffer); u->encoder_buffer = pa_xmalloc(u->write_link_mtu); } /* Encoder buffer cannot be larger then link MTU, otherwise * encode method would produce larger packets then link MTU */ u->encoder_buffer_size = u->write_link_mtu; } /* Run from IO thread */ static void a2dp_prepare_decoder_buffer(struct userdata *u) { pa_assert(u); if (u->decoder_buffer_size < u->read_link_mtu) { pa_xfree(u->decoder_buffer); u->decoder_buffer = pa_xmalloc(u->read_link_mtu); } /* Decoder buffer cannot be larger then link MTU, otherwise * decode method would produce larger output then read_block_size */ u->decoder_buffer_size = u->read_link_mtu; } /* Run from IO thread */ static int a2dp_write_buffer(struct userdata *u, size_t nbytes) { int ret = 0; /* Encoder function of A2DP codec may provide empty buffer, in this case do * not post any empty buffer via A2DP socket. It may be because of codec * internal state, e.g. encoder is waiting for more samples so it can * provide encoded data. */ if (PA_UNLIKELY(!nbytes)) { u->write_index += (uint64_t) u->write_memchunk.length; pa_memblock_unref(u->write_memchunk.memblock); pa_memchunk_reset(&u->write_memchunk); return 0; } for (;;) { ssize_t l; l = pa_write(u->stream_fd, u->encoder_buffer, nbytes, &u->stream_write_type); pa_assert(l != 0); if (l < 0) { if (errno == EINTR) /* Retry right away if we got interrupted */ continue; else if (errno == EAGAIN) { /* Hmm, apparently the socket was not writable, give up for now */ pa_log_debug("Got EAGAIN on write() after POLLOUT, probably there is a temporary connection loss."); break; } pa_log_error("Failed to write data to socket: %s", pa_cstrerror(errno)); ret = -1; break; } pa_assert((size_t) l <= nbytes); if ((size_t) l != nbytes) { pa_log_warn("Wrote memory block to socket only partially! %llu written, wanted to write %llu.", (unsigned long long) l, (unsigned long long) nbytes); ret = -1; break; } u->write_index += (uint64_t) u->write_memchunk.length; pa_memblock_unref(u->write_memchunk.memblock); pa_memchunk_reset(&u->write_memchunk); ret = 1; break; } return ret; } /* Run from IO thread */ static int a2dp_process_render(struct userdata *u) { const uint8_t *ptr; size_t processed; size_t length; pa_assert(u); pa_assert(u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK); pa_assert(u->sink); pa_assert(u->a2dp_codec); /* First, render some data */ if (!u->write_memchunk.memblock) pa_sink_render_full(u->sink, u->write_block_size, &u->write_memchunk); pa_assert(u->write_memchunk.length == u->write_block_size); a2dp_prepare_encoder_buffer(u); /* Try to create a packet of the full MTU */ ptr = (const uint8_t *) pa_memblock_acquire_chunk(&u->write_memchunk); length = u->a2dp_codec->encode_buffer(u->encoder_info, u->write_index / pa_frame_size(&u->encoder_sample_spec), ptr, u->write_memchunk.length, u->encoder_buffer, u->encoder_buffer_size, &processed); pa_memblock_release(u->write_memchunk.memblock); if (processed != u->write_memchunk.length) { pa_log_error("Encoding error"); return -1; } return a2dp_write_buffer(u, length); } /* Run from IO thread */ static int a2dp_process_push(struct userdata *u) { int ret = 0; pa_memchunk memchunk; pa_assert(u); pa_assert(u->profile == PA_BLUETOOTH_PROFILE_A2DP_SOURCE); pa_assert(u->source); pa_assert(u->read_smoother); pa_assert(u->a2dp_codec); memchunk.memblock = pa_memblock_new(u->core->mempool, u->read_block_size); memchunk.index = memchunk.length = 0; a2dp_prepare_decoder_buffer(u); for (;;) { uint8_t aux[1024]; struct iovec iov; struct cmsghdr *cm; struct msghdr m; bool found_tstamp = false; pa_usec_t tstamp; uint8_t *ptr; ssize_t l; size_t processed; pa_zero(m); pa_zero(aux); pa_zero(iov); m.msg_iov = &iov; m.msg_iovlen = 1; m.msg_control = aux; m.msg_controllen = sizeof(aux); iov.iov_base = u->decoder_buffer; iov.iov_len = u->decoder_buffer_size; l = recvmsg(u->stream_fd, &m, 0); if (l <= 0) { if (l < 0 && errno == EINTR) /* Retry right away if we got interrupted */ continue; else if (l < 0 && errno == EAGAIN) /* Hmm, apparently the socket was not readable, give up for now. */ break; pa_log_error("Failed to read data from socket: %s", l < 0 ? pa_cstrerror(errno) : "EOF"); ret = -1; break; } pa_assert((size_t) l <= u->decoder_buffer_size); /* TODO: get timestamp from rtp */ for (cm = CMSG_FIRSTHDR(&m); cm; cm = CMSG_NXTHDR(&m, cm)) { if (cm->cmsg_level == SOL_SOCKET && cm->cmsg_type == SO_TIMESTAMP) { struct timeval *tv = (struct timeval*) CMSG_DATA(cm); pa_rtclock_from_wallclock(tv); tstamp = pa_timeval_load(tv); found_tstamp = true; break; } } if (!found_tstamp) { PA_ONCE_BEGIN { pa_log_warn("Couldn't find SO_TIMESTAMP data in auxiliary recvmsg() data!"); } PA_ONCE_END; tstamp = pa_rtclock_now(); } ptr = pa_memblock_acquire(memchunk.memblock); memchunk.length = pa_memblock_get_length(memchunk.memblock); memchunk.length = u->a2dp_codec->decode_buffer(u->decoder_info, u->decoder_buffer, l, ptr, memchunk.length, &processed); pa_memblock_release(memchunk.memblock); if (processed != (size_t) l) { pa_log_error("Decoding error"); ret = -1; break; } u->read_index += (uint64_t) memchunk.length; pa_smoother_put(u->read_smoother, tstamp, pa_bytes_to_usec(u->read_index, &u->decoder_sample_spec)); pa_smoother_resume(u->read_smoother, tstamp, true); /* Decoding of A2DP codec data may result in empty buffer, in this case * do not post empty audio samples. It may happen due to algorithmic * delay of audio codec. */ if (PA_LIKELY(memchunk.length)) pa_source_post(u->source, &memchunk); ret = l; break; } pa_memblock_unref(memchunk.memblock); return ret; } static void update_sink_buffer_size(struct userdata *u) { int old_bufsize; socklen_t len = sizeof(int); int ret; ret = getsockopt(u->stream_fd, SOL_SOCKET, SO_SNDBUF, &old_bufsize, &len); if (ret == -1) { pa_log_warn("Changing bluetooth buffer size: Failed to getsockopt(SO_SNDBUF): %s", pa_cstrerror(errno)); } else { int new_bufsize; /* Set send buffer size as small as possible. The minimum value is 1024 according to the * socket man page. The data is written to the socket in chunks of write_block_size, so * there should at least be room for two chunks in the buffer. Generally, write_block_size * is larger than 512. If not, use the next multiple of write_block_size which is larger * than 1024. */ new_bufsize = 2 * u->write_block_size; if (new_bufsize < 1024) new_bufsize = (1024 / u->write_block_size + 1) * u->write_block_size; /* The kernel internally doubles the buffer size that was set by setsockopt and getsockopt * returns the doubled value. */ if (new_bufsize != old_bufsize / 2) { ret = setsockopt(u->stream_fd, SOL_SOCKET, SO_SNDBUF, &new_bufsize, len); if (ret == -1) pa_log_warn("Changing bluetooth buffer size: Failed to change from %d to %d: %s", old_bufsize / 2, new_bufsize, pa_cstrerror(errno)); else pa_log_info("Changing bluetooth buffer size: Changed from %d to %d", old_bufsize / 2, new_bufsize); } } } static void teardown_stream(struct userdata *u) { if (u->rtpoll_item) { pa_rtpoll_item_free(u->rtpoll_item); u->rtpoll_item = NULL; } if (u->stream_fd >= 0) { pa_close(u->stream_fd); u->stream_fd = -1; } if (u->read_smoother) { pa_smoother_free(u->read_smoother); u->read_smoother = NULL; } if (u->write_memchunk.memblock) { pa_memblock_unref(u->write_memchunk.memblock); pa_memchunk_reset(&u->write_memchunk); } pa_log_debug("Audio stream torn down"); u->stream_setup_done = false; } static int transport_acquire(struct userdata *u, bool optional) { pa_assert(u->transport); if (u->transport_acquired) return 0; pa_log_debug("Acquiring transport %s", u->transport->path); u->stream_fd = u->transport->acquire(u->transport, optional, &u->read_link_mtu, &u->write_link_mtu); if (u->stream_fd < 0) return u->stream_fd; /* transport_acquired must be set before calling * pa_bluetooth_transport_set_state() */ u->transport_acquired = true; pa_log_info("Transport %s acquired: fd %d", u->transport->path, u->stream_fd); if (u->transport->state == PA_BLUETOOTH_TRANSPORT_STATE_IDLE) { if (pa_thread_mq_get() != NULL) pa_asyncmsgq_post(pa_thread_mq_get()->outq, PA_MSGOBJECT(u->msg), BLUETOOTH_MESSAGE_SET_TRANSPORT_PLAYING, NULL, 0, NULL, NULL); else pa_bluetooth_transport_set_state(u->transport, PA_BLUETOOTH_TRANSPORT_STATE_PLAYING); } return 0; } static void transport_release(struct userdata *u) { pa_assert(u->transport); /* Ignore if already released */ if (!u->transport_acquired) return; pa_log_debug("Releasing transport %s", u->transport->path); u->transport->release(u->transport); u->transport_acquired = false; teardown_stream(u); /* Set transport state to idle if this was not already done by the remote end closing * the file descriptor. Only do this when called from the I/O thread */ if (pa_thread_mq_get() != NULL && u->transport->state == PA_BLUETOOTH_TRANSPORT_STATE_PLAYING) pa_asyncmsgq_post(pa_thread_mq_get()->outq, PA_MSGOBJECT(u->msg), BLUETOOTH_MESSAGE_STREAM_FD_HUP, NULL, 0, NULL, NULL); } /* Run from I/O thread */ static void handle_sink_block_size_change(struct userdata *u) { pa_sink_set_max_request_within_thread(u->sink, u->write_block_size); pa_sink_set_fixed_latency_within_thread(u->sink, (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK ? FIXED_LATENCY_PLAYBACK_A2DP : FIXED_LATENCY_PLAYBACK_SCO) + pa_bytes_to_usec(u->write_block_size, &u->encoder_sample_spec)); /* If there is still data in the memchunk, we have to discard it * because the write_block_size may have changed. */ if (u->write_memchunk.memblock) { pa_memblock_unref(u->write_memchunk.memblock); pa_memchunk_reset(&u->write_memchunk); } update_sink_buffer_size(u); } /* Run from I/O thread */ static void transport_config_mtu(struct userdata *u) { if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT || u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) { u->read_block_size = u->read_link_mtu; u->write_block_size = u->write_link_mtu; if (!pa_frame_aligned(u->read_block_size, &u->source->sample_spec)) { pa_log_debug("Got invalid read MTU: %lu, rounding down", u->read_block_size); u->read_block_size = pa_frame_align(u->read_block_size, &u->source->sample_spec); } if (!pa_frame_aligned(u->write_block_size, &u->sink->sample_spec)) { pa_log_debug("Got invalid write MTU: %lu, rounding down", u->write_block_size); u->write_block_size = pa_frame_align(u->write_block_size, &u->sink->sample_spec); } } else { pa_assert(u->a2dp_codec); if (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK) { u->write_block_size = u->a2dp_codec->get_write_block_size(u->encoder_info, u->write_link_mtu); } else { u->read_block_size = u->a2dp_codec->get_read_block_size(u->decoder_info, u->read_link_mtu); } } if (u->sink) handle_sink_block_size_change(u); if (u->source) pa_source_set_fixed_latency_within_thread(u->source, (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SOURCE ? FIXED_LATENCY_RECORD_A2DP : FIXED_LATENCY_RECORD_SCO) + pa_bytes_to_usec(u->read_block_size, &u->decoder_sample_spec)); } /* Run from I/O thread */ static int setup_stream(struct userdata *u) { struct pollfd *pollfd; int one; pa_assert(u->stream_fd >= 0); /* return if stream is already set up */ if (u->stream_setup_done) return 0; pa_log_info("Transport %s resuming", u->transport->path); if (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK) { pa_assert(u->a2dp_codec); if (u->a2dp_codec->reset(u->encoder_info) < 0) return -1; } else if (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SOURCE) { pa_assert(u->a2dp_codec); if (u->a2dp_codec->reset(u->decoder_info) < 0) return -1; } transport_config_mtu(u); pa_make_fd_nonblock(u->stream_fd); pa_make_socket_low_delay(u->stream_fd); one = 1; if (setsockopt(u->stream_fd, SOL_SOCKET, SO_TIMESTAMP, &one, sizeof(one)) < 0) pa_log_warn("Failed to enable SO_TIMESTAMP: %s", pa_cstrerror(errno)); pa_log_debug("Stream properly set up, we're ready to roll!"); u->rtpoll_item = pa_rtpoll_item_new(u->rtpoll, PA_RTPOLL_NEVER, 1); pollfd = pa_rtpoll_item_get_pollfd(u->rtpoll_item, NULL); pollfd->fd = u->stream_fd; pollfd->events = pollfd->revents = 0; u->read_index = u->write_index = 0; u->started_at = 0; u->stream_setup_done = true; if (u->source) u->read_smoother = pa_smoother_new(PA_USEC_PER_SEC, 2*PA_USEC_PER_SEC, true, true, 10, pa_rtclock_now(), true); return 0; } /* Called from I/O thread, returns true if the transport was acquired or * a connection was requested successfully. */ static bool setup_transport_and_stream(struct userdata *u) { int transport_error; transport_error = transport_acquire(u, false); if (transport_error < 0) { if (transport_error != -EAGAIN) return false; } else { if (setup_stream(u) < 0) return false; } return true; } /* Run from IO thread */ static int source_process_msg(pa_msgobject *o, int code, void *data, int64_t offset, pa_memchunk *chunk) { struct userdata *u = PA_SOURCE(o)->userdata; pa_assert(u->source == PA_SOURCE(o)); pa_assert(u->transport); switch (code) { case PA_SOURCE_MESSAGE_GET_LATENCY: { int64_t wi, ri; if (u->read_smoother) { wi = pa_smoother_get(u->read_smoother, pa_rtclock_now()); ri = pa_bytes_to_usec(u->read_index, &u->decoder_sample_spec); *((int64_t*) data) = u->source->thread_info.fixed_latency + wi - ri; } else *((int64_t*) data) = 0; return 0; } case PA_SOURCE_MESSAGE_SETUP_STREAM: /* Skip stream setup if stream_fd has been invalidated. This can occur if the stream has already been set up and then immediately received POLLHUP. If the stream has already been set up earlier, then this setup_stream() call is redundant anyway, but currently the code is such that this kind of unnecessary setup_stream() calls can happen. */ if (u->stream_fd < 0) pa_log_debug("Skip source stream setup while closing"); else setup_stream(u); return 0; } return pa_source_process_msg(o, code, data, offset, chunk); } /* Called from the IO thread. */ static int source_set_state_in_io_thread_cb(pa_source *s, pa_source_state_t new_state, pa_suspend_cause_t new_suspend_cause) { struct userdata *u; pa_assert(s); pa_assert_se(u = s->userdata); switch (new_state) { case PA_SOURCE_SUSPENDED: /* Ignore if transition is PA_SOURCE_INIT->PA_SOURCE_SUSPENDED */ if (!PA_SOURCE_IS_OPENED(s->thread_info.state)) break; /* Stop the device if the sink is suspended as well */ if (!u->sink || u->sink->state == PA_SINK_SUSPENDED) transport_release(u); if (u->read_smoother) pa_smoother_pause(u->read_smoother, pa_rtclock_now()); break; case PA_SOURCE_IDLE: case PA_SOURCE_RUNNING: if (s->thread_info.state != PA_SOURCE_SUSPENDED) break; /* Resume the device if the sink was suspended as well */ if (!u->sink || !PA_SINK_IS_OPENED(u->sink->thread_info.state)) if (!setup_transport_and_stream(u)) return -1; /* We don't resume the smoother here. Instead we * wait until the first packet arrives */ break; case PA_SOURCE_UNLINKED: case PA_SOURCE_INIT: case PA_SOURCE_INVALID_STATE: break; } return 0; } /* Run from main thread */ static void source_set_volume_cb(pa_source *s) { uint16_t gain; pa_volume_t volume; struct userdata *u; pa_assert(s); pa_assert(s->core); u = s->userdata; pa_assert(u); pa_assert(u->source == s); if (u->transport->set_microphone_gain == NULL) return; gain = (pa_cvolume_max(&s->real_volume) * HSP_MAX_GAIN) / PA_VOLUME_NORM; if (gain > HSP_MAX_GAIN) gain = HSP_MAX_GAIN; volume = (pa_volume_t) (gain * PA_VOLUME_NORM / HSP_MAX_GAIN); /* increment volume by one to correct rounding errors */ if (volume < PA_VOLUME_NORM) volume++; pa_cvolume_set(&s->real_volume, u->decoder_sample_spec.channels, volume); /* Set soft volume when in headset role */ if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) pa_cvolume_set(&s->soft_volume, u->decoder_sample_spec.channels, volume); /* If we are in the AG role, we send a command to the head set to change * the microphone gain. In the HS role, source and sink are swapped, so * in this case we notify the AG that the speaker gain has changed */ u->transport->set_microphone_gain(u->transport, gain); } /* Run from main thread */ static int add_source(struct userdata *u) { pa_source_new_data data; pa_assert(u->transport); pa_source_new_data_init(&data); data.module = u->module; data.card = u->card; data.driver = __FILE__; data.name = pa_sprintf_malloc("bluez_source.%s.%s", u->device->address, pa_bluetooth_profile_to_string(u->profile)); data.namereg_fail = false; pa_proplist_sets(data.proplist, "bluetooth.protocol", pa_bluetooth_profile_to_string(u->profile)); pa_source_new_data_set_sample_spec(&data, &u->decoder_sample_spec); if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT) pa_proplist_sets(data.proplist, PA_PROP_DEVICE_INTENDED_ROLES, "phone"); connect_ports(u, &data, PA_DIRECTION_INPUT); if (!u->transport_acquired) switch (u->profile) { case PA_BLUETOOTH_PROFILE_A2DP_SOURCE: case PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY: data.suspend_cause = PA_SUSPEND_USER; break; case PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT: /* u->stream_fd contains the error returned by the last transport_acquire() * EAGAIN means we are waiting for a NewConnection signal */ if (u->stream_fd == -EAGAIN) data.suspend_cause = PA_SUSPEND_USER; else pa_assert_not_reached(); break; case PA_BLUETOOTH_PROFILE_A2DP_SINK: case PA_BLUETOOTH_PROFILE_OFF: pa_assert_not_reached(); break; } u->source = pa_source_new(u->core, &data, PA_SOURCE_HARDWARE|PA_SOURCE_LATENCY); pa_source_new_data_done(&data); if (!u->source) { pa_log_error("Failed to create source"); return -1; } u->source->userdata = u; u->source->parent.process_msg = source_process_msg; u->source->set_state_in_io_thread = source_set_state_in_io_thread_cb; if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT || u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) { pa_source_set_set_volume_callback(u->source, source_set_volume_cb); u->source->n_volume_steps = 16; } return 0; } /* Run from IO thread */ static int sink_process_msg(pa_msgobject *o, int code, void *data, int64_t offset, pa_memchunk *chunk) { struct userdata *u = PA_SINK(o)->userdata; pa_assert(u->sink == PA_SINK(o)); pa_assert(u->transport); switch (code) { case PA_SINK_MESSAGE_GET_LATENCY: { int64_t wi = 0, ri = 0; if (u->read_smoother) { ri = pa_smoother_get(u->read_smoother, pa_rtclock_now()); wi = pa_bytes_to_usec(u->write_index + u->write_block_size, &u->encoder_sample_spec); } else if (u->started_at) { ri = pa_rtclock_now() - u->started_at; wi = pa_bytes_to_usec(u->write_index, &u->encoder_sample_spec); } *((int64_t*) data) = u->sink->thread_info.fixed_latency + wi - ri; return 0; } case PA_SINK_MESSAGE_SETUP_STREAM: /* Skip stream setup if stream_fd has been invalidated. This can occur if the stream has already been set up and then immediately received POLLHUP. If the stream has already been set up earlier, then this setup_stream() call is redundant anyway, but currently the code is such that this kind of unnecessary setup_stream() calls can happen. */ if (u->stream_fd < 0) pa_log_debug("Skip sink stream setup while closing"); else setup_stream(u); return 0; } return pa_sink_process_msg(o, code, data, offset, chunk); } /* Called from the IO thread. */ static int sink_set_state_in_io_thread_cb(pa_sink *s, pa_sink_state_t new_state, pa_suspend_cause_t new_suspend_cause) { struct userdata *u; pa_assert(s); pa_assert_se(u = s->userdata); switch (new_state) { case PA_SINK_SUSPENDED: /* Ignore if transition is PA_SINK_INIT->PA_SINK_SUSPENDED */ if (!PA_SINK_IS_OPENED(s->thread_info.state)) break; /* Stop the device if the source is suspended as well */ if (!u->source || u->source->state == PA_SOURCE_SUSPENDED) /* We deliberately ignore whether stopping * actually worked. Since the stream_fd is * closed it doesn't really matter */ transport_release(u); break; case PA_SINK_IDLE: case PA_SINK_RUNNING: if (s->thread_info.state != PA_SINK_SUSPENDED) break; /* Resume the device if the source was suspended as well */ if (!u->source || !PA_SOURCE_IS_OPENED(u->source->thread_info.state)) if (!setup_transport_and_stream(u)) return -1; break; case PA_SINK_UNLINKED: case PA_SINK_INIT: case PA_SINK_INVALID_STATE: break; } return 0; } /* Run from main thread */ static void sink_set_volume_cb(pa_sink *s) { uint16_t gain; pa_volume_t volume; struct userdata *u; pa_assert(s); pa_assert(s->core); u = s->userdata; pa_assert(u); pa_assert(u->sink == s); if (u->transport->set_speaker_gain == NULL) return; gain = (pa_cvolume_max(&s->real_volume) * HSP_MAX_GAIN) / PA_VOLUME_NORM; if (gain > HSP_MAX_GAIN) gain = HSP_MAX_GAIN; volume = (pa_volume_t) (gain * PA_VOLUME_NORM / HSP_MAX_GAIN); /* increment volume by one to correct rounding errors */ if (volume < PA_VOLUME_NORM) volume++; pa_cvolume_set(&s->real_volume, u->encoder_sample_spec.channels, volume); /* Set soft volume when in headset role */ if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) pa_cvolume_set(&s->soft_volume, u->encoder_sample_spec.channels, volume); /* If we are in the AG role, we send a command to the head set to change * the speaker gain. In the HS role, source and sink are swapped, so * in this case we notify the AG that the microphone gain has changed */ u->transport->set_speaker_gain(u->transport, gain); } /* Run from main thread */ static int add_sink(struct userdata *u) { pa_sink_new_data data; pa_assert(u->transport); pa_sink_new_data_init(&data); data.module = u->module; data.card = u->card; data.driver = __FILE__; data.name = pa_sprintf_malloc("bluez_sink.%s.%s", u->device->address, pa_bluetooth_profile_to_string(u->profile)); data.namereg_fail = false; pa_proplist_sets(data.proplist, "bluetooth.protocol", pa_bluetooth_profile_to_string(u->profile)); pa_sink_new_data_set_sample_spec(&data, &u->encoder_sample_spec); if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT) pa_proplist_sets(data.proplist, PA_PROP_DEVICE_INTENDED_ROLES, "phone"); connect_ports(u, &data, PA_DIRECTION_OUTPUT); if (!u->transport_acquired) switch (u->profile) { case PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY: data.suspend_cause = PA_SUSPEND_USER; break; case PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT: /* u->stream_fd contains the error returned by the last transport_acquire() * EAGAIN means we are waiting for a NewConnection signal */ if (u->stream_fd == -EAGAIN) data.suspend_cause = PA_SUSPEND_USER; else pa_assert_not_reached(); break; case PA_BLUETOOTH_PROFILE_A2DP_SINK: /* Profile switch should have failed */ case PA_BLUETOOTH_PROFILE_A2DP_SOURCE: case PA_BLUETOOTH_PROFILE_OFF: pa_assert_not_reached(); break; } u->sink = pa_sink_new(u->core, &data, PA_SINK_HARDWARE|PA_SINK_LATENCY); pa_sink_new_data_done(&data); if (!u->sink) { pa_log_error("Failed to create sink"); return -1; } u->sink->userdata = u; u->sink->parent.process_msg = sink_process_msg; u->sink->set_state_in_io_thread = sink_set_state_in_io_thread_cb; if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT || u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) { pa_sink_set_set_volume_callback(u->sink, sink_set_volume_cb); u->sink->n_volume_steps = 16; } return 0; } /* Run from main thread */ static int transport_config(struct userdata *u) { if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT || u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) { u->encoder_sample_spec.format = PA_SAMPLE_S16LE; u->encoder_sample_spec.channels = 1; u->encoder_sample_spec.rate = 8000; u->decoder_sample_spec.format = PA_SAMPLE_S16LE; u->decoder_sample_spec.channels = 1; u->decoder_sample_spec.rate = 8000; return 0; } else { bool is_a2dp_sink = u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK; void *info; pa_assert(u->transport); pa_assert(!u->a2dp_codec); pa_assert(!u->encoder_info); pa_assert(!u->decoder_info); u->a2dp_codec = u->transport->a2dp_codec; pa_assert(u->a2dp_codec); info = u->a2dp_codec->init(is_a2dp_sink, false, u->transport->config, u->transport->config_size, is_a2dp_sink ? &u->encoder_sample_spec : &u->decoder_sample_spec); if (is_a2dp_sink) u->encoder_info = info; else u->decoder_info = info; if (!info) return -1; return 0; } } /* Run from main thread */ static int setup_transport(struct userdata *u) { pa_bluetooth_transport *t; pa_assert(u); pa_assert(!u->transport); pa_assert(u->profile != PA_BLUETOOTH_PROFILE_OFF); /* check if profile has a transport */ t = u->device->transports[u->profile]; if (!t || t->state <= PA_BLUETOOTH_TRANSPORT_STATE_DISCONNECTED) { pa_log_warn("Profile %s has no transport", pa_bluetooth_profile_to_string(u->profile)); return -1; } u->transport = t; if (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SOURCE || u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) transport_acquire(u, true); /* In case of error, the sink/sources will be created suspended */ else { int transport_error; transport_error = transport_acquire(u, false); if (transport_error < 0 && transport_error != -EAGAIN) return -1; /* We need to fail here until the interactions with module-suspend-on-idle and alike get improved */ } return transport_config(u); } /* Run from main thread */ static pa_direction_t get_profile_direction(pa_bluetooth_profile_t p) { static const pa_direction_t profile_direction[] = { [PA_BLUETOOTH_PROFILE_A2DP_SINK] = PA_DIRECTION_OUTPUT, [PA_BLUETOOTH_PROFILE_A2DP_SOURCE] = PA_DIRECTION_INPUT, [PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT] = PA_DIRECTION_INPUT | PA_DIRECTION_OUTPUT, [PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY] = PA_DIRECTION_INPUT | PA_DIRECTION_OUTPUT, [PA_BLUETOOTH_PROFILE_OFF] = 0 }; return profile_direction[p]; } /* Run from main thread */ static int init_profile(struct userdata *u) { int r = 0; pa_assert(u); pa_assert(u->profile != PA_BLUETOOTH_PROFILE_OFF); if (setup_transport(u) < 0) return -1; pa_assert(u->transport); if (get_profile_direction (u->profile) & PA_DIRECTION_OUTPUT) if (add_sink(u) < 0) r = -1; if (get_profile_direction (u->profile) & PA_DIRECTION_INPUT) if (add_source(u) < 0) r = -1; return r; } static int write_block(struct userdata *u) { int n_written; if (u->write_index <= 0) u->started_at = pa_rtclock_now(); if (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK) { if ((n_written = a2dp_process_render(u)) < 0) return -1; } else { if ((n_written = sco_process_render(u)) < 0) return -1; } return n_written; } /* I/O thread function */ static void thread_func(void *userdata) { struct userdata *u = userdata; unsigned blocks_to_write = 0; unsigned bytes_to_write = 0; pa_assert(u); pa_assert(u->transport); pa_log_debug("IO Thread starting up"); if (u->core->realtime_scheduling) pa_thread_make_realtime(u->core->realtime_priority); pa_thread_mq_install(&u->thread_mq); /* Setup the stream only if the transport was already acquired */ if (u->transport_acquired) setup_stream(u); for (;;) { struct pollfd *pollfd; int ret; bool disable_timer = true; bool writable = false; bool have_source = u->source ? PA_SOURCE_IS_LINKED(u->source->thread_info.state) : false; bool have_sink = u->sink ? PA_SINK_IS_LINKED(u->sink->thread_info.state) : false; pollfd = u->rtpoll_item ? pa_rtpoll_item_get_pollfd(u->rtpoll_item, NULL) : NULL; /* Check for stream error or close */ if (pollfd && (pollfd->revents & ~(POLLOUT|POLLIN))) { pa_log_info("FD error: %s%s%s%s", pollfd->revents & POLLERR ? "POLLERR " :"", pollfd->revents & POLLHUP ? "POLLHUP " :"", pollfd->revents & POLLPRI ? "POLLPRI " :"", pollfd->revents & POLLNVAL ? "POLLNVAL " :""); if (pollfd->revents & POLLHUP) { pollfd = NULL; teardown_stream(u); blocks_to_write = 0; bytes_to_write = 0; pa_asyncmsgq_post(pa_thread_mq_get()->outq, PA_MSGOBJECT(u->msg), BLUETOOTH_MESSAGE_STREAM_FD_HUP, NULL, 0, NULL, NULL); } else goto fail; } /* If there is a pollfd, the stream is set up and we need to do something */ if (pollfd) { /* Handle source if present */ if (have_source) { /* We should send two blocks to the device before we expect a response. */ if (have_sink && u->write_index == 0 && u->read_index <= 0) blocks_to_write = 2; /* If we got woken up by POLLIN let's do some reading */ if (pollfd->revents & POLLIN) { int n_read; if (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SOURCE) n_read = a2dp_process_push(u); else n_read = sco_process_push(u); if (n_read < 0) goto fail; if (have_sink && n_read > 0) { /* We just read something, so we are supposed to write something, too */ bytes_to_write += n_read; blocks_to_write += bytes_to_write / u->write_block_size; bytes_to_write = bytes_to_write % u->write_block_size; } } } /* Handle sink if present */ if (have_sink) { /* Process rewinds */ if (PA_UNLIKELY(u->sink->thread_info.rewind_requested)) pa_sink_process_rewind(u->sink, 0); /* Test if the stream is writable */ if (pollfd->revents & POLLOUT) writable = true; /* If we have a source, we let the source determine the timing * for the sink */ if (have_source) { if (writable && blocks_to_write > 0) { int result; if ((result = write_block(u)) < 0) goto fail; blocks_to_write -= result; /* writable controls whether we set POLLOUT when polling - we set it to * false to enable POLLOUT. If there are more blocks to write, we want to * be woken up immediately when the socket becomes writable. If there * aren't currently any more blocks to write, then we'll have to wait * until we've received more data, so in that case we only want to set * POLLIN. Note that when we are woken up the next time, POLLOUT won't be * set in revents even if the socket has meanwhile become writable, which * may seem bad, but in that case we'll set POLLOUT in the subsequent * poll, and the poll will return immediately, so our writes won't be * delayed. */ if (blocks_to_write > 0) writable = false; } /* There is no source, we have to use the system clock for timing */ } else { bool have_written = false; pa_usec_t time_passed = 0; pa_usec_t audio_sent = 0; if (u->started_at) { time_passed = pa_rtclock_now() - u->started_at; audio_sent = pa_bytes_to_usec(u->write_index, &u->encoder_sample_spec); } /* A new block needs to be sent. */ if (audio_sent <= time_passed) { size_t bytes_to_send = pa_usec_to_bytes(time_passed - audio_sent, &u->encoder_sample_spec); /* There are more than two blocks that need to be written. It seems that * the socket has not been accepting data fast enough (could be due to * hiccups in the wireless transmission). We need to discard everything * older than two block sizes to keep the latency from growing. */ if (bytes_to_send > 2 * u->write_block_size) { uint64_t skip_bytes; pa_memchunk tmp; size_t mempool_max_block_size = pa_mempool_block_size_max(u->core->mempool); pa_usec_t skip_usec; skip_bytes = bytes_to_send - 2 * u->write_block_size; skip_usec = pa_bytes_to_usec(skip_bytes, &u->encoder_sample_spec); pa_log_debug("Skipping %llu us (= %llu bytes) in audio stream", (unsigned long long) skip_usec, (unsigned long long) skip_bytes); while (skip_bytes > 0) { size_t bytes_to_render; if (skip_bytes > mempool_max_block_size) bytes_to_render = mempool_max_block_size; else bytes_to_render = skip_bytes; pa_sink_render_full(u->sink, bytes_to_render, &tmp); pa_memblock_unref(tmp.memblock); u->write_index += bytes_to_render; skip_bytes -= bytes_to_render; } if (u->write_index > 0 && u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK) { size_t new_write_block_size = u->a2dp_codec->reduce_encoder_bitrate(u->encoder_info, u->write_link_mtu); if (new_write_block_size) { u->write_block_size = new_write_block_size; handle_sink_block_size_change(u); } } } blocks_to_write = 1; } /* If the stream is writable, send some data if necessary */ if (writable && blocks_to_write > 0) { int result; if ((result = write_block(u)) < 0) goto fail; blocks_to_write -= result; writable = false; if (result) have_written = true; } /* If nothing was written during this iteration, either the stream * is not writable or there was no write pending. Set up a timer that * will wake up the thread when the next data needs to be written. */ if (!have_written) { pa_usec_t sleep_for; pa_usec_t next_write_at; if (writable) { /* There was no write pending on this iteration of the loop. * Let's estimate when we need to wake up next */ next_write_at = pa_bytes_to_usec(u->write_index, &u->encoder_sample_spec); sleep_for = time_passed < next_write_at ? next_write_at - time_passed : 0; /* pa_log("Sleeping for %lu; time passed %lu, next write at %lu", (unsigned long) sleep_for, (unsigned long) time_passed, (unsigned long)next_write_at); */ } else /* We could not write because the stream was not ready. Let's try * again in 500 ms and drop audio if we still can't write. The * thread will also be woken up when we can write again. */ sleep_for = PA_USEC_PER_MSEC * 500; pa_rtpoll_set_timer_relative(u->rtpoll, sleep_for); disable_timer = false; } } } /* Set events to wake up the thread */ pollfd->events = (short) (((have_sink && !writable) ? POLLOUT : 0) | (have_source ? POLLIN : 0)); } if (disable_timer) pa_rtpoll_set_timer_disabled(u->rtpoll); if ((ret = pa_rtpoll_run(u->rtpoll)) < 0) { pa_log_debug("pa_rtpoll_run failed with: %d", ret); goto fail; } if (ret == 0) { pa_log_debug("IO thread shutdown requested, stopping cleanly"); transport_release(u); goto finish; } } fail: /* If this was no regular exit from the loop we have to continue processing messages until we receive PA_MESSAGE_SHUTDOWN */ pa_log_debug("IO thread failed"); pa_asyncmsgq_post(pa_thread_mq_get()->outq, PA_MSGOBJECT(u->msg), BLUETOOTH_MESSAGE_IO_THREAD_FAILED, NULL, 0, NULL, NULL); pa_asyncmsgq_wait_for(u->thread_mq.inq, PA_MESSAGE_SHUTDOWN); finish: pa_log_debug("IO thread shutting down"); } /* Run from main thread */ static int start_thread(struct userdata *u) { pa_assert(u); pa_assert(!u->thread); pa_assert(!u->rtpoll); pa_assert(!u->rtpoll_item); u->rtpoll = pa_rtpoll_new(); if (pa_thread_mq_init(&u->thread_mq, u->core->mainloop, u->rtpoll) < 0) { pa_log("pa_thread_mq_init() failed."); return -1; } if (!(u->thread = pa_thread_new("bluetooth", thread_func, u))) { pa_log_error("Failed to create IO thread"); return -1; } if (u->sink) { pa_sink_set_asyncmsgq(u->sink, u->thread_mq.inq); pa_sink_set_rtpoll(u->sink, u->rtpoll); /* If we are in the headset role, the sink should not become default * unless there is no other sound device available. */ if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY) u->sink->priority = 1500; pa_sink_put(u->sink); if (u->sink->set_volume) u->sink->set_volume(u->sink); } if (u->source) { pa_source_set_asyncmsgq(u->source, u->thread_mq.inq); pa_source_set_rtpoll(u->source, u->rtpoll); /* If we are in the headset role or the device is an a2dp source, * the source should not become default unless there is no other * sound device available. */ if (u->profile == PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY || u->profile == PA_BLUETOOTH_PROFILE_A2DP_SOURCE) u->source->priority = 1500; pa_source_put(u->source); if (u->source->set_volume) u->source->set_volume(u->source); } return 0; } /* Run from main thread */ static void stop_thread(struct userdata *u) { pa_assert(u); if (u->sink) pa_sink_unlink(u->sink); if (u->source) pa_source_unlink(u->source); if (u->thread) { pa_asyncmsgq_send(u->thread_mq.inq, NULL, PA_MESSAGE_SHUTDOWN, NULL, 0, NULL); pa_thread_free(u->thread); u->thread = NULL; } if (u->rtpoll_item) { pa_rtpoll_item_free(u->rtpoll_item); u->rtpoll_item = NULL; } if (u->rtpoll) { pa_rtpoll_free(u->rtpoll); u->rtpoll = NULL; pa_thread_mq_done(&u->thread_mq); } if (u->transport) { transport_release(u); u->transport = NULL; } if (u->sink) { pa_sink_unref(u->sink); u->sink = NULL; } if (u->source) { pa_source_unref(u->source); u->source = NULL; } if (u->read_smoother) { pa_smoother_free(u->read_smoother); u->read_smoother = NULL; } if (u->profile == PA_BLUETOOTH_PROFILE_A2DP_SINK || u->profile == PA_BLUETOOTH_PROFILE_A2DP_SOURCE) { if (u->encoder_info) { u->a2dp_codec->deinit(u->encoder_info); u->encoder_info = NULL; } if (u->decoder_info) { u->a2dp_codec->deinit(u->decoder_info); u->decoder_info = NULL; } u->a2dp_codec = NULL; } } /* Run from main thread */ static pa_available_t get_port_availability(struct userdata *u, pa_direction_t direction) { pa_available_t result = PA_AVAILABLE_NO; unsigned i; pa_assert(u); pa_assert(u->device); for (i = 0; i < PA_BLUETOOTH_PROFILE_COUNT; i++) { pa_bluetooth_transport *transport; if (!(get_profile_direction(i) & direction)) continue; if (!(transport = u->device->transports[i])) continue; switch(transport->state) { case PA_BLUETOOTH_TRANSPORT_STATE_DISCONNECTED: continue; case PA_BLUETOOTH_TRANSPORT_STATE_IDLE: if (result == PA_AVAILABLE_NO) result = PA_AVAILABLE_UNKNOWN; break; case PA_BLUETOOTH_TRANSPORT_STATE_PLAYING: return PA_AVAILABLE_YES; } } return result; } /* Run from main thread */ static pa_available_t transport_state_to_availability(pa_bluetooth_transport_state_t state) { switch (state) { case PA_BLUETOOTH_TRANSPORT_STATE_DISCONNECTED: return PA_AVAILABLE_NO; case PA_BLUETOOTH_TRANSPORT_STATE_PLAYING: return PA_AVAILABLE_YES; default: return PA_AVAILABLE_UNKNOWN; } } /* Run from main thread */ static void create_card_ports(struct userdata *u, pa_hashmap *ports) { pa_device_port *port; pa_device_port_new_data port_data; pa_device_port_type_t input_type, output_type; const char *name_prefix, *input_description, *output_description; pa_assert(u); pa_assert(ports); pa_assert(u->device); name_prefix = "unknown"; input_description = _("Bluetooth Input"); output_description = _("Bluetooth Output"); input_type = output_type = PA_DEVICE_PORT_TYPE_BLUETOOTH; switch (form_factor_from_class(u->device->class_of_device)) { case PA_BLUETOOTH_FORM_FACTOR_HEADSET: name_prefix = "headset"; input_description = output_description = _("Headset"); input_type = output_type = PA_DEVICE_PORT_TYPE_HEADSET; break; case PA_BLUETOOTH_FORM_FACTOR_HANDSFREE: name_prefix = "handsfree"; input_description = output_description = _("Handsfree"); input_type = output_type = PA_DEVICE_PORT_TYPE_HANDSFREE; break; case PA_BLUETOOTH_FORM_FACTOR_MICROPHONE: name_prefix = "microphone"; input_description = _("Microphone"); output_description = _("Bluetooth Output"); input_type = PA_DEVICE_PORT_TYPE_MIC; break; case PA_BLUETOOTH_FORM_FACTOR_SPEAKER: name_prefix = "speaker"; input_description = _("Bluetooth Input"); output_description = _("Speaker"); output_type = PA_DEVICE_PORT_TYPE_SPEAKER; break; case PA_BLUETOOTH_FORM_FACTOR_HEADPHONE: name_prefix = "headphone"; input_description = _("Bluetooth Input"); output_description = _("Headphone"); output_type = PA_DEVICE_PORT_TYPE_HEADPHONES; break; case PA_BLUETOOTH_FORM_FACTOR_PORTABLE: name_prefix = "portable"; input_description = output_description = _("Portable"); input_type = output_type = PA_DEVICE_PORT_TYPE_PORTABLE; break; case PA_BLUETOOTH_FORM_FACTOR_CAR: name_prefix = "car"; input_description = output_description = _("Car"); input_type = output_type = PA_DEVICE_PORT_TYPE_CAR; break; case PA_BLUETOOTH_FORM_FACTOR_HIFI: name_prefix = "hifi"; input_description = output_description = _("HiFi"); input_type = output_type = PA_DEVICE_PORT_TYPE_HIFI; break; case PA_BLUETOOTH_FORM_FACTOR_PHONE: name_prefix = "phone"; input_description = output_description = _("Phone"); input_type = output_type = PA_DEVICE_PORT_TYPE_PHONE; break; case PA_BLUETOOTH_FORM_FACTOR_UNKNOWN: break; } u->output_port_name = pa_sprintf_malloc("%s-output", name_prefix); pa_device_port_new_data_init(&port_data); pa_device_port_new_data_set_name(&port_data, u->output_port_name); pa_device_port_new_data_set_description(&port_data, output_description); pa_device_port_new_data_set_direction(&port_data, PA_DIRECTION_OUTPUT); pa_device_port_new_data_set_type(&port_data, output_type); pa_device_port_new_data_set_available(&port_data, get_port_availability(u, PA_DIRECTION_OUTPUT)); pa_assert_se(port = pa_device_port_new(u->core, &port_data, 0)); pa_assert_se(pa_hashmap_put(ports, port->name, port) >= 0); pa_device_port_new_data_done(&port_data); u->input_port_name = pa_sprintf_malloc("%s-input", name_prefix); pa_device_port_new_data_init(&port_data); pa_device_port_new_data_set_name(&port_data, u->input_port_name); pa_device_port_new_data_set_description(&port_data, input_description); pa_device_port_new_data_set_direction(&port_data, PA_DIRECTION_INPUT); pa_device_port_new_data_set_type(&port_data, input_type); pa_device_port_new_data_set_available(&port_data, get_port_availability(u, PA_DIRECTION_INPUT)); pa_assert_se(port = pa_device_port_new(u->core, &port_data, 0)); pa_assert_se(pa_hashmap_put(ports, port->name, port) >= 0); pa_device_port_new_data_done(&port_data); } /* Run from main thread */ static pa_card_profile *create_card_profile(struct userdata *u, pa_bluetooth_profile_t profile, pa_hashmap *ports) { pa_device_port *input_port, *output_port; const char *name; pa_card_profile *cp = NULL; pa_bluetooth_profile_t *p; pa_assert(u->input_port_name); pa_assert(u->output_port_name); pa_assert_se(input_port = pa_hashmap_get(ports, u->input_port_name)); pa_assert_se(output_port = pa_hashmap_get(ports, u->output_port_name)); name = pa_bluetooth_profile_to_string(profile); switch (profile) { case PA_BLUETOOTH_PROFILE_A2DP_SINK: cp = pa_card_profile_new(name, _("High Fidelity Playback (A2DP Sink)"), sizeof(pa_bluetooth_profile_t)); cp->priority = 40; cp->n_sinks = 1; cp->n_sources = 0; cp->max_sink_channels = 2; cp->max_source_channels = 0; pa_hashmap_put(output_port->profiles, cp->name, cp); p = PA_CARD_PROFILE_DATA(cp); break; case PA_BLUETOOTH_PROFILE_A2DP_SOURCE: cp = pa_card_profile_new(name, _("High Fidelity Capture (A2DP Source)"), sizeof(pa_bluetooth_profile_t)); cp->priority = 20; cp->n_sinks = 0; cp->n_sources = 1; cp->max_sink_channels = 0; cp->max_source_channels = 2; pa_hashmap_put(input_port->profiles, cp->name, cp); p = PA_CARD_PROFILE_DATA(cp); break; case PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT: cp = pa_card_profile_new(name, _("Headset Head Unit (HSP/HFP)"), sizeof(pa_bluetooth_profile_t)); cp->priority = 30; cp->n_sinks = 1; cp->n_sources = 1; cp->max_sink_channels = 1; cp->max_source_channels = 1; pa_hashmap_put(input_port->profiles, cp->name, cp); pa_hashmap_put(output_port->profiles, cp->name, cp); p = PA_CARD_PROFILE_DATA(cp); break; case PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY: cp = pa_card_profile_new(name, _("Headset Audio Gateway (HSP/HFP)"), sizeof(pa_bluetooth_profile_t)); cp->priority = 10; cp->n_sinks = 1; cp->n_sources = 1; cp->max_sink_channels = 1; cp->max_source_channels = 1; pa_hashmap_put(input_port->profiles, cp->name, cp); pa_hashmap_put(output_port->profiles, cp->name, cp); p = PA_CARD_PROFILE_DATA(cp); break; case PA_BLUETOOTH_PROFILE_OFF: pa_assert_not_reached(); } *p = profile; if (u->device->transports[*p]) cp->available = transport_state_to_availability(u->device->transports[*p]->state); else cp->available = PA_AVAILABLE_NO; return cp; } /* Run from main thread */ static int set_profile_cb(pa_card *c, pa_card_profile *new_profile) { struct userdata *u; pa_bluetooth_profile_t *p; pa_assert(c); pa_assert(new_profile); pa_assert_se(u = c->userdata); p = PA_CARD_PROFILE_DATA(new_profile); if (*p != PA_BLUETOOTH_PROFILE_OFF) { const pa_bluetooth_device *d = u->device; if (!d->transports[*p] || d->transports[*p]->state <= PA_BLUETOOTH_TRANSPORT_STATE_DISCONNECTED) { pa_log_warn("Refused to switch profile to %s: Not connected", new_profile->name); return -PA_ERR_IO; } } stop_thread(u); u->profile = *p; if (u->profile != PA_BLUETOOTH_PROFILE_OFF) if (init_profile(u) < 0) goto off; if (u->sink || u->source) if (start_thread(u) < 0) goto off; return 0; off: stop_thread(u); pa_assert_se(pa_card_set_profile(u->card, pa_hashmap_get(u->card->profiles, "off"), false) >= 0); return -PA_ERR_IO; } static int uuid_to_profile(const char *uuid, pa_bluetooth_profile_t *_r) { if (pa_streq(uuid, PA_BLUETOOTH_UUID_A2DP_SINK)) *_r = PA_BLUETOOTH_PROFILE_A2DP_SINK; else if (pa_streq(uuid, PA_BLUETOOTH_UUID_A2DP_SOURCE)) *_r = PA_BLUETOOTH_PROFILE_A2DP_SOURCE; else if (pa_bluetooth_uuid_is_hsp_hs(uuid) || pa_streq(uuid, PA_BLUETOOTH_UUID_HFP_HF)) *_r = PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT; else if (pa_streq(uuid, PA_BLUETOOTH_UUID_HSP_AG) || pa_streq(uuid, PA_BLUETOOTH_UUID_HFP_AG)) *_r = PA_BLUETOOTH_PROFILE_HEADSET_AUDIO_GATEWAY; else return -PA_ERR_INVALID; return 0; } /* Run from main thread */ static int add_card(struct userdata *u) { const pa_bluetooth_device *d; pa_card_new_data data; char *alias; pa_bluetooth_form_factor_t ff; pa_card_profile *cp; pa_bluetooth_profile_t *p; const char *uuid; void *state; pa_assert(u); pa_assert(u->device); d = u->device; pa_card_new_data_init(&data); data.driver = __FILE__; data.module = u->module; alias = pa_utf8_filter(d->alias); pa_proplist_sets(data.proplist, PA_PROP_DEVICE_DESCRIPTION, alias); pa_xfree(alias); pa_proplist_sets(data.proplist, PA_PROP_DEVICE_STRING, d->address); pa_proplist_sets(data.proplist, PA_PROP_DEVICE_API, "bluez"); pa_proplist_sets(data.proplist, PA_PROP_DEVICE_CLASS, "sound"); pa_proplist_sets(data.proplist, PA_PROP_DEVICE_BUS, "bluetooth"); if ((ff = form_factor_from_class(d->class_of_device)) != PA_BLUETOOTH_FORM_FACTOR_UNKNOWN) pa_proplist_sets(data.proplist, PA_PROP_DEVICE_FORM_FACTOR, form_factor_to_string(ff)); pa_proplist_sets(data.proplist, "bluez.path", d->path); pa_proplist_setf(data.proplist, "bluez.class", "0x%06x", d->class_of_device); pa_proplist_sets(data.proplist, "bluez.alias", d->alias); data.name = pa_sprintf_malloc("bluez_card.%s", d->address); data.namereg_fail = false; create_card_ports(u, data.ports); PA_HASHMAP_FOREACH(uuid, d->uuids, state) { pa_bluetooth_profile_t profile; if (uuid_to_profile(uuid, &profile) < 0) continue; if (pa_hashmap_get(data.profiles, pa_bluetooth_profile_to_string(profile))) continue; cp = create_card_profile(u, profile, data.ports); pa_hashmap_put(data.profiles, cp->name, cp); } pa_assert(!pa_hashmap_isempty(data.profiles)); cp = pa_card_profile_new("off", _("Off"), sizeof(pa_bluetooth_profile_t)); cp->available = PA_AVAILABLE_YES; p = PA_CARD_PROFILE_DATA(cp); *p = PA_BLUETOOTH_PROFILE_OFF; pa_hashmap_put(data.profiles, cp->name, cp); u->card = pa_card_new(u->core, &data); pa_card_new_data_done(&data); if (!u->card) { pa_log("Failed to allocate card."); return -1; } u->card->userdata = u; u->card->set_profile = set_profile_cb; pa_card_choose_initial_profile(u->card); pa_card_put(u->card); p = PA_CARD_PROFILE_DATA(u->card->active_profile); u->profile = *p; return 0; } /* Run from main thread */ static void handle_transport_state_change(struct userdata *u, struct pa_bluetooth_transport *t) { bool acquire = false; bool release = false; pa_card_profile *cp; pa_device_port *port; pa_available_t oldavail; pa_assert(u); pa_assert(t); pa_assert_se(cp = pa_hashmap_get(u->card->profiles, pa_bluetooth_profile_to_string(t->profile))); oldavail = cp->available; pa_card_profile_set_available(cp, transport_state_to_availability(t->state)); /* Update port availability */ pa_assert_se(port = pa_hashmap_get(u->card->ports, u->output_port_name)); pa_device_port_set_available(port, get_port_availability(u, PA_DIRECTION_OUTPUT)); pa_assert_se(port = pa_hashmap_get(u->card->ports, u->input_port_name)); pa_device_port_set_available(port, get_port_availability(u, PA_DIRECTION_INPUT)); /* Acquire or release transport as needed */ acquire = (t->state == PA_BLUETOOTH_TRANSPORT_STATE_PLAYING && u->profile == t->profile); release = (oldavail != PA_AVAILABLE_NO && t->state != PA_BLUETOOTH_TRANSPORT_STATE_PLAYING && u->profile == t->profile); if (acquire && transport_acquire(u, true) >= 0) { if (u->source) { pa_log_debug("Resuming source %s because its transport state changed to playing", u->source->name); /* When the ofono backend resumes source or sink when in the audio gateway role, the * state of source or sink may already be RUNNING before the transport is acquired via * hf_audio_agent_new_connection(), so the pa_source_suspend() call will not lead to a * state change message. In this case we explicitly need to signal the I/O thread to * set up the stream. */ if (PA_SOURCE_IS_OPENED(u->source->state)) pa_asyncmsgq_send(u->source->asyncmsgq, PA_MSGOBJECT(u->source), PA_SOURCE_MESSAGE_SETUP_STREAM, NULL, 0, NULL); /* We remove the IDLE suspend cause, because otherwise * module-loopback doesn't uncork its streams. FIXME: Messing with * the IDLE suspend cause here is wrong, the correct way to handle * this would probably be to uncork the loopback streams not only * when the other end is unsuspended, but also when the other end's * suspend cause changes to IDLE only (currently there's no * notification mechanism for suspend cause changes, though). */ pa_source_suspend(u->source, false, PA_SUSPEND_IDLE|PA_SUSPEND_USER); } if (u->sink) { pa_log_debug("Resuming sink %s because its transport state changed to playing", u->sink->name); /* Same comment as above */ if (PA_SINK_IS_OPENED(u->sink->state)) pa_asyncmsgq_send(u->sink->asyncmsgq, PA_MSGOBJECT(u->sink), PA_SINK_MESSAGE_SETUP_STREAM, NULL, 0, NULL); /* FIXME: See the previous comment. */ pa_sink_suspend(u->sink, false, PA_SUSPEND_IDLE|PA_SUSPEND_USER); } } if (release && u->transport_acquired) { /* FIXME: this release is racy, since the audio stream might have * been set up again in the meantime (but not processed yet by PA). * BlueZ should probably release the transport automatically, and in * that case we would just mark the transport as released */ /* Remote side closed the stream so we consider it PA_SUSPEND_USER */ if (u->source) { pa_log_debug("Suspending source %s because the remote end closed the stream", u->source->name); pa_source_suspend(u->source, true, PA_SUSPEND_USER); } if (u->sink) { pa_log_debug("Suspending sink %s because the remote end closed the stream", u->sink->name); pa_sink_suspend(u->sink, true, PA_SUSPEND_USER); } } } /* Run from main thread */ static pa_hook_result_t device_connection_changed_cb(pa_bluetooth_discovery *y, const pa_bluetooth_device *d, struct userdata *u) { pa_assert(d); pa_assert(u); if (d != u->device || pa_bluetooth_device_any_transport_connected(d)) return PA_HOOK_OK; pa_log_debug("Unloading module for device %s", d->path); pa_module_unload(u->module, true); return PA_HOOK_OK; } /* Run from main thread */ static pa_hook_result_t transport_state_changed_cb(pa_bluetooth_discovery *y, pa_bluetooth_transport *t, struct userdata *u) { pa_assert(t); pa_assert(u); if (t == u->transport && t->state <= PA_BLUETOOTH_TRANSPORT_STATE_DISCONNECTED) pa_assert_se(pa_card_set_profile(u->card, pa_hashmap_get(u->card->profiles, "off"), false) >= 0); if (t->device == u->device) handle_transport_state_change(u, t); return PA_HOOK_OK; } static pa_hook_result_t transport_speaker_gain_changed_cb(pa_bluetooth_discovery *y, pa_bluetooth_transport *t, struct userdata *u) { pa_volume_t volume; pa_cvolume v; uint16_t gain; pa_assert(t); pa_assert(u); if (t != u->transport) return PA_HOOK_OK; gain = t->speaker_gain; volume = (pa_volume_t) (gain * PA_VOLUME_NORM / HSP_MAX_GAIN); /* increment volume by one to correct rounding errors */ if (volume < PA_VOLUME_NORM) volume++; pa_cvolume_set(&v, u->encoder_sample_spec.channels, volume); if (t->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT) pa_sink_volume_changed(u->sink, &v); else pa_sink_set_volume(u->sink, &v, true, true); return PA_HOOK_OK; } static pa_hook_result_t transport_microphone_gain_changed_cb(pa_bluetooth_discovery *y, pa_bluetooth_transport *t, struct userdata *u) { pa_volume_t volume; pa_cvolume v; uint16_t gain; pa_assert(t); pa_assert(u); if (t != u->transport) return PA_HOOK_OK; gain = t->microphone_gain; volume = (pa_volume_t) (gain * PA_VOLUME_NORM / HSP_MAX_GAIN); /* increment volume by one to correct rounding errors */ if (volume < PA_VOLUME_NORM) volume++; pa_cvolume_set(&v, u->decoder_sample_spec.channels, volume); if (t->profile == PA_BLUETOOTH_PROFILE_HEADSET_HEAD_UNIT) pa_source_volume_changed(u->source, &v); else pa_source_set_volume(u->source, &v, true, true); return PA_HOOK_OK; } /* Run from main thread context */ static int device_process_msg(pa_msgobject *obj, int code, void *data, int64_t offset, pa_memchunk *chunk) { struct bluetooth_msg *m = BLUETOOTH_MSG(obj); struct userdata *u = m->card->userdata; switch (code) { case BLUETOOTH_MESSAGE_IO_THREAD_FAILED: if (m->card->module->unload_requested) break; pa_log_debug("Switching the profile to off due to IO thread failure."); pa_assert_se(pa_card_set_profile(m->card, pa_hashmap_get(m->card->profiles, "off"), false) >= 0); break; case BLUETOOTH_MESSAGE_STREAM_FD_HUP: if (u->transport->state > PA_BLUETOOTH_TRANSPORT_STATE_IDLE) pa_bluetooth_transport_set_state(u->transport, PA_BLUETOOTH_TRANSPORT_STATE_IDLE); break; case BLUETOOTH_MESSAGE_SET_TRANSPORT_PLAYING: /* transport_acquired needs to be checked here, because a message could have been * pending when the profile was switched. If the new transport has been acquired * correctly, the call below will have no effect because the transport state is * already PLAYING. If transport_acquire() failed for the new profile, the transport * state should not be changed. If the transport has been released for other reasons * (I/O thread shutdown), transport_acquired will also be false. */ if (u->transport_acquired) pa_bluetooth_transport_set_state(u->transport, PA_BLUETOOTH_TRANSPORT_STATE_PLAYING); break; } return 0; } int pa__init(pa_module* m) { struct userdata *u; const char *path; pa_modargs *ma; bool autodetect_mtu; pa_assert(m); m->userdata = u = pa_xnew0(struct userdata, 1); u->module = m; u->core = m->core; if (!(ma = pa_modargs_new(m->argument, valid_modargs))) { pa_log_error("Failed to parse module arguments"); goto fail_free_modargs; } if (!(path = pa_modargs_get_value(ma, "path", NULL))) { pa_log_error("Failed to get device path from module arguments"); goto fail_free_modargs; } if ((u->discovery = pa_shared_get(u->core, "bluetooth-discovery"))) pa_bluetooth_discovery_ref(u->discovery); else { pa_log_error("module-bluez5-discover doesn't seem to be loaded, refusing to load module-bluez5-device"); goto fail_free_modargs; } if (!(u->device = pa_bluetooth_discovery_get_device_by_path(u->discovery, path))) { pa_log_error("%s is unknown", path); goto fail_free_modargs; } autodetect_mtu = false; if (pa_modargs_get_value_boolean(ma, "autodetect_mtu", &autodetect_mtu) < 0) { pa_log("Invalid boolean value for autodetect_mtu parameter"); goto fail_free_modargs; } u->device->autodetect_mtu = autodetect_mtu; pa_modargs_free(ma); u->device_connection_changed_slot = pa_hook_connect(pa_bluetooth_discovery_hook(u->discovery, PA_BLUETOOTH_HOOK_DEVICE_CONNECTION_CHANGED), PA_HOOK_NORMAL, (pa_hook_cb_t) device_connection_changed_cb, u); u->transport_state_changed_slot = pa_hook_connect(pa_bluetooth_discovery_hook(u->discovery, PA_BLUETOOTH_HOOK_TRANSPORT_STATE_CHANGED), PA_HOOK_NORMAL, (pa_hook_cb_t) transport_state_changed_cb, u); u->transport_speaker_gain_changed_slot = pa_hook_connect(pa_bluetooth_discovery_hook(u->discovery, PA_BLUETOOTH_HOOK_TRANSPORT_SPEAKER_GAIN_CHANGED), PA_HOOK_NORMAL, (pa_hook_cb_t) transport_speaker_gain_changed_cb, u); u->transport_microphone_gain_changed_slot = pa_hook_connect(pa_bluetooth_discovery_hook(u->discovery, PA_BLUETOOTH_HOOK_TRANSPORT_MICROPHONE_GAIN_CHANGED), PA_HOOK_NORMAL, (pa_hook_cb_t) transport_microphone_gain_changed_cb, u); if (add_card(u) < 0) goto fail; if (!(u->msg = pa_msgobject_new(bluetooth_msg))) goto fail; u->msg->parent.process_msg = device_process_msg; u->msg->card = u->card; u->stream_setup_done = false; if (u->profile != PA_BLUETOOTH_PROFILE_OFF) if (init_profile(u) < 0) goto off; if (u->sink || u->source) if (start_thread(u) < 0) goto off; return 0; off: stop_thread(u); pa_assert_se(pa_card_set_profile(u->card, pa_hashmap_get(u->card->profiles, "off"), false) >= 0); return 0; fail_free_modargs: if (ma) pa_modargs_free(ma); fail: pa__done(m); return -1; } void pa__done(pa_module *m) { struct userdata *u; pa_assert(m); if (!(u = m->userdata)) return; stop_thread(u); if (u->device_connection_changed_slot) pa_hook_slot_free(u->device_connection_changed_slot); if (u->transport_state_changed_slot) pa_hook_slot_free(u->transport_state_changed_slot); if (u->transport_speaker_gain_changed_slot) pa_hook_slot_free(u->transport_speaker_gain_changed_slot); if (u->transport_microphone_gain_changed_slot) pa_hook_slot_free(u->transport_microphone_gain_changed_slot); if (u->encoder_buffer) pa_xfree(u->encoder_buffer); if (u->decoder_buffer) pa_xfree(u->decoder_buffer); if (u->msg) pa_xfree(u->msg); if (u->card) pa_card_free(u->card); if (u->discovery) pa_bluetooth_discovery_unref(u->discovery); pa_xfree(u->output_port_name); pa_xfree(u->input_port_name); pa_xfree(u); } int pa__get_n_used(pa_module *m) { struct userdata *u; pa_assert(m); pa_assert_se(u = m->userdata); return (u->sink ? pa_sink_linked_by(u->sink) : 0) + (u->source ? pa_source_linked_by(u->source) : 0); }