1747ec646SMilanka Ringwald /* 2747ec646SMilanka Ringwald * Copyright (C) 2016 BlueKitchen GmbH 3747ec646SMilanka Ringwald * 4747ec646SMilanka Ringwald * Redistribution and use in source and binary forms, with or without 5747ec646SMilanka Ringwald * modification, are permitted provided that the following conditions 6747ec646SMilanka Ringwald * are met: 7747ec646SMilanka Ringwald * 8747ec646SMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 9747ec646SMilanka Ringwald * notice, this list of conditions and the following disclaimer. 10747ec646SMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 11747ec646SMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 12747ec646SMilanka Ringwald * documentation and/or other materials provided with the distribution. 13747ec646SMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 14747ec646SMilanka Ringwald * contributors may be used to endorse or promote products derived 15747ec646SMilanka Ringwald * from this software without specific prior written permission. 16747ec646SMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 17747ec646SMilanka Ringwald * personal benefit and not for any commercial purpose or for 18747ec646SMilanka Ringwald * monetary gain. 19747ec646SMilanka Ringwald * 20747ec646SMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 21747ec646SMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22747ec646SMilanka Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 23747ec646SMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 24747ec646SMilanka Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 25747ec646SMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26747ec646SMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 27747ec646SMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28747ec646SMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29747ec646SMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 30747ec646SMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31747ec646SMilanka Ringwald * SUCH DAMAGE. 32747ec646SMilanka Ringwald * 33747ec646SMilanka Ringwald * Please inquire about commercial licensing options at 34747ec646SMilanka Ringwald * [email protected] 35747ec646SMilanka Ringwald * 36747ec646SMilanka Ringwald */ 37747ec646SMilanka Ringwald 38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "avdtp.c" 39ab2c6ae4SMatthias Ringwald 40747ec646SMilanka Ringwald 41747ec646SMilanka Ringwald #include <stdint.h> 42747ec646SMilanka Ringwald #include <string.h> 43747ec646SMilanka Ringwald 4484e3541eSMilanka Ringwald #include "bluetooth_psm.h" 4584e3541eSMilanka Ringwald #include "bluetooth_sdp.h" 4684e3541eSMilanka Ringwald #include "btstack_debug.h" 4784e3541eSMilanka Ringwald #include "btstack_event.h" 4884e3541eSMilanka Ringwald #include "btstack_memory.h" 494cb889a5SMilanka Ringwald #include "classic/avdtp.h" 504cb889a5SMilanka Ringwald #include "classic/avdtp_acceptor.h" 514cb889a5SMilanka Ringwald #include "classic/avdtp_initiator.h" 5284e3541eSMilanka Ringwald #include "classic/avdtp_util.h" 5384e3541eSMilanka Ringwald #include "classic/sdp_client.h" 5484e3541eSMilanka Ringwald #include "classic/sdp_util.h" 55747ec646SMilanka Ringwald 56d8b859a2SMilanka Ringwald #define CONFIGURATION_TIMEOUT_MS 300 57d8b859a2SMilanka Ringwald 58692c0605SMilanka Ringwald static int record_id = -1; 59692c0605SMilanka Ringwald static uint8_t attribute_value[1000]; 60692c0605SMilanka Ringwald static const unsigned int attribute_value_buffer_size = sizeof(attribute_value); 61747ec646SMilanka Ringwald 622f6083d0SMilanka Ringwald // typedef struct { 632f6083d0SMilanka Ringwald // btstack_linked_list_t * avdtp_connections; 642f6083d0SMilanka Ringwald // avdtp_connection_t * connection; 652f6083d0SMilanka Ringwald // btstack_packet_handler_t avdtp_callback; 662f6083d0SMilanka Ringwald // avdtp_sep_type_t query_role; 672f6083d0SMilanka Ringwald // btstack_packet_handler_t packet_handler; 682f6083d0SMilanka Ringwald // uint16_t avdtp_l2cap_psm; 692f6083d0SMilanka Ringwald // uint16_t avdtp_version; 702f6083d0SMilanka Ringwald // uint8_t role_supported; 712f6083d0SMilanka Ringwald // } avdtp_sdp_query_context_t; 72692c0605SMilanka Ringwald 732f6083d0SMilanka Ringwald static avdtp_context_t * sdp_query_context; 74af121d54SMilanka Ringwald static uint16_t avdtp_cid_counter = 0; 75747ec646SMilanka Ringwald 76fd58c900SMilanka Ringwald static void (*handle_media_data)(uint8_t local_seid, uint8_t *packet, uint16_t size); 77692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size); 78692c0605SMilanka Ringwald 79d8b859a2SMilanka Ringwald void avdtp_configuration_timeout_handler(btstack_timer_source_t * timer){ 80d8b859a2SMilanka Ringwald avdtp_connection_t * connection = (avdtp_connection_t *) btstack_run_loop_get_timer_context(timer); 81d8b859a2SMilanka Ringwald if (!connection){ 82d8b859a2SMilanka Ringwald log_error("Context of avdtp_configuration_timeout_handler is NULL"); 83d8b859a2SMilanka Ringwald return; 84d8b859a2SMilanka Ringwald } 85bdbc3ef6SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t*) connection->active_stream_endpoint; 86bdbc3ef6SMilanka Ringwald if (!stream_endpoint) { 87bdbc3ef6SMilanka Ringwald log_error("avdtp_configuration_timeout_handler: no initiator stream endpoint for seid %d", connection->local_seid); 88bdbc3ef6SMilanka Ringwald return; 89bdbc3ef6SMilanka Ringwald } 90bdbc3ef6SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_CONFIGURATION_SUBSTATEMACHINE) return; 91bdbc3ef6SMilanka Ringwald connection->initiator_transaction_label++; 92bdbc3ef6SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION; 93d8b859a2SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 94d8b859a2SMilanka Ringwald } 95d8b859a2SMilanka Ringwald 96d8b859a2SMilanka Ringwald void avdtp_configuration_timer_start(avdtp_connection_t * connection){ 97bdbc3ef6SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t*) connection->active_stream_endpoint; 98bdbc3ef6SMilanka Ringwald if (!stream_endpoint) { 99bdbc3ef6SMilanka Ringwald log_error("avdtp_configuration_timeout_handler: no initiator stream endpoint for seid %d", connection->local_seid); 100bdbc3ef6SMilanka Ringwald return; 101bdbc3ef6SMilanka Ringwald } 102bdbc3ef6SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_CONFIGURATION_SUBSTATEMACHINE) return; 103bdbc3ef6SMilanka Ringwald 104d8b859a2SMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 105d8b859a2SMilanka Ringwald btstack_run_loop_set_timer_handler(&connection->configuration_timer, avdtp_configuration_timeout_handler); 106d8b859a2SMilanka Ringwald btstack_run_loop_set_timer_context(&connection->configuration_timer, connection); 107d8b859a2SMilanka Ringwald btstack_run_loop_set_timer(&connection->configuration_timer, CONFIGURATION_TIMEOUT_MS); 108d8b859a2SMilanka Ringwald btstack_run_loop_add_timer(&connection->configuration_timer); 109d8b859a2SMilanka Ringwald } 110d8b859a2SMilanka Ringwald 111d8b859a2SMilanka Ringwald void avdtp_configuration_timer_stop(avdtp_connection_t * connection){ 112d8b859a2SMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 113d8b859a2SMilanka Ringwald } 114d8b859a2SMilanka Ringwald 115b0d75c91SMilanka Ringwald static uint16_t avdtp_get_next_initiator_transaction_label(avdtp_context_t * context){ 116b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter++; 117b0d75c91SMilanka Ringwald if (context->initiator_transaction_id_counter == 0){ 118b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter = 1; 119b0d75c91SMilanka Ringwald } 120b0d75c91SMilanka Ringwald return context->initiator_transaction_id_counter; 121b0d75c91SMilanka Ringwald } 122b0d75c91SMilanka Ringwald 123af121d54SMilanka Ringwald static uint16_t avdtp_get_next_avdtp_cid(avdtp_context_t * context){ 124af121d54SMilanka Ringwald do { 125af121d54SMilanka Ringwald if (avdtp_cid_counter == 0xffff) { 1264ccacc40SMilanka Ringwald avdtp_cid_counter = 1; 127af121d54SMilanka Ringwald } else { 128af121d54SMilanka Ringwald avdtp_cid_counter++; 1294ccacc40SMilanka Ringwald } 130af121d54SMilanka Ringwald } while (avdtp_connection_for_avdtp_cid(avdtp_cid_counter, context) != NULL) ; 1314ccacc40SMilanka Ringwald return avdtp_cid_counter; 1324ccacc40SMilanka Ringwald } 1334ccacc40SMilanka Ringwald 134af121d54SMilanka Ringwald static avdtp_stream_endpoint_t * avdtp_stream_endpoint_for_id(avdtp_context_t * context, uint16_t stream_endpoint_id) { 135af121d54SMilanka Ringwald btstack_linked_item_t *it; 136af121d54SMilanka Ringwald for (it = (btstack_linked_item_t *) context->stream_endpoints; it ; it = it->next){ 137af121d54SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = ((avdtp_stream_endpoint_t *) it); 138af121d54SMilanka Ringwald 139af121d54SMilanka Ringwald if (stream_endpoint->sep.seid == stream_endpoint_id) { 140af121d54SMilanka Ringwald return stream_endpoint; 141af121d54SMilanka Ringwald }; 142af121d54SMilanka Ringwald } 143af121d54SMilanka Ringwald return NULL; 144af121d54SMilanka Ringwald } 145af121d54SMilanka Ringwald 1464ccacc40SMilanka Ringwald static uint16_t avdtp_get_next_local_seid(avdtp_context_t * context){ 147af121d54SMilanka Ringwald uint16_t stream_endpoint_id = context->stream_endpoints_id_counter; 148af121d54SMilanka Ringwald do { 149af121d54SMilanka Ringwald if (stream_endpoint_id == 0xffff) { 150af121d54SMilanka Ringwald stream_endpoint_id = 1; 151af121d54SMilanka Ringwald } else { 152af121d54SMilanka Ringwald stream_endpoint_id++; 1534ccacc40SMilanka Ringwald } 154af121d54SMilanka Ringwald } while (avdtp_stream_endpoint_for_id(context, stream_endpoint_id) != NULL) ; 155af121d54SMilanka Ringwald return stream_endpoint_id; 1564ccacc40SMilanka Ringwald } 1574ccacc40SMilanka Ringwald 158af121d54SMilanka Ringwald 1594ccacc40SMilanka Ringwald uint8_t avdtp_connect(bd_addr_t remote, avdtp_sep_type_t query_role, avdtp_context_t * avdtp_context, uint16_t * avdtp_cid){ 1602f6083d0SMilanka Ringwald sdp_query_context = avdtp_context; 161692c0605SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_bd_addr(remote, avdtp_context); 162692c0605SMilanka Ringwald if (!connection){ 163692c0605SMilanka Ringwald connection = avdtp_create_connection(remote, avdtp_context); 1644567cc17SMilanka Ringwald if (!connection){ 1659900b7faSMilanka Ringwald log_error("Not enough memory to create connection."); 1664567cc17SMilanka Ringwald return BTSTACK_MEMORY_ALLOC_FAILED; 1674567cc17SMilanka Ringwald } 168692c0605SMilanka Ringwald } 1694ccacc40SMilanka Ringwald 1702ad6b656SMilanka Ringwald if (avdtp_cid != NULL) { 171194cd2dfSandryblack *avdtp_cid = connection->avdtp_cid; 1722ad6b656SMilanka Ringwald } 1732ad6b656SMilanka Ringwald 1745448c259SMilanka Ringwald avdtp_context->avdtp_cid = connection->avdtp_cid; 1755448c259SMilanka Ringwald 1765448c259SMilanka Ringwald uint8_t err; 1775448c259SMilanka Ringwald switch (connection->state){ 1785448c259SMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_IDLE: 1795448c259SMilanka Ringwald connection->state = AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE; 1805448c259SMilanka Ringwald sdp_query_context = avdtp_context; 1812f6083d0SMilanka Ringwald avdtp_context->avdtp_l2cap_psm = 0; 1822f6083d0SMilanka Ringwald avdtp_context->avdtp_version = 0; 1832f6083d0SMilanka Ringwald avdtp_context->query_role = query_role; 1845448c259SMilanka Ringwald err = sdp_client_query_uuid16(&avdtp_handle_sdp_client_query_result, remote, BLUETOOTH_PROTOCOL_AVDTP); 1851e1ae2bcSMilanka Ringwald if (err != ERROR_CODE_SUCCESS){ 1861e1ae2bcSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 1872f6083d0SMilanka Ringwald btstack_linked_list_remove(&avdtp_context->connections, (btstack_linked_item_t*) connection); 1882f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 1895448c259SMilanka Ringwald } 1901e1ae2bcSMilanka Ringwald return err; 191a0b8a58cSMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_OPENED:{ 192a0b8a58cSMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_signaling_cid(connection->l2cap_signaling_cid, avdtp_context); 193a0b8a58cSMilanka Ringwald if (stream_endpoint){ 194a0b8a58cSMilanka Ringwald avdtp_streaming_emit_connection_established(avdtp_context->avdtp_callback, connection->avdtp_cid, remote, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint), 0); 195a0b8a58cSMilanka Ringwald break; 196a0b8a58cSMilanka Ringwald } 197596b7fdcSMilanka Ringwald avdtp_signaling_emit_connection_established(avdtp_context->avdtp_callback, connection->avdtp_cid, connection->remote_addr, ERROR_CODE_SUCCESS); 1985448c259SMilanka Ringwald break; 199a0b8a58cSMilanka Ringwald } 2005448c259SMilanka Ringwald default: 2015448c259SMilanka Ringwald log_error("avdtp_connect: sink in wrong state"); 2025448c259SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 2035448c259SMilanka Ringwald 2041e1ae2bcSMilanka Ringwald } 2054ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 206692c0605SMilanka Ringwald } 207747ec646SMilanka Ringwald 208747ec646SMilanka Ringwald void avdtp_register_media_transport_category(avdtp_stream_endpoint_t * stream_endpoint){ 209747ec646SMilanka Ringwald if (!stream_endpoint){ 2109900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 211747ec646SMilanka Ringwald return; 212747ec646SMilanka Ringwald } 213747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MEDIA_TRANSPORT, 1); 214747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 215747ec646SMilanka Ringwald } 216747ec646SMilanka Ringwald 217747ec646SMilanka Ringwald void avdtp_register_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 218747ec646SMilanka Ringwald if (!stream_endpoint){ 2199900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 220747ec646SMilanka Ringwald return; 221747ec646SMilanka Ringwald } 222747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_REPORTING, 1); 223747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 224747ec646SMilanka Ringwald } 225747ec646SMilanka Ringwald 226747ec646SMilanka Ringwald void avdtp_register_delay_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 227747ec646SMilanka Ringwald if (!stream_endpoint){ 2289900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 229747ec646SMilanka Ringwald return; 230747ec646SMilanka Ringwald } 231747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_DELAY_REPORTING, 1); 232747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 233747ec646SMilanka Ringwald } 234747ec646SMilanka Ringwald 235747ec646SMilanka Ringwald void avdtp_register_recovery_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t maximum_recovery_window_size, uint8_t maximum_number_media_packets){ 236747ec646SMilanka Ringwald if (!stream_endpoint){ 2379900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 238747ec646SMilanka Ringwald return; 239747ec646SMilanka Ringwald } 240747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_RECOVERY, 1); 241747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 242747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.recovery_type = 0x01; // 0x01 = RFC2733 243747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_recovery_window_size = maximum_recovery_window_size; 244747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_number_media_packets = maximum_number_media_packets; 245747ec646SMilanka Ringwald } 246747ec646SMilanka Ringwald 247747ec646SMilanka Ringwald void avdtp_register_content_protection_category(avdtp_stream_endpoint_t * stream_endpoint, uint16_t cp_type, const uint8_t * cp_type_value, uint8_t cp_type_value_len){ 248747ec646SMilanka Ringwald if (!stream_endpoint){ 2499900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 250747ec646SMilanka Ringwald return; 251747ec646SMilanka Ringwald } 252747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_CONTENT_PROTECTION, 1); 253747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 254747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type = cp_type; 2556535961aSMatthias Ringwald (void)memcpy(stream_endpoint->sep.capabilities.content_protection.cp_type_value, 2566535961aSMatthias Ringwald cp_type_value, 2576535961aSMatthias Ringwald btstack_min(cp_type_value_len, AVDTP_MAX_CONTENT_PROTECTION_TYPE_VALUE_LEN)); 25867ae582dSMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type_value_len = btstack_min(cp_type_value_len, AVDTP_MAX_CONTENT_PROTECTION_TYPE_VALUE_LEN); 259747ec646SMilanka Ringwald } 260747ec646SMilanka Ringwald 261747ec646SMilanka Ringwald void avdtp_register_header_compression_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t back_ch, uint8_t media, uint8_t recovery){ 262747ec646SMilanka Ringwald if (!stream_endpoint){ 2639900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 264747ec646SMilanka Ringwald return; 265747ec646SMilanka Ringwald } 266747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_HEADER_COMPRESSION, 1); 267747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 268747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.back_ch = back_ch; 269747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.media = media; 270747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.recovery = recovery; 271747ec646SMilanka Ringwald } 272747ec646SMilanka Ringwald 27378d08d09SMilanka Ringwald void avdtp_register_media_codec_category(avdtp_stream_endpoint_t * stream_endpoint, avdtp_media_type_t media_type, avdtp_media_codec_type_t media_codec_type, uint8_t * media_codec_info, uint16_t media_codec_info_len){ 274747ec646SMilanka Ringwald if (!stream_endpoint){ 2759900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 276747ec646SMilanka Ringwald return; 277747ec646SMilanka Ringwald } 278747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MEDIA_CODEC, 1); 279747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 280747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_type = media_type; 281747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_type = media_codec_type; 282747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information = media_codec_info; 283747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information_len = media_codec_info_len; 284747ec646SMilanka Ringwald } 285747ec646SMilanka Ringwald 286747ec646SMilanka Ringwald void avdtp_register_multiplexing_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t fragmentation){ 287747ec646SMilanka Ringwald if (!stream_endpoint){ 2889900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 289747ec646SMilanka Ringwald return; 290747ec646SMilanka Ringwald } 291747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MULTIPLEXING, 1); 292747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 293747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.multiplexing_mode.fragmentation = fragmentation; 294747ec646SMilanka Ringwald } 295747ec646SMilanka Ringwald 296747ec646SMilanka Ringwald 297747ec646SMilanka Ringwald /* START: tracking can send now requests pro l2cap cid */ 298747ec646SMilanka Ringwald void avdtp_handle_can_send_now(avdtp_connection_t * connection, uint16_t l2cap_cid, avdtp_context_t * context){ 299747ec646SMilanka Ringwald if (connection->wait_to_send_acceptor){ 300747ec646SMilanka Ringwald connection->wait_to_send_acceptor = 0; 301747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm_run(connection, context); 302747ec646SMilanka Ringwald } else if (connection->wait_to_send_initiator){ 303747ec646SMilanka Ringwald connection->wait_to_send_initiator = 0; 304747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm_run(connection, context); 305747ec646SMilanka Ringwald } else if (connection->wait_to_send_self){ 306747ec646SMilanka Ringwald connection->wait_to_send_self = 0; 307747ec646SMilanka Ringwald if (connection->disconnect){ 308747ec646SMilanka Ringwald btstack_linked_list_iterator_t it; 309747ec646SMilanka Ringwald btstack_linked_list_iterator_init(&it, &context->stream_endpoints); 310747ec646SMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 311747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 312747ec646SMilanka Ringwald if (stream_endpoint->connection == connection){ 3130e588213SMatthias Ringwald if ((stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_OPENED) && (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED)){ 314747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED; 3159413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 316747ec646SMilanka Ringwald l2cap_disconnect(stream_endpoint->l2cap_media_cid, 0); 317747ec646SMilanka Ringwald return; 318747ec646SMilanka Ringwald } 319747ec646SMilanka Ringwald } 320747ec646SMilanka Ringwald } 321747ec646SMilanka Ringwald connection->disconnect = 0; 322747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED; 3239413b167SMilanka Ringwald l2cap_disconnect(connection->l2cap_signaling_cid, 0); 324747ec646SMilanka Ringwald return; 325747ec646SMilanka Ringwald } 326747ec646SMilanka Ringwald } 327747ec646SMilanka Ringwald 328747ec646SMilanka Ringwald // re-register 329747ec646SMilanka Ringwald int more_to_send = connection->wait_to_send_acceptor || connection->wait_to_send_initiator || connection->wait_to_send_self; 330747ec646SMilanka Ringwald if (more_to_send){ 331747ec646SMilanka Ringwald l2cap_request_can_send_now_event(l2cap_cid); 332747ec646SMilanka Ringwald } 333747ec646SMilanka Ringwald } 334747ec646SMilanka Ringwald /* END: tracking can send now requests pro l2cap cid */ 335747ec646SMilanka Ringwald 336747ec646SMilanka Ringwald avdtp_connection_t * avdtp_create_connection(bd_addr_t remote_addr, avdtp_context_t * context){ 337747ec646SMilanka Ringwald avdtp_connection_t * connection = btstack_memory_avdtp_connection_get(); 3384567cc17SMilanka Ringwald if (!connection){ 3399900b7faSMilanka Ringwald log_error("Not enough memory to create connection"); 3404567cc17SMilanka Ringwald return NULL; 3414567cc17SMilanka Ringwald } 342747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 343b0d75c91SMilanka Ringwald connection->initiator_transaction_label = avdtp_get_next_initiator_transaction_label(context); 344af121d54SMilanka Ringwald connection->avdtp_cid = avdtp_get_next_avdtp_cid(context); 345e57d7404SMatthias Ringwald connection->configuration_state = AVDTP_CONFIGURATION_STATE_IDLE; 34663331bf4SMilanka Ringwald context->avdtp_cid = connection->avdtp_cid; 3476535961aSMatthias Ringwald (void)memcpy(connection->remote_addr, remote_addr, 6); 348747ec646SMilanka Ringwald btstack_linked_list_add(&context->connections, (btstack_linked_item_t *) connection); 349747ec646SMilanka Ringwald return connection; 350747ec646SMilanka Ringwald } 351747ec646SMilanka Ringwald 352747ec646SMilanka Ringwald avdtp_stream_endpoint_t * avdtp_create_stream_endpoint(avdtp_sep_type_t sep_type, avdtp_media_type_t media_type, avdtp_context_t * context){ 353747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = btstack_memory_avdtp_stream_endpoint_get(); 3544567cc17SMilanka Ringwald if (!stream_endpoint){ 3559900b7faSMilanka Ringwald log_error("Not enough memory to create stream endpoint"); 3564567cc17SMilanka Ringwald return NULL; 3574567cc17SMilanka Ringwald } 3584ccacc40SMilanka Ringwald stream_endpoint->sep.seid = avdtp_get_next_local_seid(context); 359747ec646SMilanka Ringwald stream_endpoint->sep.media_type = media_type; 360747ec646SMilanka Ringwald stream_endpoint->sep.type = sep_type; 361747ec646SMilanka Ringwald btstack_linked_list_add(&context->stream_endpoints, (btstack_linked_item_t *) stream_endpoint); 362747ec646SMilanka Ringwald return stream_endpoint; 363747ec646SMilanka Ringwald } 364747ec646SMilanka Ringwald 365747ec646SMilanka Ringwald 366747ec646SMilanka Ringwald static void handle_l2cap_data_packet_for_signaling_connection(avdtp_connection_t * connection, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 367747ec646SMilanka Ringwald int offset = avdtp_read_signaling_header(&connection->signaling_packet, packet, size); 368747ec646SMilanka Ringwald switch (connection->signaling_packet.message_type){ 369747ec646SMilanka Ringwald case AVDTP_CMD_MSG: 370747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm(connection, packet, size, offset, context); 371747ec646SMilanka Ringwald break; 372747ec646SMilanka Ringwald default: 373747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm(connection, packet, size, offset, context); 374747ec646SMilanka Ringwald break; 375747ec646SMilanka Ringwald } 376747ec646SMilanka Ringwald } 377747ec646SMilanka Ringwald 378692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 3792f6083d0SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(sdp_query_context->avdtp_cid, sdp_query_context); 3802f6083d0SMilanka Ringwald if (!connection) { 381355ac553SMilanka Ringwald log_error("SDP query, connection with 0x%02x cid not found", sdp_query_context->avdtp_cid); 3822f6083d0SMilanka Ringwald return; 3832f6083d0SMilanka Ringwald } 3842f6083d0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE) return; 3852f6083d0SMilanka Ringwald 386692c0605SMilanka Ringwald UNUSED(packet_type); 387692c0605SMilanka Ringwald UNUSED(channel); 388692c0605SMilanka Ringwald UNUSED(size); 389692c0605SMilanka Ringwald 390692c0605SMilanka Ringwald des_iterator_t des_list_it; 391692c0605SMilanka Ringwald des_iterator_t prot_it; 3921e1ae2bcSMilanka Ringwald uint8_t status; 393692c0605SMilanka Ringwald 394692c0605SMilanka Ringwald switch (hci_event_packet_get_type(packet)){ 395692c0605SMilanka Ringwald case SDP_EVENT_QUERY_ATTRIBUTE_VALUE: 396692c0605SMilanka Ringwald // Handle new SDP record 397692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_record_id(packet) != record_id) { 398692c0605SMilanka Ringwald record_id = sdp_event_query_attribute_byte_get_record_id(packet); 3998587e32cSMilanka Ringwald // log_info("SDP Record: Nr: %d", record_id); 400692c0605SMilanka Ringwald } 401692c0605SMilanka Ringwald 402692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_attribute_length(packet) <= attribute_value_buffer_size) { 403692c0605SMilanka Ringwald attribute_value[sdp_event_query_attribute_byte_get_data_offset(packet)] = sdp_event_query_attribute_byte_get_data(packet); 404692c0605SMilanka Ringwald 405692c0605SMilanka Ringwald if ((uint16_t)(sdp_event_query_attribute_byte_get_data_offset(packet)+1) == sdp_event_query_attribute_byte_get_attribute_length(packet)) { 406692c0605SMilanka Ringwald 407692c0605SMilanka Ringwald switch(sdp_event_query_attribute_byte_get_attribute_id(packet)) { 408692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_SERVICE_CLASS_ID_LIST: 409692c0605SMilanka Ringwald if (de_get_element_type(attribute_value) != DE_DES) break; 410692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 411692c0605SMilanka Ringwald uint8_t * element = des_iterator_get_element(&des_list_it); 412692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 413692c0605SMilanka Ringwald uint32_t uuid = de_get_uuid32(element); 414692c0605SMilanka Ringwald switch (uuid){ 415692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SOURCE: 4162f6083d0SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SOURCE) { 4172f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 418692c0605SMilanka Ringwald break; 419692c0605SMilanka Ringwald } 4208587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SOURCE protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 421eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 422692c0605SMilanka Ringwald break; 423692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SINK: 4245448c259SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SINK) { 4252f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 426692c0605SMilanka Ringwald break; 427692c0605SMilanka Ringwald } 4288587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SINK protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 429eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 430692c0605SMilanka Ringwald break; 431692c0605SMilanka Ringwald default: 432692c0605SMilanka Ringwald break; 433692c0605SMilanka Ringwald } 434692c0605SMilanka Ringwald } 435692c0605SMilanka Ringwald break; 436692c0605SMilanka Ringwald 437692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_PROTOCOL_DESCRIPTOR_LIST: { 4388587e32cSMilanka Ringwald // log_info("SDP Attribute: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet)); 439692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 440692c0605SMilanka Ringwald uint8_t *des_element; 441692c0605SMilanka Ringwald uint8_t *element; 442692c0605SMilanka Ringwald uint32_t uuid; 443692c0605SMilanka Ringwald 444692c0605SMilanka Ringwald if (des_iterator_get_type(&des_list_it) != DE_DES) continue; 445692c0605SMilanka Ringwald 446692c0605SMilanka Ringwald des_element = des_iterator_get_element(&des_list_it); 447692c0605SMilanka Ringwald des_iterator_init(&prot_it, des_element); 448692c0605SMilanka Ringwald element = des_iterator_get_element(&prot_it); 449692c0605SMilanka Ringwald 450692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 451692c0605SMilanka Ringwald 452692c0605SMilanka Ringwald uuid = de_get_uuid32(element); 45314fd128cSMatthias Ringwald des_iterator_next(&prot_it); 454692c0605SMilanka Ringwald switch (uuid){ 455692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_L2CAP: 456692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 4572f6083d0SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_l2cap_psm); 458692c0605SMilanka Ringwald break; 459692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_AVDTP: 460692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 4612f6083d0SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_version); 462692c0605SMilanka Ringwald break; 463692c0605SMilanka Ringwald default: 464692c0605SMilanka Ringwald break; 465692c0605SMilanka Ringwald } 466692c0605SMilanka Ringwald } 467692c0605SMilanka Ringwald } 468692c0605SMilanka Ringwald break; 469692c0605SMilanka Ringwald default: 470692c0605SMilanka Ringwald break; 471692c0605SMilanka Ringwald } 472692c0605SMilanka Ringwald } 473692c0605SMilanka Ringwald } else { 4748587e32cSMilanka Ringwald log_error("SDP attribute value buffer size exceeded: available %d, required %d", attribute_value_buffer_size, sdp_event_query_attribute_byte_get_attribute_length(packet)); 475692c0605SMilanka Ringwald } 476692c0605SMilanka Ringwald break; 477692c0605SMilanka Ringwald 478692c0605SMilanka Ringwald case SDP_EVENT_QUERY_COMPLETE: 4790d4a198eSMatthias Ringwald if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE){ 4800d4a198eSMatthias Ringwald // bail out, we must have had an incoming connection in the meantime; 4810d4a198eSMatthias Ringwald break; 4820d4a198eSMatthias Ringwald } 4831e1ae2bcSMilanka Ringwald status = sdp_event_query_complete_get_status(packet); 4842f6083d0SMilanka Ringwald if (status != ERROR_CODE_SUCCESS){ 4855448c259SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, status); 4862f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4872f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 488355ac553SMilanka Ringwald log_info("SDP query failed with status 0x%02x.", status); 4891e1ae2bcSMilanka Ringwald break; 4901e1ae2bcSMilanka Ringwald } 4912f6083d0SMilanka Ringwald if (!sdp_query_context->role_supported){ 4922f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4932f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 4942f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, SDP_SERVICE_NOT_FOUND); 495355ac553SMilanka Ringwald log_info("SDP query, remote device does not support required role."); 496974d4d6eSMilanka Ringwald break; 497974d4d6eSMilanka Ringwald } 4982f6083d0SMilanka Ringwald if (!sdp_query_context->avdtp_l2cap_psm) { 4992f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*)connection); 5002f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 5012f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, L2CAP_SERVICE_DOES_NOT_EXIST); 502355ac553SMilanka Ringwald log_info("SDP query, no l2cap psm found."); 5032f6083d0SMilanka Ringwald break; 5042f6083d0SMilanka Ringwald } 5052f6083d0SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 5062f6083d0SMilanka Ringwald l2cap_create_channel(sdp_query_context->packet_handler, connection->remote_addr, sdp_query_context->avdtp_l2cap_psm, l2cap_max_mtu(), NULL); 507692c0605SMilanka Ringwald break; 508692c0605SMilanka Ringwald } 509692c0605SMilanka Ringwald } 510692c0605SMilanka Ringwald 511692c0605SMilanka Ringwald 512747ec646SMilanka Ringwald void avdtp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 513747ec646SMilanka Ringwald bd_addr_t event_addr; 514747ec646SMilanka Ringwald uint16_t psm; 515747ec646SMilanka Ringwald uint16_t local_cid; 5161e1ae2bcSMilanka Ringwald uint8_t status; 5170d4a198eSMatthias Ringwald int accept_signaling_connection; 518747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = NULL; 519747ec646SMilanka Ringwald avdtp_connection_t * connection = NULL; 520747ec646SMilanka Ringwald btstack_linked_list_t * avdtp_connections = &context->connections; 521747ec646SMilanka Ringwald btstack_linked_list_t * stream_endpoints = &context->stream_endpoints; 522747ec646SMilanka Ringwald handle_media_data = context->handle_media_data; 5238587e32cSMilanka Ringwald // log_info("avdtp_packet_handler packet type %02x, event %02x ", packet_type, hci_event_packet_get_type(packet)); 524747ec646SMilanka Ringwald switch (packet_type) { 525747ec646SMilanka Ringwald case L2CAP_DATA_PACKET: 5269413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 527747ec646SMilanka Ringwald if (connection){ 528747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 529747ec646SMilanka Ringwald break; 530747ec646SMilanka Ringwald } 531747ec646SMilanka Ringwald 532747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 533747ec646SMilanka Ringwald if (!stream_endpoint){ 534747ec646SMilanka Ringwald if (!connection) break; 535747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 536747ec646SMilanka Ringwald break; 537747ec646SMilanka Ringwald } 538747ec646SMilanka Ringwald 5398c0f3635SMilanka Ringwald if (stream_endpoint->connection){ 5409413b167SMilanka Ringwald if (channel == stream_endpoint->connection->l2cap_signaling_cid){ 541a466d508SMilanka Ringwald int offset = avdtp_read_signaling_header(&stream_endpoint->connection->signaling_packet, packet, size); 542a466d508SMilanka Ringwald if (stream_endpoint->connection->signaling_packet.message_type == AVDTP_CMD_MSG){ 543a466d508SMilanka Ringwald avdtp_acceptor_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context); 544a466d508SMilanka Ringwald } else { 545a466d508SMilanka Ringwald avdtp_initiator_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context); 546a466d508SMilanka Ringwald } 547747ec646SMilanka Ringwald break; 548747ec646SMilanka Ringwald } 5498c0f3635SMilanka Ringwald } 550747ec646SMilanka Ringwald 551747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_media_cid){ 5528b097e29SMilanka Ringwald if (handle_media_data){ 553fd58c900SMilanka Ringwald (*handle_media_data)(avdtp_local_seid(stream_endpoint), packet, size); 5548b097e29SMilanka Ringwald } 555747ec646SMilanka Ringwald break; 556747ec646SMilanka Ringwald } 557747ec646SMilanka Ringwald 558747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_reporting_cid){ 559747ec646SMilanka Ringwald // TODO 5608587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for reporting: NOT IMPLEMENTED"); 561747ec646SMilanka Ringwald } else if (channel == stream_endpoint->l2cap_recovery_cid){ 562747ec646SMilanka Ringwald // TODO 5638587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for recovery: NOT IMPLEMENTED"); 564747ec646SMilanka Ringwald } else { 565747ec646SMilanka Ringwald log_error("avdtp packet handler L2CAP_DATA_PACKET: local cid 0x%02x not found", channel); 566747ec646SMilanka Ringwald } 567747ec646SMilanka Ringwald break; 568747ec646SMilanka Ringwald 569747ec646SMilanka Ringwald case HCI_EVENT_PACKET: 570747ec646SMilanka Ringwald switch (hci_event_packet_get_type(packet)) { 571747ec646SMilanka Ringwald case L2CAP_EVENT_INCOMING_CONNECTION: 572747ec646SMilanka Ringwald l2cap_event_incoming_connection_get_address(packet, event_addr); 573747ec646SMilanka Ringwald local_cid = l2cap_event_incoming_connection_get_local_cid(packet); 574747ec646SMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 575355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION, local cid 0x%02x ", local_cid); 5760d4a198eSMatthias Ringwald if (!connection){ 5770d4a198eSMatthias Ringwald // create connection struct for regular inconming connection request 578747ec646SMilanka Ringwald connection = avdtp_create_connection(event_addr, context); 5790d4a198eSMatthias Ringwald if (!connection){ 5800d4a198eSMatthias Ringwald log_error("Could not create connection, reject"); 5810d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 5820d4a198eSMatthias Ringwald break; 5830d4a198eSMatthias Ringwald } 5840d4a198eSMatthias Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 5850d4a198eSMatthias Ringwald } 5860d4a198eSMatthias Ringwald 5870d4a198eSMatthias Ringwald // connection exists, check states 5880d4a198eSMatthias Ringwald log_info("Connection state %d", connection->state); 589cf837c28SMatthias Ringwald accept_signaling_connection = 0; 5900d4a198eSMatthias Ringwald switch (connection->state){ 5910d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_IDLE: 5920d4a198eSMatthias Ringwald // regular incoming connection 5930d4a198eSMatthias Ringwald accept_signaling_connection = 1; 5940d4a198eSMatthias Ringwald break; 5950d4a198eSMatthias Ringwald case AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE: 5960d4a198eSMatthias Ringwald // incoming connection during sdp query, just accept it 5970d4a198eSMatthias Ringwald accept_signaling_connection = 1; 5980d4a198eSMatthias Ringwald break; 5990d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED: 6000d4a198eSMatthias Ringwald log_info("Reject incoming connection after creating outgoing"); 6010d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 6020d4a198eSMatthias Ringwald return; 6030d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_OPENED: 6040d4a198eSMatthias Ringwald // handled below 6050d4a198eSMatthias Ringwald break; 6060d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED: 6070d4a198eSMatthias Ringwald log_info("Reject incoming connection during disconnect"); 6080d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 6090d4a198eSMatthias Ringwald return; 6100d4a198eSMatthias Ringwald } 6110d4a198eSMatthias Ringwald 6120d4a198eSMatthias Ringwald if (accept_signaling_connection){ 613747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 614355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION, connection %p, state connection %d, avdtp cid 0x%02x", connection, connection->state, connection->avdtp_cid); 615747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 616747ec646SMilanka Ringwald break; 617747ec646SMilanka Ringwald } 618747ec646SMilanka Ringwald 6190d4a198eSMatthias Ringwald // now, we're only dealing with media connections 6206b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 621747ec646SMilanka Ringwald if (!stream_endpoint) { 622355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION no streamendpoint found for seid %d", connection->local_seid); 6230d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 624747ec646SMilanka Ringwald break; 625747ec646SMilanka Ringwald } 626747ec646SMilanka Ringwald 627*a3ce0109SMatthias Ringwald log_info("Checking l2cap_media_cid %d, for local seid %d, state of stream endpoint %d", stream_endpoint->l2cap_media_cid, connection->local_seid, stream_endpoint->state); 628*a3ce0109SMatthias Ringwald if (stream_endpoint->l2cap_media_cid != 0){ 629*a3ce0109SMatthias Ringwald l2cap_decline_connection(local_cid); 630*a3ce0109SMatthias Ringwald break; 631*a3ce0109SMatthias Ringwald } 632*a3ce0109SMatthias Ringwald if (connection->configuration_state != AVDTP_CONFIGURATION_STATE_REMOTE_CONFIGURED){ 6339900b7faSMilanka Ringwald l2cap_decline_connection(local_cid); 6349900b7faSMilanka Ringwald break; 6359900b7faSMilanka Ringwald } 636747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 637747ec646SMilanka Ringwald break; 638747ec646SMilanka Ringwald 639747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_OPENED: 640a0b8a58cSMilanka Ringwald psm = l2cap_event_channel_opened_get_psm(packet); 64184e3541eSMilanka Ringwald if (psm != BLUETOOTH_PSM_AVDTP){ 642355ac553SMilanka Ringwald log_info("Unexpected PSM - Not implemented yet, avdtp sink: L2CAP_EVENT_CHANNEL_OPENED "); 643a0b8a58cSMilanka Ringwald return; 644a0b8a58cSMilanka Ringwald } 645a0b8a58cSMilanka Ringwald 6461e1ae2bcSMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 647747ec646SMilanka Ringwald // inform about new l2cap connection 648747ec646SMilanka Ringwald l2cap_event_channel_opened_get_address(packet, event_addr); 6497050d2caSMilanka Ringwald local_cid = l2cap_event_channel_opened_get_local_cid(packet); 650a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 651a0b8a58cSMilanka Ringwald 652355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED: status %d, cid 0x%02x , signaling connection %p ", status, local_cid, connection); 653a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 654a0b8a58cSMilanka Ringwald if (!connection){ 655355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED 2: status %d, signaling connection %p ", status, connection); 656a0b8a58cSMilanka Ringwald break; 657a0b8a58cSMilanka Ringwald } 658a0b8a58cSMilanka Ringwald 659a0b8a58cSMilanka Ringwald if (connection->l2cap_signaling_cid == 0) { 6601e1ae2bcSMilanka Ringwald if (status){ 661355ac553SMilanka Ringwald log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 66255ddebccSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 663a0b8a58cSMilanka Ringwald connection->l2cap_signaling_cid = 0; 6644ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, l2cap_event_channel_opened_get_status(packet)); 665747ec646SMilanka Ringwald break; 666747ec646SMilanka Ringwald } 667a0b8a58cSMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED) { 668355ac553SMilanka Ringwald log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 669a0b8a58cSMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, AVDTP_CONNECTION_IN_WRONG_STATE); 670a0b8a58cSMilanka Ringwald break; 67155ddebccSMilanka Ringwald } 6729413b167SMilanka Ringwald connection->l2cap_signaling_cid = local_cid; 6736b0ee1d0SMilanka Ringwald connection->local_seid = 0; 674747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_OPENED; 675355ac553SMilanka Ringwald log_info("AVDTP_SIGNALING_CONNECTION_OPENED, connection %p, l2cap_signaling_cid 0x%02x, avdtp_cid 0x%02x", connection, local_cid, connection->avdtp_cid); 6764ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, 0); 677747ec646SMilanka Ringwald break; 678747ec646SMilanka Ringwald } 679747ec646SMilanka Ringwald 6806b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 681747ec646SMilanka Ringwald if (!stream_endpoint){ 682355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED: stream_endpoint not found"); 683747ec646SMilanka Ringwald return; 684747ec646SMilanka Ringwald } 685a466d508SMilanka Ringwald 686a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 687a466d508SMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 688a466d508SMilanka Ringwald if (status){ 689355ac553SMilanka Ringwald log_info("AVDTP_STREAM_ENDPOINT_OPENED failed with status %d, avdtp cid 0x%02x, l2cap_media_cid 0x%02x, local seid %d, remote seid %d", status, connection->avdtp_cid, stream_endpoint->l2cap_media_cid, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint)); 690a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_IDLE; 691596b7fdcSMilanka Ringwald avdtp_streaming_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint), status); 692a466d508SMilanka Ringwald break; 693a466d508SMilanka Ringwald } 694a466d508SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED){ 695355ac553SMilanka Ringwald log_info("AVDTP_STREAM_ENDPOINT_OPENED failed - stream endpoint in wrong state %d, avdtp cid 0x%02x, l2cap_media_cid 0x%02x, local seid %d, remote seid %d", stream_endpoint->state, connection->avdtp_cid, stream_endpoint->l2cap_media_cid, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint)); 696596b7fdcSMilanka Ringwald avdtp_streaming_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint), AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE); 697a466d508SMilanka Ringwald break; 698a466d508SMilanka Ringwald } 699a466d508SMilanka Ringwald 700a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 701a466d508SMilanka Ringwald stream_endpoint->connection = connection; 702a466d508SMilanka Ringwald stream_endpoint->l2cap_media_cid = l2cap_event_channel_opened_get_local_cid(packet); 703a466d508SMilanka Ringwald stream_endpoint->media_con_handle = l2cap_event_channel_opened_get_handle(packet); 704d1207cd8SMilanka Ringwald 705355ac553SMilanka Ringwald log_info("AVDTP_STREAM_ENDPOINT_OPENED, avdtp cid 0x%02x, l2cap_media_cid 0x%02x, local seid %d, remote seid %d", connection->avdtp_cid, stream_endpoint->l2cap_media_cid, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint)); 706596b7fdcSMilanka Ringwald avdtp_streaming_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint), 0); 707a466d508SMilanka Ringwald return; 708a466d508SMilanka Ringwald } 709747ec646SMilanka Ringwald break; 710747ec646SMilanka Ringwald 711747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_CLOSED: 712747ec646SMilanka Ringwald local_cid = l2cap_event_channel_closed_get_local_cid(packet); 7139413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(local_cid, context); 7140bd7cb1fSMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(local_cid, context); 715485c0a4cSMilanka Ringwald log_info("Received L2CAP_EVENT_CHANNEL_CLOSED, cid 0x%2x, connection %p, stream_endpoint %p", local_cid, connection, stream_endpoint); 7160bd7cb1fSMilanka Ringwald 717a466d508SMilanka Ringwald if (stream_endpoint){ 718a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == local_cid){ 719a466d508SMilanka Ringwald connection = stream_endpoint->connection; 720596b7fdcSMilanka Ringwald if (connection) { 721596b7fdcSMilanka Ringwald avdtp_streaming_emit_connection_released(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint)); 722596b7fdcSMilanka Ringwald } 723485c0a4cSMilanka Ringwald avdtp_reset_stream_endpoint(stream_endpoint); 724a466d508SMilanka Ringwald if (connection && connection->disconnect){ 725a466d508SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 726a466d508SMilanka Ringwald } 727a466d508SMilanka Ringwald break; 728a466d508SMilanka Ringwald } 729a466d508SMilanka Ringwald if (stream_endpoint->l2cap_recovery_cid == local_cid){ 730355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_CLOSED recovery cid 0x%0x", local_cid); 731a466d508SMilanka Ringwald stream_endpoint->l2cap_recovery_cid = 0; 732a466d508SMilanka Ringwald break; 733a466d508SMilanka Ringwald } 734a466d508SMilanka Ringwald 735a466d508SMilanka Ringwald if (stream_endpoint->l2cap_reporting_cid == local_cid){ 736355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_CLOSED reporting cid 0x%0x", local_cid); 737a466d508SMilanka Ringwald stream_endpoint->l2cap_reporting_cid = 0; 738a466d508SMilanka Ringwald break; 739a466d508SMilanka Ringwald } 740a466d508SMilanka Ringwald } 741596b7fdcSMilanka Ringwald 742596b7fdcSMilanka Ringwald if (connection){ 743596b7fdcSMilanka Ringwald btstack_linked_list_iterator_t it; 744596b7fdcSMilanka Ringwald btstack_linked_list_iterator_init(&it, stream_endpoints); 745596b7fdcSMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 746596b7fdcSMilanka Ringwald avdtp_stream_endpoint_t * _stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 747596b7fdcSMilanka Ringwald if (_stream_endpoint->connection == connection){ 748485c0a4cSMilanka Ringwald avdtp_reset_stream_endpoint(_stream_endpoint); 749596b7fdcSMilanka Ringwald } 750596b7fdcSMilanka Ringwald } 751485c0a4cSMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 752485c0a4cSMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 753485c0a4cSMilanka Ringwald btstack_linked_list_remove(avdtp_connections, (btstack_linked_item_t*) connection); 754596b7fdcSMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 755596b7fdcSMilanka Ringwald break; 756596b7fdcSMilanka Ringwald } 757596b7fdcSMilanka Ringwald 758747ec646SMilanka Ringwald break; 759747ec646SMilanka Ringwald 760747ec646SMilanka Ringwald case HCI_EVENT_DISCONNECTION_COMPLETE: 761747ec646SMilanka Ringwald break; 762747ec646SMilanka Ringwald 763747ec646SMilanka Ringwald case L2CAP_EVENT_CAN_SEND_NOW: 7649413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 765747ec646SMilanka Ringwald if (!connection) { 766747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 767747ec646SMilanka Ringwald if (!stream_endpoint->connection) break; 768747ec646SMilanka Ringwald connection = stream_endpoint->connection; 769747ec646SMilanka Ringwald } 770747ec646SMilanka Ringwald avdtp_handle_can_send_now(connection, channel, context); 771747ec646SMilanka Ringwald break; 772747ec646SMilanka Ringwald default: 773355ac553SMilanka Ringwald log_info("Unknown HCI event type %02x", hci_event_packet_get_type(packet)); 774747ec646SMilanka Ringwald break; 775747ec646SMilanka Ringwald } 776747ec646SMilanka Ringwald break; 777747ec646SMilanka Ringwald 778747ec646SMilanka Ringwald default: 779747ec646SMilanka Ringwald // other packet type 780747ec646SMilanka Ringwald break; 781747ec646SMilanka Ringwald } 782747ec646SMilanka Ringwald } 783747ec646SMilanka Ringwald 7844ccacc40SMilanka Ringwald uint8_t avdtp_disconnect(uint16_t avdtp_cid, avdtp_context_t * context){ 7854ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 7862ad6b656SMilanka Ringwald if (!connection) return AVDTP_CONNECTION_DOES_NOT_EXIST; 787a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_IDLE){ 788a466d508SMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 789a466d508SMilanka Ringwald return ERROR_CODE_SUCCESS; 790a466d508SMilanka Ringwald } 791a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED) return ERROR_CODE_SUCCESS; 792747ec646SMilanka Ringwald 793747ec646SMilanka Ringwald connection->disconnect = 1; 7949413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 7954ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 796747ec646SMilanka Ringwald } 797747ec646SMilanka Ringwald 7986b0ee1d0SMilanka Ringwald uint8_t avdtp_open_stream(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, avdtp_context_t * context){ 7994ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 800747ec646SMilanka Ringwald if (!connection){ 8018587e32cSMilanka Ringwald log_error("avdtp_media_connect: no connection for signaling cid 0x%02x found", avdtp_cid); 8024ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 803747ec646SMilanka Ringwald } 804747ec646SMilanka Ringwald 805747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) { 8068587e32cSMilanka Ringwald log_error("avdtp_media_connect: wrong connection state %d", connection->state); 8074ccacc40SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 808747ec646SMilanka Ringwald } 809747ec646SMilanka Ringwald 8106b0ee1d0SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 811747ec646SMilanka Ringwald if (!stream_endpoint) { 8126b0ee1d0SMilanka Ringwald log_error("avdtp_media_connect: no stream_endpoint with seid %d found", local_seid); 8134ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 814747ec646SMilanka Ringwald } 815747ec646SMilanka Ringwald 816485c0a4cSMilanka Ringwald if (stream_endpoint->remote_sep.seid != remote_seid){ 817485c0a4cSMilanka Ringwald log_error("avdtp_media_connect: no remote sep with seid %d registered with the stream endpoint", remote_seid); 818485c0a4cSMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 819485c0a4cSMilanka Ringwald } 820485c0a4cSMilanka Ringwald 8214ccacc40SMilanka Ringwald if (stream_endpoint->state < AVDTP_STREAM_ENDPOINT_CONFIGURED) return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 822747ec646SMilanka Ringwald 823747ec646SMilanka Ringwald connection->initiator_transaction_label++; 8249413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8256b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 826747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_OPEN_STREAM; 827747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W2_REQUEST_OPEN_STREAM; 8289413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8294ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 830747ec646SMilanka Ringwald } 831747ec646SMilanka Ringwald 8324ccacc40SMilanka Ringwald uint8_t avdtp_start_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8334ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 83446e6b063SMilanka Ringwald if (!connection){ 8354ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8364ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 83746e6b063SMilanka Ringwald } 8385cfe7f4cSMilanka Ringwald 8394ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8404ccacc40SMilanka Ringwald if (!stream_endpoint) { 8414ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no stream_endpoint with seid %d found", local_seid); 8424ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8434ccacc40SMilanka Ringwald } 8444ccacc40SMilanka Ringwald 8454ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8464ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8474ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8484ccacc40SMilanka Ringwald } 8494ccacc40SMilanka Ringwald 850485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint)){ 851485c0a4cSMilanka Ringwald log_error("avdtp_media_connect: no remote sep registered with the stream endpoint"); 852485c0a4cSMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8534ccacc40SMilanka Ringwald } 8544ccacc40SMilanka Ringwald 85560ec20d0SMilanka Ringwald stream_endpoint->start_stream = 1; 8564ccacc40SMilanka Ringwald connection->local_seid = local_seid; 857c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 8589413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8594ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 860747ec646SMilanka Ringwald } 861747ec646SMilanka Ringwald 8624ccacc40SMilanka Ringwald uint8_t avdtp_stop_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8634ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 864747ec646SMilanka Ringwald if (!connection){ 8654ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8664ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 867747ec646SMilanka Ringwald } 8684ccacc40SMilanka Ringwald 8694ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8704ccacc40SMilanka Ringwald if (!stream_endpoint) { 8714ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no stream_endpoint with seid %d found", local_seid); 8724ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8734ccacc40SMilanka Ringwald } 8744ccacc40SMilanka Ringwald 8754ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8764ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8774ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8784ccacc40SMilanka Ringwald } 879485c0a4cSMilanka Ringwald 880485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->stop_stream){ 881485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 882485c0a4cSMilanka Ringwald } 8834ccacc40SMilanka Ringwald 88460ec20d0SMilanka Ringwald stream_endpoint->stop_stream = 1; 8854ccacc40SMilanka Ringwald connection->local_seid = local_seid; 886c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 8879413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8884ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 889747ec646SMilanka Ringwald } 890747ec646SMilanka Ringwald 8914ccacc40SMilanka Ringwald uint8_t avdtp_abort_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8924ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 89360ec20d0SMilanka Ringwald if (!connection){ 8944ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8954ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 896747ec646SMilanka Ringwald } 8974ccacc40SMilanka Ringwald 8984ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8994ccacc40SMilanka Ringwald if (!stream_endpoint) { 9004ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no stream_endpoint with seid %d found", local_seid); 9014ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 9024ccacc40SMilanka Ringwald } 9034ccacc40SMilanka Ringwald 9044ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 9054ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no media connection for stream_endpoint with seid %d found", local_seid); 9064ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 9074ccacc40SMilanka Ringwald } 908485c0a4cSMilanka Ringwald 909485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->abort_stream){ 910485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 911485c0a4cSMilanka Ringwald } 9124ccacc40SMilanka Ringwald 91360ec20d0SMilanka Ringwald stream_endpoint->abort_stream = 1; 9146b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 915c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 9169413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 9174ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 918747ec646SMilanka Ringwald } 919747ec646SMilanka Ringwald 9204ccacc40SMilanka Ringwald uint8_t avdtp_suspend_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 9214ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 922747ec646SMilanka Ringwald if (!connection){ 9234ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 9244ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 92560ec20d0SMilanka Ringwald } 9264ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 9274ccacc40SMilanka Ringwald if (!stream_endpoint) { 9284ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no stream_endpoint with seid %d found", local_seid); 9294ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 9304ccacc40SMilanka Ringwald } 9314ccacc40SMilanka Ringwald 9324ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 9334ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no media connection for stream_endpoint with seid %d found", local_seid); 9344ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 9354ccacc40SMilanka Ringwald } 936485c0a4cSMilanka Ringwald 937485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->suspend_stream){ 938485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 939485c0a4cSMilanka Ringwald } 9404ccacc40SMilanka Ringwald 94160ec20d0SMilanka Ringwald stream_endpoint->suspend_stream = 1; 9424ccacc40SMilanka Ringwald connection->local_seid = local_seid; 943c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 9449413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 9454ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 946747ec646SMilanka Ringwald } 947747ec646SMilanka Ringwald 9489974aee0SMilanka Ringwald uint8_t avdtp_discover_stream_endpoints(uint16_t avdtp_cid, avdtp_context_t * context){ 9494ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 950747ec646SMilanka Ringwald if (!connection){ 9518587e32cSMilanka Ringwald log_error("avdtp_discover_stream_endpoints: no connection for signaling cid 0x%02x found", avdtp_cid); 9529974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 9539974aee0SMilanka Ringwald } 9540e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 955c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9569974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 957747ec646SMilanka Ringwald } 958ec3d71e3SMilanka Ringwald 959747ec646SMilanka Ringwald connection->initiator_transaction_label++; 960747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_DISCOVER_SEPS; 9619974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 962747ec646SMilanka Ringwald } 963747ec646SMilanka Ringwald 964747ec646SMilanka Ringwald 9659974aee0SMilanka Ringwald uint8_t avdtp_get_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9664ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 967747ec646SMilanka Ringwald if (!connection){ 9689900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9699974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 970747ec646SMilanka Ringwald } 9710e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 972c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9739974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9749974aee0SMilanka Ringwald } 9759974aee0SMilanka Ringwald 976747ec646SMilanka Ringwald connection->initiator_transaction_label++; 977747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CAPABILITIES; 9789413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9799974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 980747ec646SMilanka Ringwald } 981747ec646SMilanka Ringwald 982747ec646SMilanka Ringwald 9839974aee0SMilanka Ringwald uint8_t avdtp_get_all_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9844ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 985747ec646SMilanka Ringwald if (!connection){ 9869900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9879974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 988747ec646SMilanka Ringwald } 9890e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 990c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9919974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9929974aee0SMilanka Ringwald } 9939974aee0SMilanka Ringwald 994747ec646SMilanka Ringwald connection->initiator_transaction_label++; 995747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_ALL_CAPABILITIES; 9969413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9979974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 998747ec646SMilanka Ringwald } 999747ec646SMilanka Ringwald 10009974aee0SMilanka Ringwald uint8_t avdtp_get_configuration(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 10014ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1002747ec646SMilanka Ringwald if (!connection){ 10039900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10049974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1005747ec646SMilanka Ringwald } 10060e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1007c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 10089974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10099974aee0SMilanka Ringwald } 10109974aee0SMilanka Ringwald 1011747ec646SMilanka Ringwald connection->initiator_transaction_label++; 1012747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CONFIGURATION; 10139413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10149974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1015747ec646SMilanka Ringwald } 1016747ec646SMilanka Ringwald 10179974aee0SMilanka Ringwald uint8_t avdtp_set_configuration(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, uint16_t configured_services_bitmap, avdtp_capabilities_t configuration, avdtp_context_t * context){ 10184ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1019747ec646SMilanka Ringwald if (!connection){ 10209900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10219974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1022747ec646SMilanka Ringwald } 10230e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1024c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 1025485c0a4cSMilanka Ringwald log_error("connection in wrong state, %d, initiator state %d", connection->state, connection->initiator_connection_state); 10269974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10279974aee0SMilanka Ringwald } 1028*a3ce0109SMatthias Ringwald if (connection->configuration_state != AVDTP_CONFIGURATION_STATE_IDLE){ 1029*a3ce0109SMatthias Ringwald log_info("configuration already started, config state %u", connection->configuration_state); 1030*a3ce0109SMatthias Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 1031*a3ce0109SMatthias Ringwald } 1032747ec646SMilanka Ringwald 10334ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 1034747ec646SMilanka Ringwald if (!stream_endpoint) { 10359900b7faSMilanka Ringwald log_error("No initiator stream endpoint for seid %d", local_seid); 10369974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST; 1037747ec646SMilanka Ringwald } 1038417b4996SMilanka Ringwald if (stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_CONFIGURED){ 1039485c0a4cSMilanka Ringwald log_error("Stream endpoint seid %d in wrong state %d", local_seid, stream_endpoint->state); 1040417b4996SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 1041417b4996SMilanka Ringwald } 1042*a3ce0109SMatthias Ringwald 1043bdbc3ef6SMilanka Ringwald connection->active_stream_endpoint = (void*) stream_endpoint; 1044*a3ce0109SMatthias Ringwald connection->configuration_state = AVDTP_CONFIGURATION_STATE_LOCAL_INITIATED; 1045747ec646SMilanka Ringwald 1046747ec646SMilanka Ringwald connection->initiator_transaction_label++; 10479413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10484ccacc40SMilanka Ringwald connection->local_seid = local_seid; 1049f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 1050f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 1051747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION; 1052ffa6c160SMilanka Ringwald 1053ffa6c160SMilanka Ringwald // cache media codec information for SBC 1054ffa6c160SMilanka Ringwald stream_endpoint->media_codec_type = configuration.media_codec.media_codec_type; 1055ffa6c160SMilanka Ringwald if (configuration.media_codec.media_codec_type == AVDTP_CODEC_SBC){ 1056ffa6c160SMilanka Ringwald stream_endpoint->media_type = configuration.media_codec.media_type; 10576535961aSMatthias Ringwald (void)memcpy(stream_endpoint->media_codec_sbc_info, 10586535961aSMatthias Ringwald configuration.media_codec.media_codec_information, 4); 1059ffa6c160SMilanka Ringwald } 10609974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1061747ec646SMilanka Ringwald } 1062747ec646SMilanka Ringwald 10639974aee0SMilanka Ringwald uint8_t avdtp_reconfigure(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, uint16_t configured_services_bitmap, avdtp_capabilities_t configuration, avdtp_context_t * context){ 10644ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1065747ec646SMilanka Ringwald if (!connection){ 10669900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10679974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1068747ec646SMilanka Ringwald } 1069747ec646SMilanka Ringwald //TODO: if opened only app capabilities, enable reconfigure for not opened 10700e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1071c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 10729974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10739974aee0SMilanka Ringwald } 10749e42cfccSMilanka Ringwald 10754ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 107678d08d09SMilanka Ringwald if (!stream_endpoint) { 10774ccacc40SMilanka Ringwald log_error("avdtp_reconfigure: no initiator stream endpoint for seid %d", local_seid); 10789974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST; 107978d08d09SMilanka Ringwald } 108078d08d09SMilanka Ringwald 1081485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint)){ 10828587e32cSMilanka Ringwald log_error("avdtp_reconfigure: no associated remote sep"); 10839974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 108478d08d09SMilanka Ringwald } 1085485c0a4cSMilanka Ringwald 1086747ec646SMilanka Ringwald connection->initiator_transaction_label++; 10879413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10886b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 1089f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 1090f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 1091747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_RECONFIGURE_STREAM_WITH_SEID; 10929974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1093747ec646SMilanka Ringwald } 1094747ec646SMilanka Ringwald 10958e7044f9SMatthias Ringwald void avdtp_set_preferred_sampling_frequeny(avdtp_stream_endpoint_t * stream_endpoint, uint32_t sampling_frequency){ 10968e7044f9SMatthias Ringwald stream_endpoint->preferred_sampling_frequency = sampling_frequency; 10978e7044f9SMatthias Ringwald } 10988e7044f9SMatthias Ringwald 109978d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_channel_mode(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_channel_mode_bitmap){ 110078d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 110178d08d09SMilanka Ringwald uint8_t channel_mode_bitmap = (media_codec[0] & 0x0F) & remote_channel_mode_bitmap; 110278d08d09SMilanka Ringwald 110378d08d09SMilanka Ringwald uint8_t channel_mode = AVDTP_SBC_STEREO; 110478d08d09SMilanka Ringwald if (channel_mode_bitmap & AVDTP_SBC_JOINT_STEREO){ 110578d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_JOINT_STEREO; 110678d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_STEREO){ 110778d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_STEREO; 110878d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_DUAL_CHANNEL){ 110978d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_DUAL_CHANNEL; 111078d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_MONO){ 111178d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_MONO; 111278d08d09SMilanka Ringwald } 111378d08d09SMilanka Ringwald return channel_mode; 111478d08d09SMilanka Ringwald } 111578d08d09SMilanka Ringwald 111678d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_allocation_method(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_allocation_method_bitmap){ 111778d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 111878d08d09SMilanka Ringwald uint8_t allocation_method_bitmap = (media_codec[1] & 0x03) & remote_allocation_method_bitmap; 111978d08d09SMilanka Ringwald 112078d08d09SMilanka Ringwald uint8_t allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 112178d08d09SMilanka Ringwald if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS){ 112278d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 112378d08d09SMilanka Ringwald } else if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_SNR){ 112478d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_SNR; 112578d08d09SMilanka Ringwald } 112678d08d09SMilanka Ringwald return allocation_method; 112778d08d09SMilanka Ringwald } 112878d08d09SMilanka Ringwald 1129bd1ecb8aSMilanka Ringwald uint8_t avdtp_stream_endpoint_seid(avdtp_stream_endpoint_t * stream_endpoint){ 1130bd1ecb8aSMilanka Ringwald if (!stream_endpoint) return 0; 1131bd1ecb8aSMilanka Ringwald return stream_endpoint->sep.seid; 1132bd1ecb8aSMilanka Ringwald } 113378d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_subbands(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_subbands_bitmap){ 113467ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 113578d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 113678d08d09SMilanka Ringwald uint8_t subbands_bitmap = ((media_codec[1] >> 2) & 0x03) & remote_subbands_bitmap; 113778d08d09SMilanka Ringwald 113878d08d09SMilanka Ringwald uint8_t subbands = AVDTP_SBC_SUBBANDS_8; 113978d08d09SMilanka Ringwald if (subbands_bitmap & AVDTP_SBC_SUBBANDS_8){ 114078d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_8; 114178d08d09SMilanka Ringwald } else if (subbands_bitmap & AVDTP_SBC_SUBBANDS_4){ 114278d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_4; 114378d08d09SMilanka Ringwald } 114478d08d09SMilanka Ringwald return subbands; 114578d08d09SMilanka Ringwald } 114678d08d09SMilanka Ringwald 114778d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_block_length(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_block_length_bitmap){ 114867ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 114978d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 115078d08d09SMilanka Ringwald uint8_t block_length_bitmap = (media_codec[1] >> 4) & remote_block_length_bitmap; 115178d08d09SMilanka Ringwald 115278d08d09SMilanka Ringwald uint8_t block_length = AVDTP_SBC_BLOCK_LENGTH_16; 115378d08d09SMilanka Ringwald if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_16){ 115478d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_16; 115578d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_12){ 115678d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_12; 115778d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_8){ 115878d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_8; 115978d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_4){ 116078d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_4; 116178d08d09SMilanka Ringwald } 116278d08d09SMilanka Ringwald return block_length; 116378d08d09SMilanka Ringwald } 116478d08d09SMilanka Ringwald 116578d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_sampling_frequency(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_sampling_frequency_bitmap){ 116667ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 116778d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 11688e7044f9SMatthias Ringwald uint8_t supported_sampling_frequency_bitmap = (media_codec[0] >> 4) & remote_sampling_frequency_bitmap; 11698e7044f9SMatthias Ringwald uint8_t sampling_frequency = AVDTP_SBC_44100; // some default 117078d08d09SMilanka Ringwald 11718e7044f9SMatthias Ringwald // use preferred sampling frequency if possible 11728e7044f9SMatthias Ringwald if ((stream_endpoint->preferred_sampling_frequency == 48000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_48000)){ 117378d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_48000; 11748e7044f9SMatthias Ringwald } else if ((stream_endpoint->preferred_sampling_frequency == 44100) && (supported_sampling_frequency_bitmap & AVDTP_SBC_44100)){ 117578d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_44100; 11768e7044f9SMatthias Ringwald } else if ((stream_endpoint->preferred_sampling_frequency == 32000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_32000)){ 117778d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_32000; 11788e7044f9SMatthias Ringwald } else if ((stream_endpoint->preferred_sampling_frequency == 16000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_16000)){ 11798e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_16000; 11808e7044f9SMatthias Ringwald } 11818e7044f9SMatthias Ringwald // otherwise, use highest available 11828e7044f9SMatthias Ringwald else if (supported_sampling_frequency_bitmap & AVDTP_SBC_48000){ 11838e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_48000; 11848e7044f9SMatthias Ringwald } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_44100){ 11858e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_44100; 11868e7044f9SMatthias Ringwald } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_32000){ 11878e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_32000; 11888e7044f9SMatthias Ringwald } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_16000){ 118978d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_16000; 119078d08d09SMilanka Ringwald } 119178d08d09SMilanka Ringwald return sampling_frequency; 119278d08d09SMilanka Ringwald } 119378d08d09SMilanka Ringwald 119478d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_max_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_max_bitpool_value){ 119567ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 119678d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 119778d08d09SMilanka Ringwald return btstack_min(media_codec[3], remote_max_bitpool_value); 119878d08d09SMilanka Ringwald } 119978d08d09SMilanka Ringwald 120078d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_min_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_min_bitpool_value){ 120167ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 120278d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 120378d08d09SMilanka Ringwald return btstack_max(media_codec[2], remote_min_bitpool_value); 1204747ec646SMilanka Ringwald } 1205485c0a4cSMilanka Ringwald 1206485c0a4cSMilanka Ringwald uint8_t is_avdtp_remote_seid_registered(avdtp_stream_endpoint_t * stream_endpoint){ 1207485c0a4cSMilanka Ringwald if (!stream_endpoint) return 0; 1208485c0a4cSMilanka Ringwald if (stream_endpoint->remote_sep.seid == 0) return 0; 1209485c0a4cSMilanka Ringwald if (stream_endpoint->remote_sep.seid > 0x3E) return 0; 1210485c0a4cSMilanka Ringwald return 1; 1211485c0a4cSMilanka Ringwald } 1212