[RFC 02/12] bluetooth: Use new API for exposing A2DP MPEG sinks

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



From: Arun Raghavan <arun.raghavan@xxxxxxxxxxxxxxx>

This replaces a separate A2DP passthrough profile with dynamically
reconfiguring the A2DP stream for MPEG streams if required. The sink
get_formats() function allows the core to introspect whether the
attached headset supports this or not.
---
 src/modules/bluetooth/bluetooth-util.h          |    1 -
 src/modules/bluetooth/module-bluetooth-device.c |  304 +++++++++++++++--------
 2 files changed, 194 insertions(+), 111 deletions(-)

diff --git a/src/modules/bluetooth/bluetooth-util.h b/src/modules/bluetooth/bluetooth-util.h
index 7ca9cb2..2752a69 100644
--- a/src/modules/bluetooth/bluetooth-util.h
+++ b/src/modules/bluetooth/bluetooth-util.h
@@ -58,7 +58,6 @@ struct pa_bluetooth_uuid {
 enum profile {
     PROFILE_A2DP,
     PROFILE_A2DP_SOURCE,
-    PROFILE_A2DP_PASSTHROUGH,
     PROFILE_HSP,
     PROFILE_HFGW,
     PROFILE_OFF
diff --git a/src/modules/bluetooth/module-bluetooth-device.c b/src/modules/bluetooth/module-bluetooth-device.c
index 049b0a8..6b97191 100644
--- a/src/modules/bluetooth/module-bluetooth-device.c
+++ b/src/modules/bluetooth/module-bluetooth-device.c
@@ -102,10 +102,17 @@ static const char* const valid_modargs[] = {
     NULL
 };
 
+typedef enum {
+    A2DP_MODE_SBC,
+    A2DP_MODE_MPEG,
+} a2dp_mode_t;
+
 struct a2dp_info {
     sbc_capabilities_t sbc_capabilities;
     mpeg_capabilities_t mpeg_capabilities;
-    pa_bool_t           has_mpeg;
+
+    a2dp_mode_t mode;
+    pa_bool_t has_mpeg;
 
     sbc_t sbc;                           /* Codec data */
     pa_bool_t sbc_initialized;           /* Keep track if the encoder is initialized */
@@ -188,7 +195,7 @@ struct userdata {
 
     pa_bool_t filter_added;
 
-    /* required for PASSTHROUGH profile */
+    /* required for MPEG transport */
     size_t leftover_bytes;
 };
 
@@ -321,7 +328,6 @@ static int parse_mpeg_caps(struct userdata *u, uint8_t seid, const struct bt_get
 
     pa_assert(u);
     pa_assert(rsp);
-    pa_assert( u->profile == PROFILE_A2DP_PASSTHROUGH );
 
     u->a2dp.has_mpeg = FALSE;
 
@@ -428,6 +434,7 @@ static int parse_caps(struct userdata *u, uint8_t seid, const struct bt_get_capa
             return codec->seid;
 
         memcpy(&u->a2dp.sbc_capabilities, codec, sizeof(u->a2dp.sbc_capabilities));
+        pa_log_info("SBC caps detected");
 
     } else if (u->profile == PROFILE_A2DP_SOURCE) {
 
@@ -472,8 +479,7 @@ static int get_caps_msg(struct userdata *u, uint8_t seid, bt_getcaps_msg_t *msg)
     msg->getcaps_req.seid = seid;
 
     pa_strlcpy(msg->getcaps_req.object, u->path, sizeof(msg->getcaps_req.object));
-    if (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_PASSTHROUGH ||
-        u->profile == PROFILE_A2DP_SOURCE)
+    if (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_SOURCE)
         msg->getcaps_req.transport = BT_CAPABILITIES_TRANSPORT_A2DP;
     else {
         pa_assert(u->profile == PROFILE_HSP || u->profile == PROFILE_HFGW);
@@ -514,27 +520,28 @@ static int get_caps(struct userdata *u, uint8_t seid) {
             return -1;
     }
 
-    if (u->profile == PROFILE_A2DP_PASSTHROUGH) {
-        seid = 0;
+    seid = 0;
 
+    if (u->profile == PROFILE_A2DP) {
         /* try to find mpeg end-point */
         if (get_caps_msg(u, seid, &msg) < 0)
-            return -1;
+            return 0;
 
         ret = parse_mpeg_caps(u, seid, &msg.getcaps_rsp);
         if (ret < 0)
-            return -1;
+            return 0;
 
         if (ret > 0) {
             /* refine seid caps */
             if (get_caps_msg(u, ret, &msg) < 0)
-                return -1;
+                return 0;
 
             ret = parse_mpeg_caps(u, ret, &msg.getcaps_rsp);
             if (ret < 0)
-                return -1;
+                return 0;
         }
     }
+
     return 0;
 }
 
@@ -600,54 +607,10 @@ static int setup_a2dp(struct userdata *u) {
     };
 
     pa_assert(u);
-    pa_assert(u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_PASSTHROUGH || u->profile == PROFILE_A2DP_SOURCE);
-
-    if (u->profile == PROFILE_A2DP_PASSTHROUGH) {
-        mpeg_capabilities_t *mcap;
-
-        if (u->a2dp.has_mpeg) {
-            int rate;
-
-            mcap = &u->a2dp.mpeg_capabilities;
-            rate = u->sample_spec.rate;
-
-            if (u->sample_spec.channels == 1)
-                mcap->channel_mode = BT_A2DP_CHANNEL_MODE_MONO;
-            else
-                mcap->channel_mode = BT_A2DP_CHANNEL_MODE_STEREO;
-
-            mcap->crc = 0; /* CRC is broken in some encoders */
-            mcap->layer = BT_MPEG_LAYER_3;
-
-            if (rate == 44100)
-                mcap->frequency = BT_MPEG_SAMPLING_FREQ_44100;
-            else if (rate == 48000)
-                mcap->frequency = BT_MPEG_SAMPLING_FREQ_48000;
-            else if (rate == 32000)
-                mcap->frequency = BT_MPEG_SAMPLING_FREQ_32000;
-            else if (rate == 24000)
-                mcap->frequency = BT_MPEG_SAMPLING_FREQ_24000;
-            else if (rate == 22050)
-                mcap->frequency = BT_MPEG_SAMPLING_FREQ_22050;
-            else if (rate == 16000)
-                mcap->frequency = BT_MPEG_SAMPLING_FREQ_16000;
-            else {
-                pa_log("unsupported sampling frequency");
-                return -1;
-            }
-
-            mcap->mpf = 0; /* don't use optional IETF payload, send raw frames */
-            mcap->bitrate = 0x8000; /* set for vbr, this covers all cases */
+    pa_assert(u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_SOURCE);
 
-            return 0;
-        }
-        else {
-            pa_log("setup_a2dp: Trying to set-up A2DP Passthrough configuration but no MPEG endpoint available");
-            return -1; /* return error, profile will not be loaded */
-        }
-    }
+    mpeg_capabilities_t *mcap;
 
-    /* other profiles */
     cap = &u->a2dp.sbc_capabilities;
 
     /* Find the lowest freq that is at least as high as the requested
@@ -735,6 +698,42 @@ static int setup_a2dp(struct userdata *u) {
     cap->min_bitpool = (uint8_t) PA_MAX(MIN_BITPOOL, cap->min_bitpool);
     cap->max_bitpool = (uint8_t) PA_MIN(a2dp_default_bitpool(cap->frequency, cap->channel_mode), cap->max_bitpool);
 
+    /* Now convigure the MPEG caps if we have them */
+    if (u->a2dp.has_mpeg) {
+        int rate;
+
+        mcap = &u->a2dp.mpeg_capabilities;
+        rate = u->sample_spec.rate;
+
+        if (u->sample_spec.channels == 1)
+            mcap->channel_mode = BT_A2DP_CHANNEL_MODE_MONO;
+        else
+            mcap->channel_mode = BT_A2DP_CHANNEL_MODE_STEREO;
+
+        mcap->crc = 0; /* CRC is broken in some encoders */
+        mcap->layer = BT_MPEG_LAYER_3;
+
+        if (rate == 44100)
+            mcap->frequency = BT_MPEG_SAMPLING_FREQ_44100;
+        else if (rate == 48000)
+            mcap->frequency = BT_MPEG_SAMPLING_FREQ_48000;
+        else if (rate == 32000)
+            mcap->frequency = BT_MPEG_SAMPLING_FREQ_32000;
+        else if (rate == 24000)
+            mcap->frequency = BT_MPEG_SAMPLING_FREQ_24000;
+        else if (rate == 22050)
+            mcap->frequency = BT_MPEG_SAMPLING_FREQ_22050;
+        else if (rate == 16000)
+            mcap->frequency = BT_MPEG_SAMPLING_FREQ_16000;
+        else {
+            pa_log("unsupported MPEG sampling frequency");
+            return -1;
+        }
+
+        mcap->mpf = 0; /* don't use optional IETF payload, send raw frames */
+        mcap->bitrate = 0x8000; /* set for vbr, this covers all cases */
+    }
+
     return 0;
 }
 
@@ -853,15 +852,16 @@ static int set_conf(struct userdata *u) {
     pa_strlcpy(msg.open_req.object, u->path, sizeof(msg.open_req.object));
 
     if (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_SOURCE) {
-        msg.open_req.seid = u->a2dp.sbc_capabilities.capability.seid;
-    } else if (u->profile == PROFILE_A2DP_PASSTHROUGH) {
-        pa_assert(u->a2dp.has_mpeg);
-        msg.open_req.seid = u->a2dp.mpeg_capabilities.capability.seid;
-    } else {
+        if (u->a2dp.mode == A2DP_MODE_SBC)
+            msg.open_req.seid = u->a2dp.sbc_capabilities.capability.seid;
+        else if (u->a2dp.mode == A2DP_MODE_MPEG) {
+            pa_assert(u->a2dp.has_mpeg);
+            msg.open_req.seid = u->a2dp.mpeg_capabilities.capability.seid;
+        }
+    } else
         msg.open_req.seid = BT_A2DP_SEID_RANGE + 1;
-    }
 
-    msg.open_req.lock = (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_PASSTHROUGH) ? BT_WRITE_LOCK : BT_READ_LOCK | BT_WRITE_LOCK;
+    msg.open_req.lock = u->profile == PROFILE_A2DP ? BT_WRITE_LOCK : BT_READ_LOCK | BT_WRITE_LOCK;
 
     if (service_send(u, &msg.open_req.h) < 0)
         return -1;
@@ -869,7 +869,7 @@ static int set_conf(struct userdata *u) {
     if (service_expect(u, &msg.open_rsp.h, sizeof(msg), BT_OPEN, sizeof(msg.open_rsp)) < 0)
         return -1;
 
-    if (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_PASSTHROUGH || u->profile == PROFILE_A2DP_SOURCE ) {
+    if (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_SOURCE ) {
 
         /* for passthrough we expect little-endian data to be compatible with the ALSA iec958
            devices */
@@ -891,10 +891,10 @@ static int set_conf(struct userdata *u) {
     msg.setconf_req.h.length = sizeof(msg.setconf_req);
 
     if (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_SOURCE) {
-        memcpy(&msg.setconf_req.codec, &u->a2dp.sbc_capabilities, sizeof(u->a2dp.sbc_capabilities));
-    } else if (u->profile == PROFILE_A2DP_PASSTHROUGH) {
-        pa_assert(u->a2dp.has_mpeg);
-        memcpy(&msg.setconf_req.codec, &u->a2dp.mpeg_capabilities, sizeof(u->a2dp.mpeg_capabilities));
+        if (u->a2dp.mode == A2DP_MODE_SBC)
+            memcpy(&msg.setconf_req.codec, &u->a2dp.sbc_capabilities, sizeof(u->a2dp.sbc_capabilities));
+        else if (u->a2dp.mode == A2DP_MODE_MPEG)
+            memcpy(&msg.setconf_req.codec, &u->a2dp.mpeg_capabilities, sizeof(u->a2dp.mpeg_capabilities));
     } else {
         msg.setconf_req.codec.transport = BT_CAPABILITIES_TRANSPORT_SCO;
         msg.setconf_req.codec.seid = BT_A2DP_SEID_RANGE + 1;
@@ -912,21 +912,21 @@ static int set_conf(struct userdata *u) {
 
     /* setup SBC encoder now we agree on parameters */
     if (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_SOURCE) {
-        setup_sbc(&u->a2dp, u->profile);
+        if (u->a2dp.mode == A2DP_MODE_SBC) {
+            setup_sbc(&u->a2dp, u->profile);
 
-        u->block_size =
-            ((u->link_mtu - sizeof(struct rtp_header) - sizeof(struct sbc_rtp_payload))
-             / u->a2dp.frame_length
-             * u->a2dp.codesize);
+            u->block_size =
+                ((u->link_mtu - sizeof(struct rtp_header) - sizeof(struct sbc_rtp_payload))
+                 / u->a2dp.frame_length
+                 * u->a2dp.codesize);
 
-        pa_log_info("SBC parameters:\n\tallocation=%u\n\tsubbands=%u\n\tblocks=%u\n\tbitpool=%u\n",
+            pa_log_info("SBC parameters:\n\tallocation=%u\n\tsubbands=%u\n\tblocks=%u\n\tbitpool=%u\n",
                     u->a2dp.sbc.allocation, u->a2dp.sbc.subbands, u->a2dp.sbc.blocks, u->a2dp.sbc.bitpool);
-    } else if (u->profile == PROFILE_A2DP_PASSTHROUGH) {
-
-        /* available payload per packet */
-        u->block_size = 1152*4; /* this is the size of an IEC61937 frame for MPEG layer 3 */
-        u->leftover_bytes=0;
-
+        } else if (u->a2dp.mode == A2DP_MODE_MPEG) {
+            /* available payload per packet */
+            u->block_size = 1152*4; /* this is the size of an IEC61937 frame for MPEG layer 3 */
+            u->leftover_bytes = 0;
+        }
     } else
         u->block_size = u->link_mtu;
 
@@ -1083,6 +1083,26 @@ static int stop_stream_fd(struct userdata *u) {
     return r;
 }
 
+static int close_stream(struct userdata *u) {
+    union {
+        struct bt_close_req close_req;
+        struct bt_close_rsp close_rsp;
+        bt_audio_error_t error;
+        uint8_t buf[BT_SUGGESTED_BUFFER_SIZE];
+    } msg;
+
+    memset(&msg, 0, sizeof(msg));
+    msg.close_req.h.type = BT_REQUEST;
+    msg.close_req.h.name = BT_CLOSE;
+    msg.close_req.h.length = sizeof(msg.close_req);
+
+    if (service_send(u, &msg.close_req.h) < 0)
+        return -1;
+
+    if (service_expect(u, &msg.close_rsp.h, sizeof(msg), BT_CLOSE, sizeof(msg.close_rsp)) < 0)
+        return -1;
+}
+
 static void bt_transport_release(struct userdata *u) {
     const char *accesstype = "rw";
     const pa_bluetooth_transport *t;
@@ -1162,6 +1182,56 @@ static int sink_process_msg(pa_msgobject *o, int code, void *data, int64_t offse
 
     switch (code) {
 
+        case PA_SINK_MESSAGE_ADD_INPUT: {
+            /* If you change anything here, make sure to change the
+             * sink input handling a few lines down at
+             * PA_SINK_MESSAGE_FINISH_MOVE, too. */
+
+            /* FIXME: returning failure here causes the server to just die.
+             * Would be better to be able to abort the stream gracefully. */
+
+            pa_sink_input *i = PA_SINK_INPUT(data);
+            a2dp_mode_t mode;
+
+            if (u->profile == PROFILE_A2DP) {
+                switch(i->format->encoding) {
+                    case PA_ENCODING_PCM:
+                        mode = A2DP_MODE_SBC;
+                        break;
+
+                    case PA_ENCODING_MPEG_IEC61937:
+                        pa_assert(u->a2dp.has_mpeg);
+                        mode = A2DP_MODE_MPEG;
+                        break;
+
+                    default:
+                        pa_assert_not_reached();
+                }
+
+                if (PA_UNLIKELY(mode != u->a2dp.mode)) {
+                    /* FIXME: Just suspend should suffice? This resets the smoother */
+                    if (stop_stream_fd(u) < 0 || close_stream(u) < 0) {
+                        failed = TRUE;
+                        break;
+                    }
+
+                    u->a2dp.mode = mode;
+                    if (set_conf(u) < 0) {
+                        failed = TRUE;
+                        break;
+                    }
+
+                    if (start_stream_fd(u) < 0) {
+                        failed = TRUE;
+                        break;
+                    }
+                }
+            }
+
+            break;
+        }
+
+
         case PA_SINK_MESSAGE_SET_STATE:
 
             switch ((pa_sink_state_t) PA_PTR_TO_UINT(data)) {
@@ -1766,7 +1836,6 @@ static int a2dp_passthrough_process_render(struct userdata *u) {
     int ret = 0;
 
     pa_assert(u);
-    pa_assert(u->profile == PROFILE_A2DP_PASSTHROUGH);
     pa_assert(u->sink);
 
     /* inits for output buffer */
@@ -2246,11 +2315,14 @@ static void thread_func(void *userdata) {
                         u->started_at = pa_rtclock_now();
 
                     if (u->profile == PROFILE_A2DP) {
-                        if ((n_written = a2dp_process_render(u)) < 0)
-                            goto fail;
-                    } else if (u->profile == PROFILE_A2DP_PASSTHROUGH) {
-                        if ((n_written = a2dp_passthrough_process_render(u)) < 0)
-                            goto fail;
+                        if (u->a2dp.mode == A2DP_MODE_SBC) {
+                            if ((n_written = a2dp_process_render(u)) < 0)
+                                goto fail;
+                        } else if (u->a2dp.mode == A2DP_MODE_MPEG) {
+                            if ((n_written = a2dp_passthrough_process_render(u)) < 0)
+                                goto fail;
+                        } else
+                            pa_assert_not_reached();
                     } else {
                         if ((n_written = hsp_process_render(u)) < 0)
                             goto fail;
@@ -2627,6 +2699,31 @@ static pa_hook_result_t source_state_changed_cb(pa_core *c, pa_source *s, struct
     return PA_HOOK_OK;
 }
 
+static pa_idxset* sink_get_formats(pa_sink *s) {
+    struct userdata *u;
+    pa_idxset *formats;
+    pa_format_info *f;
+
+    pa_assert(s);
+
+    formats = pa_idxset_new(NULL, NULL);
+
+    f = pa_format_info_new();
+    f->encoding = PA_ENCODING_PCM;
+    pa_idxset_put(formats, f, NULL);
+
+    u = (struct userdata *) s->userdata;
+
+    if (u->profile == PROFILE_A2DP && u->a2dp.has_mpeg) {
+        f = pa_format_info_new();
+        f->encoding = PA_ENCODING_MPEG_IEC61937;
+        /* FIXME: Populate supported rates, layers, ... */
+        pa_idxset_put(formats, f, NULL);
+    }
+
+    return formats;
+}
+
 /* Run from main thread */
 static int add_sink(struct userdata *u) {
     char *k;
@@ -2651,7 +2748,7 @@ static int add_sink(struct userdata *u) {
         data.driver = __FILE__;
         data.module = u->module;
         pa_sink_new_data_set_sample_spec(&data, &u->sample_spec);
-        pa_proplist_sets(data.proplist, "bluetooth.protocol", (u->profile == PROFILE_A2DP || u->profile == PROFILE_A2DP_PASSTHROUGH) ? "a2dp" : "sco");
+        pa_proplist_sets(data.proplist, "bluetooth.protocol", (u->profile == PROFILE_A2DP) ? "a2dp" : "sco");
         if (u->profile == PROFILE_HSP)
             pa_proplist_sets(data.proplist, PA_PROP_DEVICE_INTENDED_ROLES, "phone");
         data.card = u->card;
@@ -2674,13 +2771,13 @@ static int add_sink(struct userdata *u) {
 
         u->sink->userdata = u;
         u->sink->parent.process_msg = sink_process_msg;
+        u->sink->get_formats = sink_get_formats;
 
         pa_sink_set_max_request(u->sink, u->block_size);
         pa_sink_set_fixed_latency(u->sink,
-                                  (((u->profile == PROFILE_A2DP)||
-                                    (u->profile == PROFILE_A2DP_PASSTHROUGH)) ?
-                                   FIXED_LATENCY_PLAYBACK_A2DP : FIXED_LATENCY_PLAYBACK_HSP) +
-                                  pa_bytes_to_usec(u->block_size, &u->sample_spec));
+                                  ((u->profile == PROFILE_A2DP) ?
+                                   FIXED_LATENCY_PLAYBACK_A2DP :
+                                   FIXED_LATENCY_PLAYBACK_HSP) + pa_bytes_to_usec(u->block_size, &u->sample_spec));
     }
 
     if (u->profile == PROFILE_HSP) {
@@ -2692,9 +2789,6 @@ static int add_sink(struct userdata *u) {
         pa_xfree(k);
     }
 
-    if (u->profile == PROFILE_A2DP_PASSTHROUGH)
-        u->sink->flags |= PA_SINK_PASSTHROUGH;
-
     return 0;
 }
 
@@ -2982,6 +3076,11 @@ static int setup_bt(struct userdata *u) {
 
     pa_log_debug("Got device capabilities");
 
+    if (u->profile == PROFILE_A2DP) {
+        /* Connect for SBC to start with, switch later if required */
+        u->a2dp.mode = A2DP_MODE_SBC;
+    }
+
     if (set_conf(u) < 0)
         return -1;
 
@@ -3007,7 +3106,6 @@ static int init_profile(struct userdata *u) {
         return -1;
 
     if (u->profile == PROFILE_A2DP ||
-        u->profile == PROFILE_A2DP_PASSTHROUGH ||
         u->profile == PROFILE_HSP ||
         u->profile == PROFILE_HFGW)
         if (add_sink(u) < 0)
@@ -3317,19 +3415,6 @@ static int add_card(struct userdata *u, const pa_bluetooth_device *device) {
 
         pa_hashmap_put(data.profiles, p->name, p);
 
-        /* add passthrough profile */
-        p = pa_card_profile_new("passthrough", _("MP3 passthrough (A2DP)"), sizeof(enum profile));
-        p->priority = 5;
-        p->n_sinks = 1;
-        p->n_sources = 0;
-        p->max_sink_channels = 2;
-        p->max_source_channels = 0;
-
-        d = PA_CARD_PROFILE_DATA(p);
-        *d = PROFILE_A2DP_PASSTHROUGH;
-
-        pa_hashmap_put(data.profiles, p->name, p);
-
     }
 
     if (pa_bluetooth_uuid_has(device->uuids, A2DP_SOURCE_UUID)) {
@@ -3404,7 +3489,6 @@ static int add_card(struct userdata *u, const pa_bluetooth_device *device) {
 
     if ((device->headset_state < PA_BT_AUDIO_STATE_CONNECTED && *d == PROFILE_HSP) ||
         (device->audio_sink_state < PA_BT_AUDIO_STATE_CONNECTED && *d == PROFILE_A2DP) ||
-        (device->audio_sink_state < PA_BT_AUDIO_STATE_CONNECTED && *d == PROFILE_A2DP_PASSTHROUGH) ||
         (device->hfgw_state < PA_BT_AUDIO_STATE_CONNECTED && *d == PROFILE_HFGW)) {
         pa_log_warn("Default profile not connected, selecting off profile");
         u->card->active_profile = pa_hashmap_get(u->card->profiles, "off");
-- 
1.7.5.4



[Index of Archives]     [Linux Audio Users]     [AMD Graphics]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux