1b442c9e6SMilanka Ringwald /* 2b442c9e6SMilanka Ringwald * Copyright (C) 2016 BlueKitchen GmbH 3b442c9e6SMilanka Ringwald * 4b442c9e6SMilanka Ringwald * Redistribution and use in source and binary forms, with or without 5b442c9e6SMilanka Ringwald * modification, are permitted provided that the following conditions 6b442c9e6SMilanka Ringwald * are met: 7b442c9e6SMilanka Ringwald * 8b442c9e6SMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 9b442c9e6SMilanka Ringwald * notice, this list of conditions and the following disclaimer. 10b442c9e6SMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 11b442c9e6SMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 12b442c9e6SMilanka Ringwald * documentation and/or other materials provided with the distribution. 13b442c9e6SMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 14b442c9e6SMilanka Ringwald * contributors may be used to endorse or promote products derived 15b442c9e6SMilanka Ringwald * from this software without specific prior written permission. 16b442c9e6SMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 17b442c9e6SMilanka Ringwald * personal benefit and not for any commercial purpose or for 18b442c9e6SMilanka Ringwald * monetary gain. 19b442c9e6SMilanka Ringwald * 20b442c9e6SMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 21b442c9e6SMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22b442c9e6SMilanka Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 232fca4dadSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN 242fca4dadSMilanka Ringwald * GMBH OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 25b442c9e6SMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26b442c9e6SMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 27b442c9e6SMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28b442c9e6SMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29b442c9e6SMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 30b442c9e6SMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31b442c9e6SMilanka Ringwald * SUCH DAMAGE. 32b442c9e6SMilanka Ringwald * 33b442c9e6SMilanka Ringwald * Please inquire about commercial licensing options at 34b442c9e6SMilanka Ringwald * [email protected] 35b442c9e6SMilanka Ringwald * 36b442c9e6SMilanka Ringwald */ 37b442c9e6SMilanka Ringwald 38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "a2dp_sink.c" 39b442c9e6SMilanka Ringwald 40b442c9e6SMilanka Ringwald #include <stdint.h> 41b442c9e6SMilanka Ringwald #include <string.h> 42b442c9e6SMilanka Ringwald 4384e3541eSMilanka Ringwald #include "bluetooth_psm.h" 4484e3541eSMilanka Ringwald #include "bluetooth_sdp.h" 4584e3541eSMilanka Ringwald #include "btstack_debug.h" 4684e3541eSMilanka Ringwald #include "btstack_event.h" 474cb889a5SMilanka Ringwald #include "classic/a2dp_sink.h" 4884e3541eSMilanka Ringwald #include "classic/avdtp_sink.h" 4984e3541eSMilanka Ringwald #include "classic/avdtp_util.h" 5084e3541eSMilanka Ringwald #include "classic/sdp_util.h" 51b442c9e6SMilanka Ringwald 52137e2954SMatthias Ringwald static const char * a2dp_sink_default_service_name = "BTstack A2DP Sink Service"; 53137e2954SMatthias Ringwald static const char * a2dp_sink_default_service_provider_name = "BTstack A2DP Sink Service Provider"; 54b442c9e6SMilanka Ringwald 55ce93555dSMatthias Ringwald static uint16_t a2dp_sink_cid; 56137e2954SMatthias Ringwald static bool a2dp_sink_stream_endpoint_configured = false; 57892a0b92SMatthias Ringwald 5838106e95SMatthias Ringwald static btstack_packet_handler_t a2dp_sink_packet_handler_user; 59a95794ceSMatthias Ringwald static uint8_t (*a2dp_sink_media_config_validator)(const avdtp_stream_endpoint_t * stream_endpoint, const uint8_t * event, uint16_t size); 6038106e95SMatthias Ringwald 6138106e95SMatthias Ringwald static void a2dp_sink_packet_handler_internal(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size); 62b442c9e6SMilanka Ringwald 63b442c9e6SMilanka Ringwald void a2dp_sink_create_sdp_record(uint8_t * service, uint32_t service_record_handle, uint16_t supported_features, const char * service_name, const char * service_provider_name){ 64b442c9e6SMilanka Ringwald uint8_t* attribute; 65b442c9e6SMilanka Ringwald de_create_sequence(service); 66b442c9e6SMilanka Ringwald 67b442c9e6SMilanka Ringwald // 0x0000 "Service Record Handle" 68b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, BLUETOOTH_ATTRIBUTE_SERVICE_RECORD_HANDLE); 69b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_32, service_record_handle); 70b442c9e6SMilanka Ringwald 71b442c9e6SMilanka Ringwald // 0x0001 "Service Class ID List" 72b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, BLUETOOTH_ATTRIBUTE_SERVICE_CLASS_ID_LIST); 73b442c9e6SMilanka Ringwald attribute = de_push_sequence(service); 74b442c9e6SMilanka Ringwald { 75b442c9e6SMilanka Ringwald de_add_number(attribute, DE_UUID, DE_SIZE_16, BLUETOOTH_SERVICE_CLASS_AUDIO_SINK); 76b442c9e6SMilanka Ringwald } 77b442c9e6SMilanka Ringwald de_pop_sequence(service, attribute); 78b442c9e6SMilanka Ringwald 79b442c9e6SMilanka Ringwald // 0x0004 "Protocol Descriptor List" 80b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, BLUETOOTH_ATTRIBUTE_PROTOCOL_DESCRIPTOR_LIST); 81b442c9e6SMilanka Ringwald attribute = de_push_sequence(service); 82b442c9e6SMilanka Ringwald { 83b442c9e6SMilanka Ringwald uint8_t* l2cpProtocol = de_push_sequence(attribute); 84b442c9e6SMilanka Ringwald { 85b442c9e6SMilanka Ringwald de_add_number(l2cpProtocol, DE_UUID, DE_SIZE_16, BLUETOOTH_PROTOCOL_L2CAP); 8684e3541eSMilanka Ringwald de_add_number(l2cpProtocol, DE_UINT, DE_SIZE_16, BLUETOOTH_PSM_AVDTP); 87b442c9e6SMilanka Ringwald } 88b442c9e6SMilanka Ringwald de_pop_sequence(attribute, l2cpProtocol); 89b442c9e6SMilanka Ringwald 90b442c9e6SMilanka Ringwald uint8_t* avProtocol = de_push_sequence(attribute); 91b442c9e6SMilanka Ringwald { 92b442c9e6SMilanka Ringwald de_add_number(avProtocol, DE_UUID, DE_SIZE_16, BLUETOOTH_PROTOCOL_AVDTP); // avProtocol_service 93b442c9e6SMilanka Ringwald de_add_number(avProtocol, DE_UINT, DE_SIZE_16, 0x0103); // version 94b442c9e6SMilanka Ringwald } 95b442c9e6SMilanka Ringwald de_pop_sequence(attribute, avProtocol); 96b442c9e6SMilanka Ringwald } 97b442c9e6SMilanka Ringwald de_pop_sequence(service, attribute); 98b442c9e6SMilanka Ringwald 99b442c9e6SMilanka Ringwald // 0x0005 "Public Browse Group" 100b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, BLUETOOTH_ATTRIBUTE_BROWSE_GROUP_LIST); // public browse group 101b442c9e6SMilanka Ringwald attribute = de_push_sequence(service); 102b442c9e6SMilanka Ringwald { 103b442c9e6SMilanka Ringwald de_add_number(attribute, DE_UUID, DE_SIZE_16, BLUETOOTH_ATTRIBUTE_PUBLIC_BROWSE_ROOT); 104b442c9e6SMilanka Ringwald } 105b442c9e6SMilanka Ringwald de_pop_sequence(service, attribute); 106b442c9e6SMilanka Ringwald 107b442c9e6SMilanka Ringwald // 0x0009 "Bluetooth Profile Descriptor List" 108b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, BLUETOOTH_ATTRIBUTE_BLUETOOTH_PROFILE_DESCRIPTOR_LIST); 109b442c9e6SMilanka Ringwald attribute = de_push_sequence(service); 110b442c9e6SMilanka Ringwald { 111b442c9e6SMilanka Ringwald uint8_t *a2dProfile = de_push_sequence(attribute); 112b442c9e6SMilanka Ringwald { 113b442c9e6SMilanka Ringwald de_add_number(a2dProfile, DE_UUID, DE_SIZE_16, BLUETOOTH_SERVICE_CLASS_ADVANCED_AUDIO_DISTRIBUTION); 114b442c9e6SMilanka Ringwald de_add_number(a2dProfile, DE_UINT, DE_SIZE_16, 0x0103); 115b442c9e6SMilanka Ringwald } 116b442c9e6SMilanka Ringwald de_pop_sequence(attribute, a2dProfile); 117b442c9e6SMilanka Ringwald } 118b442c9e6SMilanka Ringwald de_pop_sequence(service, attribute); 119b442c9e6SMilanka Ringwald 120b442c9e6SMilanka Ringwald 121b442c9e6SMilanka Ringwald // 0x0100 "Service Name" 122b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, 0x0100); 123b442c9e6SMilanka Ringwald if (service_name){ 124b442c9e6SMilanka Ringwald de_add_data(service, DE_STRING, strlen(service_name), (uint8_t *) service_name); 125b442c9e6SMilanka Ringwald } else { 126137e2954SMatthias Ringwald de_add_data(service, DE_STRING, strlen(a2dp_sink_default_service_name), (uint8_t *) a2dp_sink_default_service_name); 127b442c9e6SMilanka Ringwald } 128b442c9e6SMilanka Ringwald 129b442c9e6SMilanka Ringwald // 0x0100 "Provider Name" 130b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, 0x0102); 131b442c9e6SMilanka Ringwald if (service_provider_name){ 132b442c9e6SMilanka Ringwald de_add_data(service, DE_STRING, strlen(service_provider_name), (uint8_t *) service_provider_name); 133b442c9e6SMilanka Ringwald } else { 134137e2954SMatthias Ringwald de_add_data(service, DE_STRING, strlen(a2dp_sink_default_service_provider_name), (uint8_t *) a2dp_sink_default_service_provider_name); 135b442c9e6SMilanka Ringwald } 136b442c9e6SMilanka Ringwald 137b442c9e6SMilanka Ringwald // 0x0311 "Supported Features" 138b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, 0x0311); 139b442c9e6SMilanka Ringwald de_add_number(service, DE_UINT, DE_SIZE_16, supported_features); 140b442c9e6SMilanka Ringwald } 1417050d2caSMilanka Ringwald 1427050d2caSMilanka Ringwald void a2dp_sink_register_packet_handler(btstack_packet_handler_t callback){ 143f1042e9dSMilanka Ringwald // avdtp_sink_register_packet_handler(callback); 144f1042e9dSMilanka Ringwald // return; 1457050d2caSMilanka Ringwald if (callback == NULL){ 1467050d2caSMilanka Ringwald log_error("a2dp_sink_register_packet_handler called with NULL callback"); 1477050d2caSMilanka Ringwald return; 1487050d2caSMilanka Ringwald } 14938106e95SMatthias Ringwald avdtp_sink_register_packet_handler(&a2dp_sink_packet_handler_internal); 15038106e95SMatthias Ringwald a2dp_sink_packet_handler_user = callback; 1517050d2caSMilanka Ringwald } 1527050d2caSMilanka Ringwald 153fd58c900SMilanka Ringwald void a2dp_sink_register_media_handler(void (*callback)(uint8_t local_seid, uint8_t *packet, uint16_t size)){ 1547050d2caSMilanka Ringwald if (callback == NULL){ 1557050d2caSMilanka Ringwald log_error("a2dp_sink_register_media_handler called with NULL callback"); 1567050d2caSMilanka Ringwald return; 1577050d2caSMilanka Ringwald } 1587050d2caSMilanka Ringwald avdtp_sink_register_media_handler(callback); 1597050d2caSMilanka Ringwald } 1607050d2caSMilanka Ringwald 1617050d2caSMilanka Ringwald void a2dp_sink_init(void){ 16277092f3eSMatthias Ringwald avdtp_sink_init(); 1637050d2caSMilanka Ringwald } 1647050d2caSMilanka Ringwald 1657569dc61SMatthias Ringwald void a2dp_sink_deinit(void){ 1667569dc61SMatthias Ringwald avdtp_sink_deinit(); 1677569dc61SMatthias Ringwald 1687569dc61SMatthias Ringwald a2dp_sink_cid = 0; 169137e2954SMatthias Ringwald a2dp_sink_packet_handler_user = NULL; 170137e2954SMatthias Ringwald a2dp_sink_media_config_validator = NULL; 171137e2954SMatthias Ringwald a2dp_sink_stream_endpoint_configured = false; 1727569dc61SMatthias Ringwald } 1737569dc61SMatthias Ringwald 1748b94010eSMilanka Ringwald avdtp_stream_endpoint_t * a2dp_sink_create_stream_endpoint(avdtp_media_type_t media_type, avdtp_media_codec_type_t media_codec_type, 1753e6cf581SMatthias Ringwald const uint8_t *codec_capabilities, uint16_t codec_capabilities_len, 17682767773SMatthias Ringwald uint8_t * codec_configuration, uint16_t codec_configuration_len){ 1777050d2caSMilanka Ringwald avdtp_stream_endpoint_t * local_stream_endpoint = avdtp_sink_create_stream_endpoint(AVDTP_SINK, media_type); 1784567cc17SMilanka Ringwald if (!local_stream_endpoint){ 1798b94010eSMilanka Ringwald return NULL; 1804567cc17SMilanka Ringwald } 1817050d2caSMilanka Ringwald avdtp_sink_register_media_transport_category(avdtp_stream_endpoint_seid(local_stream_endpoint)); 1827050d2caSMilanka Ringwald avdtp_sink_register_media_codec_category(avdtp_stream_endpoint_seid(local_stream_endpoint), media_type, media_codec_type, 1837050d2caSMilanka Ringwald codec_capabilities, codec_capabilities_len); 1848b94010eSMilanka Ringwald avdtp_sink_register_delay_reporting_category(avdtp_stream_endpoint_seid(local_stream_endpoint)); 18582767773SMatthias Ringwald 18682767773SMatthias Ringwald // store user codec configuration buffer 18782767773SMatthias Ringwald local_stream_endpoint->media_codec_configuration_info = codec_configuration; 18882767773SMatthias Ringwald local_stream_endpoint->media_codec_configuration_len = codec_configuration_len; 18982767773SMatthias Ringwald 1908b94010eSMilanka Ringwald return local_stream_endpoint; 1917050d2caSMilanka Ringwald } 1927050d2caSMilanka Ringwald 19317ddf501SMatthias Ringwald void a2dp_sink_finalize_stream_endpoint(avdtp_stream_endpoint_t * stream_endpoint){ 19417ddf501SMatthias Ringwald avdtp_sink_finalize_stream_endpoint(stream_endpoint); 19517ddf501SMatthias Ringwald } 19617ddf501SMatthias Ringwald 1974ccacc40SMilanka Ringwald uint8_t a2dp_sink_establish_stream(bd_addr_t bd_addr, uint8_t local_seid, uint16_t * avdtp_cid){ 19882767773SMatthias Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_get_stream_endpoint_for_seid(local_seid); 19982767773SMatthias Ringwald if (stream_endpoint == NULL){ 200326e5cedSMilanka Ringwald log_info("No local_stream_endpoint for seid %d", local_seid); 201ce93555dSMatthias Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 2027050d2caSMilanka Ringwald } 203*2e3fcb13SMatthias Ringwald uint16_t outgoing_cid; 204*2e3fcb13SMatthias Ringwald 205*2e3fcb13SMatthias Ringwald uint8_t status = avdtp_sink_connect(bd_addr, &outgoing_cid); 206*2e3fcb13SMatthias Ringwald if (status != ERROR_CODE_SUCCESS){ 207*2e3fcb13SMatthias Ringwald return status; 208*2e3fcb13SMatthias Ringwald } 209*2e3fcb13SMatthias Ringwald 210*2e3fcb13SMatthias Ringwald avdtp_connection_t * connection = avdtp_get_connection_for_avdtp_cid(outgoing_cid); 211*2e3fcb13SMatthias Ringwald btstack_assert(connection != NULL); 212*2e3fcb13SMatthias Ringwald 213*2e3fcb13SMatthias Ringwald // setup state 214*2e3fcb13SMatthias Ringwald connection->a2dp_sink_outgoing_active = true; 215*2e3fcb13SMatthias Ringwald *avdtp_cid = outgoing_cid; 216*2e3fcb13SMatthias Ringwald 217*2e3fcb13SMatthias Ringwald return ERROR_CODE_SUCCESS; 2187050d2caSMilanka Ringwald } 2197050d2caSMilanka Ringwald 22048ce193cSMilanka Ringwald #ifdef ENABLE_AVDTP_ACCEPTOR_EXPLICIT_START_STREAM_CONFIRMATION 22148ce193cSMilanka Ringwald uint8_t a2dp_sink_start_stream_accept(uint16_t a2dp_cid, uint8_t local_seid){ 22248ce193cSMilanka Ringwald return avdtp_start_stream_accept(a2dp_cid, local_seid); 22348ce193cSMilanka Ringwald } 22448ce193cSMilanka Ringwald 22548ce193cSMilanka Ringwald uint8_t a2dp_sink_start_stream_reject(uint16_t a2dp_cid, uint8_t local_seid){ 22648ce193cSMilanka Ringwald return avdtp_start_stream_reject(a2dp_cid, local_seid); 22748ce193cSMilanka Ringwald } 22848ce193cSMilanka Ringwald #endif 22948ce193cSMilanka Ringwald 2307050d2caSMilanka Ringwald void a2dp_sink_disconnect(uint16_t a2dp_cid){ 231b401ff59SMilanka Ringwald avdtp_disconnect(a2dp_cid); 2327050d2caSMilanka Ringwald } 2337050d2caSMilanka Ringwald 23438106e95SMatthias Ringwald static void a2dp_sink_packet_handler_internal(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 2357050d2caSMilanka Ringwald UNUSED(channel); 2367050d2caSMilanka Ringwald UNUSED(size); 237426010aaSMilanka Ringwald 238*2e3fcb13SMatthias Ringwald uint16_t cid; 239*2e3fcb13SMatthias Ringwald avdtp_connection_t * connection; 240e0d13a19SMilanka Ringwald uint8_t status; 241426010aaSMilanka Ringwald uint8_t local_seid; 24226f6cd9dSMilanka Ringwald uint8_t signal_identifier; 2436c5b303cSMatthias Ringwald bool reconfigure; 24498137b16SMatthias Ringwald uint8_t subevent_id; 2457050d2caSMilanka Ringwald 24634b22aacSMilanka Ringwald if (packet_type != HCI_EVENT_PACKET) return; 24734b22aacSMilanka Ringwald if (hci_event_packet_get_type(packet) != HCI_EVENT_AVDTP_META) return; 24834b22aacSMilanka Ringwald 2497050d2caSMilanka Ringwald switch (packet[2]){ 2507050d2caSMilanka Ringwald case AVDTP_SUBEVENT_SIGNALING_CONNECTION_ESTABLISHED: 251137e2954SMatthias Ringwald if (a2dp_sink_stream_endpoint_configured) return; 252c58b84e8SMilanka Ringwald 253*2e3fcb13SMatthias Ringwald cid = avdtp_subevent_signaling_connection_established_get_avdtp_cid(packet); 254*2e3fcb13SMatthias Ringwald connection = avdtp_get_connection_for_avdtp_cid(cid); 255*2e3fcb13SMatthias Ringwald btstack_assert(connection != NULL); 256*2e3fcb13SMatthias Ringwald 25734b22aacSMilanka Ringwald status = avdtp_subevent_signaling_connection_established_get_status(packet); 25842230d27SMilanka Ringwald if (status != ERROR_CODE_SUCCESS){ 259*2e3fcb13SMatthias Ringwald // notify about connection error only if we're initiator 260*2e3fcb13SMatthias Ringwald if (connection->a2dp_sink_outgoing_active) { 261ce93555dSMatthias Ringwald log_info("A2DP sink signaling connection failed status %d", status); 262*2e3fcb13SMatthias Ringwald connection->a2dp_sink_outgoing_active = false; 263*2e3fcb13SMatthias Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_SIGNALING_CONNECTION_ESTABLISHED); 264*2e3fcb13SMatthias Ringwald } 2654ccacc40SMilanka Ringwald } 26626f6cd9dSMilanka Ringwald 26722bed8e1SMilanka Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_SIGNALING_CONNECTION_ESTABLISHED); 268e0d13a19SMilanka Ringwald log_info("A2DP sink signaling connection established avdtp_cid 0x%02x", avdtp_subevent_signaling_connection_established_get_avdtp_cid(packet)); 269f1042e9dSMilanka Ringwald break; 270f1042e9dSMilanka Ringwald 2716c5b303cSMatthias Ringwald case AVDTP_SUBEVENT_SIGNALING_MEDIA_CODEC_SBC_CONFIGURATION: 2726c5b303cSMatthias Ringwald case AVDTP_SUBEVENT_SIGNALING_MEDIA_CODEC_MPEG_AUDIO_CONFIGURATION: 2736c5b303cSMatthias Ringwald case AVDTP_SUBEVENT_SIGNALING_MEDIA_CODEC_MPEG_AAC_CONFIGURATION: 2746c5b303cSMatthias Ringwald case AVDTP_SUBEVENT_SIGNALING_MEDIA_CODEC_ATRAC_CONFIGURATION: 275f1042e9dSMilanka Ringwald case AVDTP_SUBEVENT_SIGNALING_MEDIA_CODEC_OTHER_CONFIGURATION: 2766c5b303cSMatthias Ringwald reconfigure = avdtp_subevent_signaling_media_codec_other_configuration_get_reconfigure(packet) != 0; 2776c5b303cSMatthias Ringwald // accept configure if not configured and reconfigure if already configured 278137e2954SMatthias Ringwald if (a2dp_sink_stream_endpoint_configured != reconfigure) break; 279137e2954SMatthias Ringwald a2dp_sink_stream_endpoint_configured = true; 280ce93555dSMatthias Ringwald a2dp_sink_cid = avdtp_subevent_signaling_media_codec_other_capability_get_avdtp_cid(packet); 28198137b16SMatthias Ringwald subevent_id = a2dp_subevent_id_for_avdtp_subevent_id(packet[2]); 28298137b16SMatthias Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, subevent_id); 283f1042e9dSMilanka Ringwald break; 284f1042e9dSMilanka Ringwald 2857050d2caSMilanka Ringwald case AVDTP_SUBEVENT_STREAMING_CONNECTION_ESTABLISHED: 286137e2954SMatthias Ringwald if (a2dp_sink_stream_endpoint_configured == false) break; 287e0d13a19SMilanka Ringwald if (a2dp_sink_cid != avdtp_subevent_streaming_connection_established_get_avdtp_cid(packet)) break; 28834b22aacSMilanka Ringwald 289*2e3fcb13SMatthias Ringwald connection = avdtp_get_connection_for_avdtp_cid(a2dp_sink_cid); 290*2e3fcb13SMatthias Ringwald btstack_assert(connection != NULL); 291*2e3fcb13SMatthias Ringwald 292892a0b92SMatthias Ringwald // about to notify client 293*2e3fcb13SMatthias Ringwald connection->a2dp_sink_outgoing_active = false; 294426010aaSMilanka Ringwald status = avdtp_subevent_streaming_connection_established_get_status(packet); 29526f6cd9dSMilanka Ringwald if (status != ERROR_CODE_SUCCESS){ 296e0d13a19SMilanka Ringwald log_info("A2DP sink streaming connection could not be established, avdtp_cid 0x%02x, status 0x%02x ---", a2dp_sink_cid, status); 29722bed8e1SMilanka Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_STREAM_ESTABLISHED); 29834b22aacSMilanka Ringwald break; 299f1042e9dSMilanka Ringwald } 30026f6cd9dSMilanka Ringwald 301e0d13a19SMilanka Ringwald log_info("A2DP streaming connection established --- avdtp_cid 0x%02x, local seid %d, remote seid %d", a2dp_sink_cid, 302e0d13a19SMilanka Ringwald avdtp_subevent_streaming_connection_established_get_local_seid(packet), 303e0d13a19SMilanka Ringwald avdtp_subevent_streaming_connection_established_get_remote_seid(packet)); 304e0d13a19SMilanka Ringwald 30522bed8e1SMilanka Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_STREAM_ESTABLISHED); 3067050d2caSMilanka Ringwald break; 307f1042e9dSMilanka Ringwald 3087050d2caSMilanka Ringwald case AVDTP_SUBEVENT_SIGNALING_ACCEPT: 309137e2954SMatthias Ringwald if (a2dp_sink_stream_endpoint_configured == false) break; 310e0d13a19SMilanka Ringwald if (a2dp_sink_cid != avdtp_subevent_signaling_accept_get_avdtp_cid(packet)) break; 311ce93555dSMatthias Ringwald 312ce93555dSMatthias Ringwald signal_identifier = avdtp_subevent_signaling_accept_get_signal_identifier(packet); 31326f6cd9dSMilanka Ringwald local_seid = avdtp_subevent_signaling_accept_get_local_seid(packet); 31426f6cd9dSMilanka Ringwald 31526f6cd9dSMilanka Ringwald switch (signal_identifier){ 31626f6cd9dSMilanka Ringwald case AVDTP_SI_START: 317e0d13a19SMilanka Ringwald a2dp_emit_stream_event(a2dp_sink_packet_handler_user, a2dp_sink_cid, local_seid, A2DP_SUBEVENT_STREAM_STARTED); 31826f6cd9dSMilanka Ringwald break; 31926f6cd9dSMilanka Ringwald case AVDTP_SI_SUSPEND: 320e0d13a19SMilanka Ringwald a2dp_emit_stream_event(a2dp_sink_packet_handler_user, a2dp_sink_cid, local_seid, A2DP_SUBEVENT_STREAM_SUSPENDED); 32126f6cd9dSMilanka Ringwald break; 32226f6cd9dSMilanka Ringwald case AVDTP_SI_ABORT: 32326f6cd9dSMilanka Ringwald case AVDTP_SI_CLOSE: 324e0d13a19SMilanka Ringwald a2dp_emit_stream_event(a2dp_sink_packet_handler_user, a2dp_sink_cid, local_seid, A2DP_SUBEVENT_STREAM_STOPPED); 32526f6cd9dSMilanka Ringwald break; 32648ce193cSMilanka Ringwald #ifdef ENABLE_AVDTP_ACCEPTOR_EXPLICIT_START_STREAM_CONFIRMATION 32748ce193cSMilanka Ringwald case AVDTP_SI_ACCEPT_START: 32848ce193cSMilanka Ringwald a2dp_emit_stream_event(a2dp_sink_packet_handler_user, a2dp_sink_cid, local_seid, A2DP_SUBEVENT_START_STREAM_REQUESTED); 32948ce193cSMilanka Ringwald break; 33048ce193cSMilanka Ringwald #endif 33126f6cd9dSMilanka Ringwald default: 33226f6cd9dSMilanka Ringwald break; 33326f6cd9dSMilanka Ringwald } 334f1042e9dSMilanka Ringwald break; 335f1042e9dSMilanka Ringwald 336f1042e9dSMilanka Ringwald case AVDTP_SUBEVENT_SIGNALING_REJECT: 337137e2954SMatthias Ringwald if (a2dp_sink_stream_endpoint_configured == false) break; 338e0d13a19SMilanka Ringwald if (a2dp_sink_cid != avdtp_subevent_signaling_reject_get_avdtp_cid(packet)) break; 33942230d27SMilanka Ringwald 340426010aaSMilanka Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_COMMAND_REJECTED); 341f1042e9dSMilanka Ringwald break; 34242230d27SMilanka Ringwald 34342230d27SMilanka Ringwald case AVDTP_SUBEVENT_SIGNALING_GENERAL_REJECT: 344137e2954SMatthias Ringwald if (a2dp_sink_stream_endpoint_configured == false) break; 345e0d13a19SMilanka Ringwald if (a2dp_sink_cid != avdtp_subevent_signaling_general_reject_get_avdtp_cid(packet)) break; 34642230d27SMilanka Ringwald 34742230d27SMilanka Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_COMMAND_REJECTED); 34842230d27SMilanka Ringwald break; 34942230d27SMilanka Ringwald 35034b22aacSMilanka Ringwald case AVDTP_SUBEVENT_STREAMING_CONNECTION_RELEASED: 351137e2954SMatthias Ringwald if (a2dp_sink_stream_endpoint_configured == false) break; 352e0d13a19SMilanka Ringwald if (a2dp_sink_cid != avdtp_subevent_streaming_connection_released_get_avdtp_cid(packet)) break; 35342230d27SMilanka Ringwald 354137e2954SMatthias Ringwald a2dp_sink_stream_endpoint_configured = false; 355b6f261d1SMatthias Ringwald 356f6906c74SMilanka Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_STREAM_RELEASED); 357f1042e9dSMilanka Ringwald break; 35842230d27SMilanka Ringwald 35934b22aacSMilanka Ringwald case AVDTP_SUBEVENT_SIGNALING_CONNECTION_RELEASED: 360*2e3fcb13SMatthias Ringwald cid = avdtp_subevent_signaling_connection_released_get_avdtp_cid(packet); 361*2e3fcb13SMatthias Ringwald connection = avdtp_get_connection_for_avdtp_cid(cid); 362*2e3fcb13SMatthias Ringwald btstack_assert(connection != NULL); 363*2e3fcb13SMatthias Ringwald 364*2e3fcb13SMatthias Ringwald connection->a2dp_sink_outgoing_active = false; 365*2e3fcb13SMatthias Ringwald 366*2e3fcb13SMatthias Ringwald if (a2dp_sink_cid != cid) break; 367892a0b92SMatthias Ringwald 368137e2954SMatthias Ringwald a2dp_sink_stream_endpoint_configured = false; 369892a0b92SMatthias Ringwald 370f6906c74SMilanka Ringwald a2dp_replace_subevent_id_and_emit_cmd(a2dp_sink_packet_handler_user, packet, size, A2DP_SUBEVENT_SIGNALING_CONNECTION_RELEASED); 371f1042e9dSMilanka Ringwald break; 372f1042e9dSMilanka Ringwald default: 3737050d2caSMilanka Ringwald break; 3747050d2caSMilanka Ringwald } 37534b22aacSMilanka Ringwald 3767050d2caSMilanka Ringwald } 3777050d2caSMilanka Ringwald 378a95794ceSMatthias Ringwald static uint8_t a2dp_sink_media_config_validator_callback(const avdtp_stream_endpoint_t * stream_endpoint, const uint8_t * event, uint16_t size){ 379a95794ceSMatthias Ringwald uint8_t error = 0; 38059dedcb9SMatthias Ringwald if (a2dp_sink_media_config_validator != NULL) { 381a95794ceSMatthias Ringwald // update subevent id and call validator 382a95794ceSMatthias Ringwald uint8_t avdtp_subevent_id = event[2]; 383a95794ceSMatthias Ringwald uint8_t a2dp_subevent_id = a2dp_subevent_id_for_avdtp_subevent_id(avdtp_subevent_id); 384a95794ceSMatthias Ringwald uint8_t * subevent_field = (uint8_t *) &event[2]; 385a95794ceSMatthias Ringwald *subevent_field = a2dp_subevent_id; 386a95794ceSMatthias Ringwald error = (*a2dp_sink_media_config_validator)(stream_endpoint, event, size); 387a95794ceSMatthias Ringwald *subevent_field = avdtp_subevent_id; 388a95794ceSMatthias Ringwald } 389a95794ceSMatthias Ringwald return error; 390a95794ceSMatthias Ringwald } 391a95794ceSMatthias Ringwald 392a95794ceSMatthias Ringwald void a2dp_sink_register_media_config_validator(uint8_t (*callback)(const avdtp_stream_endpoint_t * stream_endpoint, const uint8_t * event, uint16_t size)){ 393a95794ceSMatthias Ringwald a2dp_sink_media_config_validator = callback; 394a95794ceSMatthias Ringwald avdtp_sink_register_media_config_validator(&a2dp_sink_media_config_validator_callback); 395a95794ceSMatthias Ringwald } 396