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 static void handle_l2cap_data_packet_for_signaling_connection(avdtp_connection_t * connection, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 366*c1c40ea1SMatthias Ringwald if (size < 2) return; 367*c1c40ea1SMatthias Ringwald 368*c1c40ea1SMatthias Ringwald uint16_t offset; 369*c1c40ea1SMatthias Ringwald avdtp_message_type_t message_type = avdtp_get_signaling_packet_type(packet); 370*c1c40ea1SMatthias Ringwald switch (message_type){ 371747ec646SMilanka Ringwald case AVDTP_CMD_MSG: 372*c1c40ea1SMatthias Ringwald offset = avdtp_read_signaling_header(&connection->signaling_packet, packet, size); 373747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm(connection, packet, size, offset, context); 374747ec646SMilanka Ringwald break; 375747ec646SMilanka Ringwald default: 376*c1c40ea1SMatthias Ringwald offset = avdtp_read_signaling_header(&connection->signaling_packet, packet, size); 377747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm(connection, packet, size, offset, context); 378747ec646SMilanka Ringwald break; 379747ec646SMilanka Ringwald } 380747ec646SMilanka Ringwald } 381747ec646SMilanka Ringwald 382692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 3832f6083d0SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(sdp_query_context->avdtp_cid, sdp_query_context); 3842f6083d0SMilanka Ringwald if (!connection) { 385355ac553SMilanka Ringwald log_error("SDP query, connection with 0x%02x cid not found", sdp_query_context->avdtp_cid); 3862f6083d0SMilanka Ringwald return; 3872f6083d0SMilanka Ringwald } 3882f6083d0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE) return; 3892f6083d0SMilanka Ringwald 390692c0605SMilanka Ringwald UNUSED(packet_type); 391692c0605SMilanka Ringwald UNUSED(channel); 392692c0605SMilanka Ringwald UNUSED(size); 393692c0605SMilanka Ringwald 394692c0605SMilanka Ringwald des_iterator_t des_list_it; 395692c0605SMilanka Ringwald des_iterator_t prot_it; 3961e1ae2bcSMilanka Ringwald uint8_t status; 397692c0605SMilanka Ringwald 398692c0605SMilanka Ringwald switch (hci_event_packet_get_type(packet)){ 399692c0605SMilanka Ringwald case SDP_EVENT_QUERY_ATTRIBUTE_VALUE: 400692c0605SMilanka Ringwald // Handle new SDP record 401692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_record_id(packet) != record_id) { 402692c0605SMilanka Ringwald record_id = sdp_event_query_attribute_byte_get_record_id(packet); 4038587e32cSMilanka Ringwald // log_info("SDP Record: Nr: %d", record_id); 404692c0605SMilanka Ringwald } 405692c0605SMilanka Ringwald 406692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_attribute_length(packet) <= attribute_value_buffer_size) { 407692c0605SMilanka Ringwald attribute_value[sdp_event_query_attribute_byte_get_data_offset(packet)] = sdp_event_query_attribute_byte_get_data(packet); 408692c0605SMilanka Ringwald 409692c0605SMilanka Ringwald if ((uint16_t)(sdp_event_query_attribute_byte_get_data_offset(packet)+1) == sdp_event_query_attribute_byte_get_attribute_length(packet)) { 410692c0605SMilanka Ringwald 411692c0605SMilanka Ringwald switch(sdp_event_query_attribute_byte_get_attribute_id(packet)) { 412692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_SERVICE_CLASS_ID_LIST: 413692c0605SMilanka Ringwald if (de_get_element_type(attribute_value) != DE_DES) break; 414692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 415692c0605SMilanka Ringwald uint8_t * element = des_iterator_get_element(&des_list_it); 416692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 417692c0605SMilanka Ringwald uint32_t uuid = de_get_uuid32(element); 418692c0605SMilanka Ringwald switch (uuid){ 419692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SOURCE: 4202f6083d0SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SOURCE) { 4212f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 422692c0605SMilanka Ringwald break; 423692c0605SMilanka Ringwald } 4248587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SOURCE protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 425eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 426692c0605SMilanka Ringwald break; 427692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SINK: 4285448c259SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SINK) { 4292f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 430692c0605SMilanka Ringwald break; 431692c0605SMilanka Ringwald } 4328587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SINK protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 433eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 434692c0605SMilanka Ringwald break; 435692c0605SMilanka Ringwald default: 436692c0605SMilanka Ringwald break; 437692c0605SMilanka Ringwald } 438692c0605SMilanka Ringwald } 439692c0605SMilanka Ringwald break; 440692c0605SMilanka Ringwald 441692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_PROTOCOL_DESCRIPTOR_LIST: { 4428587e32cSMilanka Ringwald // log_info("SDP Attribute: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet)); 443692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 444692c0605SMilanka Ringwald uint8_t *des_element; 445692c0605SMilanka Ringwald uint8_t *element; 446692c0605SMilanka Ringwald uint32_t uuid; 447692c0605SMilanka Ringwald 448692c0605SMilanka Ringwald if (des_iterator_get_type(&des_list_it) != DE_DES) continue; 449692c0605SMilanka Ringwald 450692c0605SMilanka Ringwald des_element = des_iterator_get_element(&des_list_it); 451692c0605SMilanka Ringwald des_iterator_init(&prot_it, des_element); 452692c0605SMilanka Ringwald element = des_iterator_get_element(&prot_it); 453692c0605SMilanka Ringwald 454692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 455692c0605SMilanka Ringwald 456692c0605SMilanka Ringwald uuid = de_get_uuid32(element); 45714fd128cSMatthias Ringwald des_iterator_next(&prot_it); 458692c0605SMilanka Ringwald switch (uuid){ 459692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_L2CAP: 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_l2cap_psm); 462692c0605SMilanka Ringwald break; 463692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_AVDTP: 464692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 4652f6083d0SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_version); 466692c0605SMilanka Ringwald break; 467692c0605SMilanka Ringwald default: 468692c0605SMilanka Ringwald break; 469692c0605SMilanka Ringwald } 470692c0605SMilanka Ringwald } 471692c0605SMilanka Ringwald } 472692c0605SMilanka Ringwald break; 473692c0605SMilanka Ringwald default: 474692c0605SMilanka Ringwald break; 475692c0605SMilanka Ringwald } 476692c0605SMilanka Ringwald } 477692c0605SMilanka Ringwald } else { 4788587e32cSMilanka 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)); 479692c0605SMilanka Ringwald } 480692c0605SMilanka Ringwald break; 481692c0605SMilanka Ringwald 482692c0605SMilanka Ringwald case SDP_EVENT_QUERY_COMPLETE: 4830d4a198eSMatthias Ringwald if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE){ 4840d4a198eSMatthias Ringwald // bail out, we must have had an incoming connection in the meantime; 4850d4a198eSMatthias Ringwald break; 4860d4a198eSMatthias Ringwald } 4871e1ae2bcSMilanka Ringwald status = sdp_event_query_complete_get_status(packet); 4882f6083d0SMilanka Ringwald if (status != ERROR_CODE_SUCCESS){ 4895448c259SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, status); 4902f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4912f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 492355ac553SMilanka Ringwald log_info("SDP query failed with status 0x%02x.", status); 4931e1ae2bcSMilanka Ringwald break; 4941e1ae2bcSMilanka Ringwald } 4952f6083d0SMilanka Ringwald if (!sdp_query_context->role_supported){ 4962f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4972f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 4982f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, SDP_SERVICE_NOT_FOUND); 499355ac553SMilanka Ringwald log_info("SDP query, remote device does not support required role."); 500974d4d6eSMilanka Ringwald break; 501974d4d6eSMilanka Ringwald } 5022f6083d0SMilanka Ringwald if (!sdp_query_context->avdtp_l2cap_psm) { 5032f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*)connection); 5042f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 5052f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, L2CAP_SERVICE_DOES_NOT_EXIST); 506355ac553SMilanka Ringwald log_info("SDP query, no l2cap psm found."); 5072f6083d0SMilanka Ringwald break; 5082f6083d0SMilanka Ringwald } 5092f6083d0SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 5102f6083d0SMilanka Ringwald l2cap_create_channel(sdp_query_context->packet_handler, connection->remote_addr, sdp_query_context->avdtp_l2cap_psm, l2cap_max_mtu(), NULL); 511692c0605SMilanka Ringwald break; 512692c0605SMilanka Ringwald } 513692c0605SMilanka Ringwald } 514692c0605SMilanka Ringwald 515692c0605SMilanka Ringwald 516747ec646SMilanka Ringwald void avdtp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 517747ec646SMilanka Ringwald bd_addr_t event_addr; 518747ec646SMilanka Ringwald uint16_t psm; 519747ec646SMilanka Ringwald uint16_t local_cid; 5201e1ae2bcSMilanka Ringwald uint8_t status; 5210d4a198eSMatthias Ringwald int accept_signaling_connection; 522747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = NULL; 523747ec646SMilanka Ringwald avdtp_connection_t * connection = NULL; 524747ec646SMilanka Ringwald btstack_linked_list_t * avdtp_connections = &context->connections; 525747ec646SMilanka Ringwald btstack_linked_list_t * stream_endpoints = &context->stream_endpoints; 526747ec646SMilanka Ringwald handle_media_data = context->handle_media_data; 5278587e32cSMilanka Ringwald // log_info("avdtp_packet_handler packet type %02x, event %02x ", packet_type, hci_event_packet_get_type(packet)); 528747ec646SMilanka Ringwald switch (packet_type) { 529747ec646SMilanka Ringwald case L2CAP_DATA_PACKET: 5309413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 531747ec646SMilanka Ringwald if (connection){ 532747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 533747ec646SMilanka Ringwald break; 534747ec646SMilanka Ringwald } 535747ec646SMilanka Ringwald 536747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 537747ec646SMilanka Ringwald if (!stream_endpoint){ 538747ec646SMilanka Ringwald if (!connection) break; 539747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 540747ec646SMilanka Ringwald break; 541747ec646SMilanka Ringwald } 542747ec646SMilanka Ringwald 5438c0f3635SMilanka Ringwald if (stream_endpoint->connection){ 5449413b167SMilanka Ringwald if (channel == stream_endpoint->connection->l2cap_signaling_cid){ 545025463b2SMatthias Ringwald handle_l2cap_data_packet_for_signaling_connection(stream_endpoint->connection, packet, size, context); 546747ec646SMilanka Ringwald break; 547747ec646SMilanka Ringwald } 5488c0f3635SMilanka Ringwald } 549747ec646SMilanka Ringwald 550747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_media_cid){ 5518b097e29SMilanka Ringwald if (handle_media_data){ 552fd58c900SMilanka Ringwald (*handle_media_data)(avdtp_local_seid(stream_endpoint), packet, size); 5538b097e29SMilanka Ringwald } 554747ec646SMilanka Ringwald break; 555747ec646SMilanka Ringwald } 556747ec646SMilanka Ringwald 557747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_reporting_cid){ 558747ec646SMilanka Ringwald // TODO 5598587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for reporting: NOT IMPLEMENTED"); 560747ec646SMilanka Ringwald } else if (channel == stream_endpoint->l2cap_recovery_cid){ 561747ec646SMilanka Ringwald // TODO 5628587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for recovery: NOT IMPLEMENTED"); 563747ec646SMilanka Ringwald } else { 564747ec646SMilanka Ringwald log_error("avdtp packet handler L2CAP_DATA_PACKET: local cid 0x%02x not found", channel); 565747ec646SMilanka Ringwald } 566747ec646SMilanka Ringwald break; 567747ec646SMilanka Ringwald 568747ec646SMilanka Ringwald case HCI_EVENT_PACKET: 569747ec646SMilanka Ringwald switch (hci_event_packet_get_type(packet)) { 570747ec646SMilanka Ringwald case L2CAP_EVENT_INCOMING_CONNECTION: 571747ec646SMilanka Ringwald l2cap_event_incoming_connection_get_address(packet, event_addr); 572747ec646SMilanka Ringwald local_cid = l2cap_event_incoming_connection_get_local_cid(packet); 573747ec646SMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 574355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION, local cid 0x%02x ", local_cid); 5750d4a198eSMatthias Ringwald if (!connection){ 5760d4a198eSMatthias Ringwald // create connection struct for regular inconming connection request 577747ec646SMilanka Ringwald connection = avdtp_create_connection(event_addr, context); 5780d4a198eSMatthias Ringwald if (!connection){ 5790d4a198eSMatthias Ringwald log_error("Could not create connection, reject"); 5800d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 5810d4a198eSMatthias Ringwald break; 5820d4a198eSMatthias Ringwald } 5830d4a198eSMatthias Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 5840d4a198eSMatthias Ringwald } 5850d4a198eSMatthias Ringwald 5860d4a198eSMatthias Ringwald // connection exists, check states 5870d4a198eSMatthias Ringwald log_info("Connection state %d", connection->state); 588cf837c28SMatthias Ringwald accept_signaling_connection = 0; 5890d4a198eSMatthias Ringwald switch (connection->state){ 5900d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_IDLE: 5910d4a198eSMatthias Ringwald // regular incoming connection 5920d4a198eSMatthias Ringwald accept_signaling_connection = 1; 5930d4a198eSMatthias Ringwald break; 5940d4a198eSMatthias Ringwald case AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE: 5950d4a198eSMatthias Ringwald // incoming connection during sdp query, just accept it 5960d4a198eSMatthias Ringwald accept_signaling_connection = 1; 5970d4a198eSMatthias Ringwald break; 5980d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED: 5990d4a198eSMatthias Ringwald log_info("Reject incoming connection after creating outgoing"); 6000d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 6010d4a198eSMatthias Ringwald return; 6020d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_OPENED: 6030d4a198eSMatthias Ringwald // handled below 6040d4a198eSMatthias Ringwald break; 6050d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED: 6060d4a198eSMatthias Ringwald log_info("Reject incoming connection during disconnect"); 6070d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 6080d4a198eSMatthias Ringwald return; 6090d4a198eSMatthias Ringwald } 6100d4a198eSMatthias Ringwald 6110d4a198eSMatthias Ringwald if (accept_signaling_connection){ 612747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 613355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION, connection %p, state connection %d, avdtp cid 0x%02x", connection, connection->state, connection->avdtp_cid); 614747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 615747ec646SMilanka Ringwald break; 616747ec646SMilanka Ringwald } 617747ec646SMilanka Ringwald 6180d4a198eSMatthias Ringwald // now, we're only dealing with media connections 6196b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 620747ec646SMilanka Ringwald if (!stream_endpoint) { 621355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION no streamendpoint found for seid %d", connection->local_seid); 6220d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 623747ec646SMilanka Ringwald break; 624747ec646SMilanka Ringwald } 625747ec646SMilanka Ringwald 626a3ce0109SMatthias 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); 627a3ce0109SMatthias Ringwald if (stream_endpoint->l2cap_media_cid != 0){ 628a3ce0109SMatthias Ringwald l2cap_decline_connection(local_cid); 629a3ce0109SMatthias Ringwald break; 630a3ce0109SMatthias Ringwald } 631a3ce0109SMatthias Ringwald if (connection->configuration_state != AVDTP_CONFIGURATION_STATE_REMOTE_CONFIGURED){ 6329900b7faSMilanka Ringwald l2cap_decline_connection(local_cid); 6339900b7faSMilanka Ringwald break; 6349900b7faSMilanka Ringwald } 635747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 636747ec646SMilanka Ringwald break; 637747ec646SMilanka Ringwald 638747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_OPENED: 639a0b8a58cSMilanka Ringwald psm = l2cap_event_channel_opened_get_psm(packet); 64084e3541eSMilanka Ringwald if (psm != BLUETOOTH_PSM_AVDTP){ 641355ac553SMilanka Ringwald log_info("Unexpected PSM - Not implemented yet, avdtp sink: L2CAP_EVENT_CHANNEL_OPENED "); 642a0b8a58cSMilanka Ringwald return; 643a0b8a58cSMilanka Ringwald } 644a0b8a58cSMilanka Ringwald 6451e1ae2bcSMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 646747ec646SMilanka Ringwald // inform about new l2cap connection 647747ec646SMilanka Ringwald l2cap_event_channel_opened_get_address(packet, event_addr); 6487050d2caSMilanka Ringwald local_cid = l2cap_event_channel_opened_get_local_cid(packet); 649a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 650a0b8a58cSMilanka Ringwald 651355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED: status %d, cid 0x%02x , signaling connection %p ", status, local_cid, connection); 652a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 653a0b8a58cSMilanka Ringwald if (!connection){ 654355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED 2: status %d, signaling connection %p ", status, connection); 655a0b8a58cSMilanka Ringwald break; 656a0b8a58cSMilanka Ringwald } 657a0b8a58cSMilanka Ringwald 658a0b8a58cSMilanka Ringwald if (connection->l2cap_signaling_cid == 0) { 6591e1ae2bcSMilanka Ringwald if (status){ 660355ac553SMilanka Ringwald log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 66155ddebccSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 662a0b8a58cSMilanka Ringwald connection->l2cap_signaling_cid = 0; 6634ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, l2cap_event_channel_opened_get_status(packet)); 664747ec646SMilanka Ringwald break; 665747ec646SMilanka Ringwald } 666a0b8a58cSMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED) { 667355ac553SMilanka Ringwald log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 668a0b8a58cSMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, AVDTP_CONNECTION_IN_WRONG_STATE); 669a0b8a58cSMilanka Ringwald break; 67055ddebccSMilanka Ringwald } 6719413b167SMilanka Ringwald connection->l2cap_signaling_cid = local_cid; 6726b0ee1d0SMilanka Ringwald connection->local_seid = 0; 673747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_OPENED; 674355ac553SMilanka Ringwald log_info("AVDTP_SIGNALING_CONNECTION_OPENED, connection %p, l2cap_signaling_cid 0x%02x, avdtp_cid 0x%02x", connection, local_cid, connection->avdtp_cid); 6754ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, 0); 676747ec646SMilanka Ringwald break; 677747ec646SMilanka Ringwald } 678747ec646SMilanka Ringwald 6796b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 680747ec646SMilanka Ringwald if (!stream_endpoint){ 681355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED: stream_endpoint not found"); 682747ec646SMilanka Ringwald return; 683747ec646SMilanka Ringwald } 684a466d508SMilanka Ringwald 685a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 686a466d508SMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 687a466d508SMilanka Ringwald if (status){ 688355ac553SMilanka 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)); 689a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_IDLE; 690596b7fdcSMilanka 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); 691a466d508SMilanka Ringwald break; 692a466d508SMilanka Ringwald } 693a466d508SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED){ 694355ac553SMilanka 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)); 695596b7fdcSMilanka 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); 696a466d508SMilanka Ringwald break; 697a466d508SMilanka Ringwald } 698a466d508SMilanka Ringwald 699a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 700a466d508SMilanka Ringwald stream_endpoint->connection = connection; 701a466d508SMilanka Ringwald stream_endpoint->l2cap_media_cid = l2cap_event_channel_opened_get_local_cid(packet); 702a466d508SMilanka Ringwald stream_endpoint->media_con_handle = l2cap_event_channel_opened_get_handle(packet); 703d1207cd8SMilanka Ringwald 704355ac553SMilanka 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)); 705596b7fdcSMilanka 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); 706a466d508SMilanka Ringwald return; 707a466d508SMilanka Ringwald } 708747ec646SMilanka Ringwald break; 709747ec646SMilanka Ringwald 710747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_CLOSED: 711747ec646SMilanka Ringwald local_cid = l2cap_event_channel_closed_get_local_cid(packet); 7129413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(local_cid, context); 7130bd7cb1fSMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(local_cid, context); 714485c0a4cSMilanka Ringwald log_info("Received L2CAP_EVENT_CHANNEL_CLOSED, cid 0x%2x, connection %p, stream_endpoint %p", local_cid, connection, stream_endpoint); 7150bd7cb1fSMilanka Ringwald 716a466d508SMilanka Ringwald if (stream_endpoint){ 717a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == local_cid){ 718a466d508SMilanka Ringwald connection = stream_endpoint->connection; 719596b7fdcSMilanka Ringwald if (connection) { 720596b7fdcSMilanka Ringwald avdtp_streaming_emit_connection_released(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint)); 721596b7fdcSMilanka Ringwald } 722485c0a4cSMilanka Ringwald avdtp_reset_stream_endpoint(stream_endpoint); 723a466d508SMilanka Ringwald if (connection && connection->disconnect){ 724a466d508SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 725a466d508SMilanka Ringwald } 726a466d508SMilanka Ringwald break; 727a466d508SMilanka Ringwald } 728a466d508SMilanka Ringwald if (stream_endpoint->l2cap_recovery_cid == local_cid){ 729355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_CLOSED recovery cid 0x%0x", local_cid); 730a466d508SMilanka Ringwald stream_endpoint->l2cap_recovery_cid = 0; 731a466d508SMilanka Ringwald break; 732a466d508SMilanka Ringwald } 733a466d508SMilanka Ringwald 734a466d508SMilanka Ringwald if (stream_endpoint->l2cap_reporting_cid == local_cid){ 735355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_CLOSED reporting cid 0x%0x", local_cid); 736a466d508SMilanka Ringwald stream_endpoint->l2cap_reporting_cid = 0; 737a466d508SMilanka Ringwald break; 738a466d508SMilanka Ringwald } 739a466d508SMilanka Ringwald } 740596b7fdcSMilanka Ringwald 741596b7fdcSMilanka Ringwald if (connection){ 742596b7fdcSMilanka Ringwald btstack_linked_list_iterator_t it; 743596b7fdcSMilanka Ringwald btstack_linked_list_iterator_init(&it, stream_endpoints); 744596b7fdcSMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 745596b7fdcSMilanka Ringwald avdtp_stream_endpoint_t * _stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 746596b7fdcSMilanka Ringwald if (_stream_endpoint->connection == connection){ 747485c0a4cSMilanka Ringwald avdtp_reset_stream_endpoint(_stream_endpoint); 748596b7fdcSMilanka Ringwald } 749596b7fdcSMilanka Ringwald } 750485c0a4cSMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 751485c0a4cSMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 752485c0a4cSMilanka Ringwald btstack_linked_list_remove(avdtp_connections, (btstack_linked_item_t*) connection); 753596b7fdcSMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 754596b7fdcSMilanka Ringwald break; 755596b7fdcSMilanka Ringwald } 756596b7fdcSMilanka Ringwald 757747ec646SMilanka Ringwald break; 758747ec646SMilanka Ringwald 759747ec646SMilanka Ringwald case HCI_EVENT_DISCONNECTION_COMPLETE: 760747ec646SMilanka Ringwald break; 761747ec646SMilanka Ringwald 762747ec646SMilanka Ringwald case L2CAP_EVENT_CAN_SEND_NOW: 7639413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 764747ec646SMilanka Ringwald if (!connection) { 765747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 766747ec646SMilanka Ringwald if (!stream_endpoint->connection) break; 767747ec646SMilanka Ringwald connection = stream_endpoint->connection; 768747ec646SMilanka Ringwald } 769747ec646SMilanka Ringwald avdtp_handle_can_send_now(connection, channel, context); 770747ec646SMilanka Ringwald break; 771747ec646SMilanka Ringwald default: 772355ac553SMilanka Ringwald log_info("Unknown HCI event type %02x", hci_event_packet_get_type(packet)); 773747ec646SMilanka Ringwald break; 774747ec646SMilanka Ringwald } 775747ec646SMilanka Ringwald break; 776747ec646SMilanka Ringwald 777747ec646SMilanka Ringwald default: 778747ec646SMilanka Ringwald // other packet type 779747ec646SMilanka Ringwald break; 780747ec646SMilanka Ringwald } 781747ec646SMilanka Ringwald } 782747ec646SMilanka Ringwald 7834ccacc40SMilanka Ringwald uint8_t avdtp_disconnect(uint16_t avdtp_cid, avdtp_context_t * context){ 7844ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 7852ad6b656SMilanka Ringwald if (!connection) return AVDTP_CONNECTION_DOES_NOT_EXIST; 786a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_IDLE){ 787a466d508SMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 788a466d508SMilanka Ringwald return ERROR_CODE_SUCCESS; 789a466d508SMilanka Ringwald } 790a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED) return ERROR_CODE_SUCCESS; 791747ec646SMilanka Ringwald 792747ec646SMilanka Ringwald connection->disconnect = 1; 7939413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 7944ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 795747ec646SMilanka Ringwald } 796747ec646SMilanka Ringwald 7976b0ee1d0SMilanka Ringwald uint8_t avdtp_open_stream(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, avdtp_context_t * context){ 7984ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 799747ec646SMilanka Ringwald if (!connection){ 8008587e32cSMilanka Ringwald log_error("avdtp_media_connect: no connection for signaling cid 0x%02x found", avdtp_cid); 8014ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 802747ec646SMilanka Ringwald } 803747ec646SMilanka Ringwald 804747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) { 8058587e32cSMilanka Ringwald log_error("avdtp_media_connect: wrong connection state %d", connection->state); 8064ccacc40SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 807747ec646SMilanka Ringwald } 808747ec646SMilanka Ringwald 8096b0ee1d0SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 810747ec646SMilanka Ringwald if (!stream_endpoint) { 8116b0ee1d0SMilanka Ringwald log_error("avdtp_media_connect: no stream_endpoint with seid %d found", local_seid); 8124ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 813747ec646SMilanka Ringwald } 814747ec646SMilanka Ringwald 815485c0a4cSMilanka Ringwald if (stream_endpoint->remote_sep.seid != remote_seid){ 816485c0a4cSMilanka Ringwald log_error("avdtp_media_connect: no remote sep with seid %d registered with the stream endpoint", remote_seid); 817485c0a4cSMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 818485c0a4cSMilanka Ringwald } 819485c0a4cSMilanka Ringwald 8204ccacc40SMilanka Ringwald if (stream_endpoint->state < AVDTP_STREAM_ENDPOINT_CONFIGURED) return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 821747ec646SMilanka Ringwald 822747ec646SMilanka Ringwald connection->initiator_transaction_label++; 8239413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8246b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 825747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_OPEN_STREAM; 826747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W2_REQUEST_OPEN_STREAM; 8279413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8284ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 829747ec646SMilanka Ringwald } 830747ec646SMilanka Ringwald 8314ccacc40SMilanka Ringwald uint8_t avdtp_start_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8324ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 83346e6b063SMilanka Ringwald if (!connection){ 8344ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8354ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 83646e6b063SMilanka Ringwald } 8375cfe7f4cSMilanka Ringwald 8384ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8394ccacc40SMilanka Ringwald if (!stream_endpoint) { 8404ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no stream_endpoint with seid %d found", local_seid); 8414ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8424ccacc40SMilanka Ringwald } 8434ccacc40SMilanka Ringwald 8444ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8454ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8464ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8474ccacc40SMilanka Ringwald } 8484ccacc40SMilanka Ringwald 849485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint)){ 850485c0a4cSMilanka Ringwald log_error("avdtp_media_connect: no remote sep registered with the stream endpoint"); 851485c0a4cSMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8524ccacc40SMilanka Ringwald } 8534ccacc40SMilanka Ringwald 85460ec20d0SMilanka Ringwald stream_endpoint->start_stream = 1; 8554ccacc40SMilanka Ringwald connection->local_seid = local_seid; 856c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 8579413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8584ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 859747ec646SMilanka Ringwald } 860747ec646SMilanka Ringwald 8614ccacc40SMilanka Ringwald uint8_t avdtp_stop_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8624ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 863747ec646SMilanka Ringwald if (!connection){ 8644ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8654ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 866747ec646SMilanka Ringwald } 8674ccacc40SMilanka Ringwald 8684ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8694ccacc40SMilanka Ringwald if (!stream_endpoint) { 8704ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no stream_endpoint with seid %d found", local_seid); 8714ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8724ccacc40SMilanka Ringwald } 8734ccacc40SMilanka Ringwald 8744ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8754ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8764ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8774ccacc40SMilanka Ringwald } 878485c0a4cSMilanka Ringwald 879485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->stop_stream){ 880485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 881485c0a4cSMilanka Ringwald } 8824ccacc40SMilanka Ringwald 88360ec20d0SMilanka Ringwald stream_endpoint->stop_stream = 1; 8844ccacc40SMilanka Ringwald connection->local_seid = local_seid; 885c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 8869413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8874ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 888747ec646SMilanka Ringwald } 889747ec646SMilanka Ringwald 8904ccacc40SMilanka Ringwald uint8_t avdtp_abort_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8914ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 89260ec20d0SMilanka Ringwald if (!connection){ 8934ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8944ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 895747ec646SMilanka Ringwald } 8964ccacc40SMilanka Ringwald 8974ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8984ccacc40SMilanka Ringwald if (!stream_endpoint) { 8994ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no stream_endpoint with seid %d found", local_seid); 9004ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 9014ccacc40SMilanka Ringwald } 9024ccacc40SMilanka Ringwald 9034ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 9044ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no media connection for stream_endpoint with seid %d found", local_seid); 9054ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 9064ccacc40SMilanka Ringwald } 907485c0a4cSMilanka Ringwald 908485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->abort_stream){ 909485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 910485c0a4cSMilanka Ringwald } 9114ccacc40SMilanka Ringwald 91260ec20d0SMilanka Ringwald stream_endpoint->abort_stream = 1; 9136b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 914c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 9159413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 9164ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 917747ec646SMilanka Ringwald } 918747ec646SMilanka Ringwald 9194ccacc40SMilanka Ringwald uint8_t avdtp_suspend_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 9204ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 921747ec646SMilanka Ringwald if (!connection){ 9224ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 9234ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 92460ec20d0SMilanka Ringwald } 9254ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 9264ccacc40SMilanka Ringwald if (!stream_endpoint) { 9274ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no stream_endpoint with seid %d found", local_seid); 9284ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 9294ccacc40SMilanka Ringwald } 9304ccacc40SMilanka Ringwald 9314ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 9324ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no media connection for stream_endpoint with seid %d found", local_seid); 9334ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 9344ccacc40SMilanka Ringwald } 935485c0a4cSMilanka Ringwald 936485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->suspend_stream){ 937485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 938485c0a4cSMilanka Ringwald } 9394ccacc40SMilanka Ringwald 94060ec20d0SMilanka Ringwald stream_endpoint->suspend_stream = 1; 9414ccacc40SMilanka Ringwald connection->local_seid = local_seid; 942c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 9439413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 9444ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 945747ec646SMilanka Ringwald } 946747ec646SMilanka Ringwald 9479974aee0SMilanka Ringwald uint8_t avdtp_discover_stream_endpoints(uint16_t avdtp_cid, avdtp_context_t * context){ 9484ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 949747ec646SMilanka Ringwald if (!connection){ 9508587e32cSMilanka Ringwald log_error("avdtp_discover_stream_endpoints: no connection for signaling cid 0x%02x found", avdtp_cid); 9519974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 9529974aee0SMilanka Ringwald } 9530e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 954c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9559974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 956747ec646SMilanka Ringwald } 957ec3d71e3SMilanka Ringwald 958747ec646SMilanka Ringwald connection->initiator_transaction_label++; 959747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_DISCOVER_SEPS; 9609974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 961747ec646SMilanka Ringwald } 962747ec646SMilanka Ringwald 963747ec646SMilanka Ringwald 9649974aee0SMilanka Ringwald uint8_t avdtp_get_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9654ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 966747ec646SMilanka Ringwald if (!connection){ 9679900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9689974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 969747ec646SMilanka Ringwald } 9700e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 971c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9729974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9739974aee0SMilanka Ringwald } 9749974aee0SMilanka Ringwald 975747ec646SMilanka Ringwald connection->initiator_transaction_label++; 976747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CAPABILITIES; 9779413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9789974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 979747ec646SMilanka Ringwald } 980747ec646SMilanka Ringwald 981747ec646SMilanka Ringwald 9829974aee0SMilanka Ringwald uint8_t avdtp_get_all_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9834ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 984747ec646SMilanka Ringwald if (!connection){ 9859900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9869974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 987747ec646SMilanka Ringwald } 9880e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 989c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9909974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9919974aee0SMilanka Ringwald } 9929974aee0SMilanka Ringwald 993747ec646SMilanka Ringwald connection->initiator_transaction_label++; 994747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_ALL_CAPABILITIES; 9959413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9969974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 997747ec646SMilanka Ringwald } 998747ec646SMilanka Ringwald 9999974aee0SMilanka Ringwald uint8_t avdtp_get_configuration(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 10004ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1001747ec646SMilanka Ringwald if (!connection){ 10029900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10039974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1004747ec646SMilanka Ringwald } 10050e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1006c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 10079974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10089974aee0SMilanka Ringwald } 10099974aee0SMilanka Ringwald 1010747ec646SMilanka Ringwald connection->initiator_transaction_label++; 1011747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CONFIGURATION; 10129413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10139974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1014747ec646SMilanka Ringwald } 1015747ec646SMilanka Ringwald 10169974aee0SMilanka 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){ 10174ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1018747ec646SMilanka Ringwald if (!connection){ 10199900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10209974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1021747ec646SMilanka Ringwald } 10220e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1023c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 1024485c0a4cSMilanka Ringwald log_error("connection in wrong state, %d, initiator state %d", connection->state, connection->initiator_connection_state); 10259974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10269974aee0SMilanka Ringwald } 1027a3ce0109SMatthias Ringwald if (connection->configuration_state != AVDTP_CONFIGURATION_STATE_IDLE){ 1028a3ce0109SMatthias Ringwald log_info("configuration already started, config state %u", connection->configuration_state); 1029a3ce0109SMatthias Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 1030a3ce0109SMatthias Ringwald } 1031747ec646SMilanka Ringwald 10324ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 1033747ec646SMilanka Ringwald if (!stream_endpoint) { 10349900b7faSMilanka Ringwald log_error("No initiator stream endpoint for seid %d", local_seid); 10359974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST; 1036747ec646SMilanka Ringwald } 1037417b4996SMilanka Ringwald if (stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_CONFIGURED){ 1038485c0a4cSMilanka Ringwald log_error("Stream endpoint seid %d in wrong state %d", local_seid, stream_endpoint->state); 1039417b4996SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 1040417b4996SMilanka Ringwald } 1041a3ce0109SMatthias Ringwald 1042bdbc3ef6SMilanka Ringwald connection->active_stream_endpoint = (void*) stream_endpoint; 1043a3ce0109SMatthias Ringwald connection->configuration_state = AVDTP_CONFIGURATION_STATE_LOCAL_INITIATED; 1044747ec646SMilanka Ringwald 1045747ec646SMilanka Ringwald connection->initiator_transaction_label++; 10469413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10474ccacc40SMilanka Ringwald connection->local_seid = local_seid; 1048f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 1049f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 1050747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION; 1051ffa6c160SMilanka Ringwald 1052ffa6c160SMilanka Ringwald // cache media codec information for SBC 1053ffa6c160SMilanka Ringwald stream_endpoint->media_codec_type = configuration.media_codec.media_codec_type; 1054ffa6c160SMilanka Ringwald if (configuration.media_codec.media_codec_type == AVDTP_CODEC_SBC){ 1055ffa6c160SMilanka Ringwald stream_endpoint->media_type = configuration.media_codec.media_type; 10566535961aSMatthias Ringwald (void)memcpy(stream_endpoint->media_codec_sbc_info, 10576535961aSMatthias Ringwald configuration.media_codec.media_codec_information, 4); 1058ffa6c160SMilanka Ringwald } 10599974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1060747ec646SMilanka Ringwald } 1061747ec646SMilanka Ringwald 10629974aee0SMilanka 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){ 10634ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1064747ec646SMilanka Ringwald if (!connection){ 10659900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10669974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1067747ec646SMilanka Ringwald } 1068747ec646SMilanka Ringwald //TODO: if opened only app capabilities, enable reconfigure for not opened 10690e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1070c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 10719974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10729974aee0SMilanka Ringwald } 10739e42cfccSMilanka Ringwald 10744ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 107578d08d09SMilanka Ringwald if (!stream_endpoint) { 10764ccacc40SMilanka Ringwald log_error("avdtp_reconfigure: no initiator stream endpoint for seid %d", local_seid); 10779974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST; 107878d08d09SMilanka Ringwald } 107978d08d09SMilanka Ringwald 1080485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint)){ 10818587e32cSMilanka Ringwald log_error("avdtp_reconfigure: no associated remote sep"); 10829974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 108378d08d09SMilanka Ringwald } 1084485c0a4cSMilanka Ringwald 1085747ec646SMilanka Ringwald connection->initiator_transaction_label++; 10869413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10876b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 1088f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 1089f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 1090747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_RECONFIGURE_STREAM_WITH_SEID; 10919974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1092747ec646SMilanka Ringwald } 1093747ec646SMilanka Ringwald 10948e7044f9SMatthias Ringwald void avdtp_set_preferred_sampling_frequeny(avdtp_stream_endpoint_t * stream_endpoint, uint32_t sampling_frequency){ 10958e7044f9SMatthias Ringwald stream_endpoint->preferred_sampling_frequency = sampling_frequency; 10968e7044f9SMatthias Ringwald } 10978e7044f9SMatthias Ringwald 109878d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_channel_mode(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_channel_mode_bitmap){ 109978d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 110078d08d09SMilanka Ringwald uint8_t channel_mode_bitmap = (media_codec[0] & 0x0F) & remote_channel_mode_bitmap; 110178d08d09SMilanka Ringwald 110278d08d09SMilanka Ringwald uint8_t channel_mode = AVDTP_SBC_STEREO; 110378d08d09SMilanka Ringwald if (channel_mode_bitmap & AVDTP_SBC_JOINT_STEREO){ 110478d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_JOINT_STEREO; 110578d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_STEREO){ 110678d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_STEREO; 110778d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_DUAL_CHANNEL){ 110878d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_DUAL_CHANNEL; 110978d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_MONO){ 111078d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_MONO; 111178d08d09SMilanka Ringwald } 111278d08d09SMilanka Ringwald return channel_mode; 111378d08d09SMilanka Ringwald } 111478d08d09SMilanka Ringwald 111578d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_allocation_method(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_allocation_method_bitmap){ 111678d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 111778d08d09SMilanka Ringwald uint8_t allocation_method_bitmap = (media_codec[1] & 0x03) & remote_allocation_method_bitmap; 111878d08d09SMilanka Ringwald 111978d08d09SMilanka Ringwald uint8_t allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 112078d08d09SMilanka Ringwald if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS){ 112178d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 112278d08d09SMilanka Ringwald } else if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_SNR){ 112378d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_SNR; 112478d08d09SMilanka Ringwald } 112578d08d09SMilanka Ringwald return allocation_method; 112678d08d09SMilanka Ringwald } 112778d08d09SMilanka Ringwald 1128bd1ecb8aSMilanka Ringwald uint8_t avdtp_stream_endpoint_seid(avdtp_stream_endpoint_t * stream_endpoint){ 1129bd1ecb8aSMilanka Ringwald if (!stream_endpoint) return 0; 1130bd1ecb8aSMilanka Ringwald return stream_endpoint->sep.seid; 1131bd1ecb8aSMilanka Ringwald } 113278d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_subbands(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_subbands_bitmap){ 113367ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 113478d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 113578d08d09SMilanka Ringwald uint8_t subbands_bitmap = ((media_codec[1] >> 2) & 0x03) & remote_subbands_bitmap; 113678d08d09SMilanka Ringwald 113778d08d09SMilanka Ringwald uint8_t subbands = AVDTP_SBC_SUBBANDS_8; 113878d08d09SMilanka Ringwald if (subbands_bitmap & AVDTP_SBC_SUBBANDS_8){ 113978d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_8; 114078d08d09SMilanka Ringwald } else if (subbands_bitmap & AVDTP_SBC_SUBBANDS_4){ 114178d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_4; 114278d08d09SMilanka Ringwald } 114378d08d09SMilanka Ringwald return subbands; 114478d08d09SMilanka Ringwald } 114578d08d09SMilanka Ringwald 114678d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_block_length(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_block_length_bitmap){ 114767ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 114878d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 114978d08d09SMilanka Ringwald uint8_t block_length_bitmap = (media_codec[1] >> 4) & remote_block_length_bitmap; 115078d08d09SMilanka Ringwald 115178d08d09SMilanka Ringwald uint8_t block_length = AVDTP_SBC_BLOCK_LENGTH_16; 115278d08d09SMilanka Ringwald if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_16){ 115378d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_16; 115478d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_12){ 115578d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_12; 115678d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_8){ 115778d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_8; 115878d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_4){ 115978d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_4; 116078d08d09SMilanka Ringwald } 116178d08d09SMilanka Ringwald return block_length; 116278d08d09SMilanka Ringwald } 116378d08d09SMilanka Ringwald 116478d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_sampling_frequency(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_sampling_frequency_bitmap){ 116567ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 116678d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 11678e7044f9SMatthias Ringwald uint8_t supported_sampling_frequency_bitmap = (media_codec[0] >> 4) & remote_sampling_frequency_bitmap; 11688e7044f9SMatthias Ringwald uint8_t sampling_frequency = AVDTP_SBC_44100; // some default 116978d08d09SMilanka Ringwald 11708e7044f9SMatthias Ringwald // use preferred sampling frequency if possible 11718e7044f9SMatthias Ringwald if ((stream_endpoint->preferred_sampling_frequency == 48000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_48000)){ 117278d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_48000; 11738e7044f9SMatthias Ringwald } else if ((stream_endpoint->preferred_sampling_frequency == 44100) && (supported_sampling_frequency_bitmap & AVDTP_SBC_44100)){ 117478d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_44100; 11758e7044f9SMatthias Ringwald } else if ((stream_endpoint->preferred_sampling_frequency == 32000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_32000)){ 117678d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_32000; 11778e7044f9SMatthias Ringwald } else if ((stream_endpoint->preferred_sampling_frequency == 16000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_16000)){ 11788e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_16000; 11798e7044f9SMatthias Ringwald } 11808e7044f9SMatthias Ringwald // otherwise, use highest available 11818e7044f9SMatthias Ringwald else if (supported_sampling_frequency_bitmap & AVDTP_SBC_48000){ 11828e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_48000; 11838e7044f9SMatthias Ringwald } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_44100){ 11848e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_44100; 11858e7044f9SMatthias Ringwald } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_32000){ 11868e7044f9SMatthias Ringwald sampling_frequency = AVDTP_SBC_32000; 11878e7044f9SMatthias Ringwald } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_16000){ 118878d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_16000; 118978d08d09SMilanka Ringwald } 119078d08d09SMilanka Ringwald return sampling_frequency; 119178d08d09SMilanka Ringwald } 119278d08d09SMilanka Ringwald 119378d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_max_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_max_bitpool_value){ 119467ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 119578d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 119678d08d09SMilanka Ringwald return btstack_min(media_codec[3], remote_max_bitpool_value); 119778d08d09SMilanka Ringwald } 119878d08d09SMilanka Ringwald 119978d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_min_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_min_bitpool_value){ 120067ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 120178d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 120278d08d09SMilanka Ringwald return btstack_max(media_codec[2], remote_min_bitpool_value); 1203747ec646SMilanka Ringwald } 1204485c0a4cSMilanka Ringwald 1205485c0a4cSMilanka Ringwald uint8_t is_avdtp_remote_seid_registered(avdtp_stream_endpoint_t * stream_endpoint){ 1206485c0a4cSMilanka Ringwald if (!stream_endpoint) return 0; 1207485c0a4cSMilanka Ringwald if (stream_endpoint->remote_sep.seid == 0) return 0; 1208485c0a4cSMilanka Ringwald if (stream_endpoint->remote_sep.seid > 0x3E) return 0; 1209485c0a4cSMilanka Ringwald return 1; 1210485c0a4cSMilanka Ringwald } 1211