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; 91d8b859a2SMilanka Ringwald connection->is_configuration_initiated_locally = 1; 92bdbc3ef6SMilanka Ringwald connection->is_initiator = 1; 93bdbc3ef6SMilanka Ringwald connection->initiator_transaction_label++; 94bdbc3ef6SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION; 95d8b859a2SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 96d8b859a2SMilanka Ringwald } 97d8b859a2SMilanka Ringwald 98d8b859a2SMilanka Ringwald void avdtp_configuration_timer_start(avdtp_connection_t * connection){ 99bdbc3ef6SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t*) connection->active_stream_endpoint; 100bdbc3ef6SMilanka Ringwald if (!stream_endpoint) { 101bdbc3ef6SMilanka Ringwald log_error("avdtp_configuration_timeout_handler: no initiator stream endpoint for seid %d", connection->local_seid); 102bdbc3ef6SMilanka Ringwald return; 103bdbc3ef6SMilanka Ringwald } 104bdbc3ef6SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_CONFIGURATION_SUBSTATEMACHINE) return; 105bdbc3ef6SMilanka Ringwald 106d8b859a2SMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 107d8b859a2SMilanka Ringwald btstack_run_loop_set_timer_handler(&connection->configuration_timer, avdtp_configuration_timeout_handler); 108d8b859a2SMilanka Ringwald btstack_run_loop_set_timer_context(&connection->configuration_timer, connection); 109d8b859a2SMilanka Ringwald btstack_run_loop_set_timer(&connection->configuration_timer, CONFIGURATION_TIMEOUT_MS); 110d8b859a2SMilanka Ringwald btstack_run_loop_add_timer(&connection->configuration_timer); 111d8b859a2SMilanka Ringwald } 112d8b859a2SMilanka Ringwald 113d8b859a2SMilanka Ringwald void avdtp_configuration_timer_stop(avdtp_connection_t * connection){ 114d8b859a2SMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 115d8b859a2SMilanka Ringwald } 116d8b859a2SMilanka Ringwald 117b0d75c91SMilanka Ringwald static uint16_t avdtp_get_next_initiator_transaction_label(avdtp_context_t * context){ 118b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter++; 119b0d75c91SMilanka Ringwald if (context->initiator_transaction_id_counter == 0){ 120b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter = 1; 121b0d75c91SMilanka Ringwald } 122b0d75c91SMilanka Ringwald return context->initiator_transaction_id_counter; 123b0d75c91SMilanka Ringwald } 124b0d75c91SMilanka Ringwald 125af121d54SMilanka Ringwald static uint16_t avdtp_get_next_avdtp_cid(avdtp_context_t * context){ 126af121d54SMilanka Ringwald do { 127af121d54SMilanka Ringwald if (avdtp_cid_counter == 0xffff) { 1284ccacc40SMilanka Ringwald avdtp_cid_counter = 1; 129af121d54SMilanka Ringwald } else { 130af121d54SMilanka Ringwald avdtp_cid_counter++; 1314ccacc40SMilanka Ringwald } 132af121d54SMilanka Ringwald } while (avdtp_connection_for_avdtp_cid(avdtp_cid_counter, context) != NULL) ; 1334ccacc40SMilanka Ringwald return avdtp_cid_counter; 1344ccacc40SMilanka Ringwald } 1354ccacc40SMilanka Ringwald 136af121d54SMilanka Ringwald static avdtp_stream_endpoint_t * avdtp_stream_endpoint_for_id(avdtp_context_t * context, uint16_t stream_endpoint_id) { 137af121d54SMilanka Ringwald btstack_linked_item_t *it; 138af121d54SMilanka Ringwald for (it = (btstack_linked_item_t *) context->stream_endpoints; it ; it = it->next){ 139af121d54SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = ((avdtp_stream_endpoint_t *) it); 140af121d54SMilanka Ringwald 141af121d54SMilanka Ringwald if (stream_endpoint->sep.seid == stream_endpoint_id) { 142af121d54SMilanka Ringwald return stream_endpoint; 143af121d54SMilanka Ringwald }; 144af121d54SMilanka Ringwald } 145af121d54SMilanka Ringwald return NULL; 146af121d54SMilanka Ringwald } 147af121d54SMilanka Ringwald 1484ccacc40SMilanka Ringwald static uint16_t avdtp_get_next_local_seid(avdtp_context_t * context){ 149af121d54SMilanka Ringwald uint16_t stream_endpoint_id = context->stream_endpoints_id_counter; 150af121d54SMilanka Ringwald do { 151af121d54SMilanka Ringwald if (stream_endpoint_id == 0xffff) { 152af121d54SMilanka Ringwald stream_endpoint_id = 1; 153af121d54SMilanka Ringwald } else { 154af121d54SMilanka Ringwald stream_endpoint_id++; 1554ccacc40SMilanka Ringwald } 156af121d54SMilanka Ringwald } while (avdtp_stream_endpoint_for_id(context, stream_endpoint_id) != NULL) ; 157af121d54SMilanka Ringwald return stream_endpoint_id; 1584ccacc40SMilanka Ringwald } 1594ccacc40SMilanka Ringwald 160af121d54SMilanka Ringwald 1614ccacc40SMilanka Ringwald uint8_t avdtp_connect(bd_addr_t remote, avdtp_sep_type_t query_role, avdtp_context_t * avdtp_context, uint16_t * avdtp_cid){ 1622f6083d0SMilanka Ringwald sdp_query_context = avdtp_context; 163692c0605SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_bd_addr(remote, avdtp_context); 164692c0605SMilanka Ringwald if (!connection){ 165692c0605SMilanka Ringwald connection = avdtp_create_connection(remote, avdtp_context); 1664567cc17SMilanka Ringwald if (!connection){ 1679900b7faSMilanka Ringwald log_error("Not enough memory to create connection."); 1684567cc17SMilanka Ringwald return BTSTACK_MEMORY_ALLOC_FAILED; 1694567cc17SMilanka Ringwald } 170692c0605SMilanka Ringwald } 1714ccacc40SMilanka Ringwald 1722ad6b656SMilanka Ringwald if (avdtp_cid != NULL) { 173194cd2dfSandryblack *avdtp_cid = connection->avdtp_cid; 1742ad6b656SMilanka Ringwald } 1752ad6b656SMilanka Ringwald 1765448c259SMilanka Ringwald avdtp_context->avdtp_cid = connection->avdtp_cid; 1775448c259SMilanka Ringwald 1785448c259SMilanka Ringwald uint8_t err; 1795448c259SMilanka Ringwald switch (connection->state){ 1805448c259SMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_IDLE: 1815448c259SMilanka Ringwald connection->state = AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE; 18263331bf4SMilanka Ringwald connection->is_initiator = 1; 1835448c259SMilanka Ringwald sdp_query_context = avdtp_context; 1842f6083d0SMilanka Ringwald avdtp_context->avdtp_l2cap_psm = 0; 1852f6083d0SMilanka Ringwald avdtp_context->avdtp_version = 0; 1862f6083d0SMilanka Ringwald avdtp_context->query_role = query_role; 1875448c259SMilanka Ringwald err = sdp_client_query_uuid16(&avdtp_handle_sdp_client_query_result, remote, BLUETOOTH_PROTOCOL_AVDTP); 1881e1ae2bcSMilanka Ringwald if (err != ERROR_CODE_SUCCESS){ 1891e1ae2bcSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 1902f6083d0SMilanka Ringwald btstack_linked_list_remove(&avdtp_context->connections, (btstack_linked_item_t*) connection); 1912f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 1925448c259SMilanka Ringwald } 1931e1ae2bcSMilanka Ringwald return err; 194a0b8a58cSMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_OPENED:{ 195a0b8a58cSMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_signaling_cid(connection->l2cap_signaling_cid, avdtp_context); 196a0b8a58cSMilanka Ringwald if (stream_endpoint){ 197a0b8a58cSMilanka 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); 198a0b8a58cSMilanka Ringwald break; 199a0b8a58cSMilanka Ringwald } 200596b7fdcSMilanka Ringwald avdtp_signaling_emit_connection_established(avdtp_context->avdtp_callback, connection->avdtp_cid, connection->remote_addr, ERROR_CODE_SUCCESS); 2015448c259SMilanka Ringwald break; 202a0b8a58cSMilanka Ringwald } 2035448c259SMilanka Ringwald default: 2045448c259SMilanka Ringwald log_error("avdtp_connect: sink in wrong state"); 2055448c259SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 2065448c259SMilanka Ringwald 2071e1ae2bcSMilanka Ringwald } 2084ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 209692c0605SMilanka Ringwald } 210747ec646SMilanka Ringwald 211747ec646SMilanka Ringwald void avdtp_register_media_transport_category(avdtp_stream_endpoint_t * stream_endpoint){ 212747ec646SMilanka Ringwald if (!stream_endpoint){ 2139900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 214747ec646SMilanka Ringwald return; 215747ec646SMilanka Ringwald } 216747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MEDIA_TRANSPORT, 1); 217747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 218747ec646SMilanka Ringwald } 219747ec646SMilanka Ringwald 220747ec646SMilanka Ringwald void avdtp_register_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 221747ec646SMilanka Ringwald if (!stream_endpoint){ 2229900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 223747ec646SMilanka Ringwald return; 224747ec646SMilanka Ringwald } 225747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_REPORTING, 1); 226747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 227747ec646SMilanka Ringwald } 228747ec646SMilanka Ringwald 229747ec646SMilanka Ringwald void avdtp_register_delay_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 230747ec646SMilanka Ringwald if (!stream_endpoint){ 2319900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 232747ec646SMilanka Ringwald return; 233747ec646SMilanka Ringwald } 234747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_DELAY_REPORTING, 1); 235747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 236747ec646SMilanka Ringwald } 237747ec646SMilanka Ringwald 238747ec646SMilanka Ringwald void avdtp_register_recovery_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t maximum_recovery_window_size, uint8_t maximum_number_media_packets){ 239747ec646SMilanka Ringwald if (!stream_endpoint){ 2409900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 241747ec646SMilanka Ringwald return; 242747ec646SMilanka Ringwald } 243747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_RECOVERY, 1); 244747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 245747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.recovery_type = 0x01; // 0x01 = RFC2733 246747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_recovery_window_size = maximum_recovery_window_size; 247747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_number_media_packets = maximum_number_media_packets; 248747ec646SMilanka Ringwald } 249747ec646SMilanka Ringwald 250747ec646SMilanka 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){ 251747ec646SMilanka Ringwald if (!stream_endpoint){ 2529900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 253747ec646SMilanka Ringwald return; 254747ec646SMilanka Ringwald } 255747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_CONTENT_PROTECTION, 1); 256747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 257747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type = cp_type; 258*6535961aSMatthias Ringwald (void)memcpy(stream_endpoint->sep.capabilities.content_protection.cp_type_value, 259*6535961aSMatthias Ringwald cp_type_value, 260*6535961aSMatthias Ringwald btstack_min(cp_type_value_len, AVDTP_MAX_CONTENT_PROTECTION_TYPE_VALUE_LEN)); 26167ae582dSMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type_value_len = btstack_min(cp_type_value_len, AVDTP_MAX_CONTENT_PROTECTION_TYPE_VALUE_LEN); 262747ec646SMilanka Ringwald } 263747ec646SMilanka Ringwald 264747ec646SMilanka Ringwald void avdtp_register_header_compression_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t back_ch, uint8_t media, uint8_t recovery){ 265747ec646SMilanka Ringwald if (!stream_endpoint){ 2669900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 267747ec646SMilanka Ringwald return; 268747ec646SMilanka Ringwald } 269747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_HEADER_COMPRESSION, 1); 270747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 271747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.back_ch = back_ch; 272747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.media = media; 273747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.recovery = recovery; 274747ec646SMilanka Ringwald } 275747ec646SMilanka Ringwald 27678d08d09SMilanka 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){ 277747ec646SMilanka Ringwald if (!stream_endpoint){ 2789900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 279747ec646SMilanka Ringwald return; 280747ec646SMilanka Ringwald } 281747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MEDIA_CODEC, 1); 282747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 283747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_type = media_type; 284747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_type = media_codec_type; 285747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information = media_codec_info; 286747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information_len = media_codec_info_len; 287747ec646SMilanka Ringwald } 288747ec646SMilanka Ringwald 289747ec646SMilanka Ringwald void avdtp_register_multiplexing_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t fragmentation){ 290747ec646SMilanka Ringwald if (!stream_endpoint){ 2919900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 292747ec646SMilanka Ringwald return; 293747ec646SMilanka Ringwald } 294747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MULTIPLEXING, 1); 295747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 296747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.multiplexing_mode.fragmentation = fragmentation; 297747ec646SMilanka Ringwald } 298747ec646SMilanka Ringwald 299747ec646SMilanka Ringwald 300747ec646SMilanka Ringwald /* START: tracking can send now requests pro l2cap cid */ 301747ec646SMilanka Ringwald void avdtp_handle_can_send_now(avdtp_connection_t * connection, uint16_t l2cap_cid, avdtp_context_t * context){ 302747ec646SMilanka Ringwald if (connection->wait_to_send_acceptor){ 303747ec646SMilanka Ringwald connection->wait_to_send_acceptor = 0; 304747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm_run(connection, context); 305747ec646SMilanka Ringwald } else if (connection->wait_to_send_initiator){ 306747ec646SMilanka Ringwald connection->wait_to_send_initiator = 0; 307747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm_run(connection, context); 308747ec646SMilanka Ringwald } else if (connection->wait_to_send_self){ 309747ec646SMilanka Ringwald connection->wait_to_send_self = 0; 310747ec646SMilanka Ringwald if (connection->disconnect){ 311747ec646SMilanka Ringwald btstack_linked_list_iterator_t it; 312747ec646SMilanka Ringwald btstack_linked_list_iterator_init(&it, &context->stream_endpoints); 313747ec646SMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 314747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 315747ec646SMilanka Ringwald if (stream_endpoint->connection == connection){ 3160e588213SMatthias Ringwald if ((stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_OPENED) && (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED)){ 317747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED; 3189413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 319747ec646SMilanka Ringwald l2cap_disconnect(stream_endpoint->l2cap_media_cid, 0); 320747ec646SMilanka Ringwald return; 321747ec646SMilanka Ringwald } 322747ec646SMilanka Ringwald } 323747ec646SMilanka Ringwald } 324747ec646SMilanka Ringwald connection->disconnect = 0; 325747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED; 3269413b167SMilanka Ringwald l2cap_disconnect(connection->l2cap_signaling_cid, 0); 327747ec646SMilanka Ringwald return; 328747ec646SMilanka Ringwald } 329747ec646SMilanka Ringwald } 330747ec646SMilanka Ringwald 331747ec646SMilanka Ringwald // re-register 332747ec646SMilanka Ringwald int more_to_send = connection->wait_to_send_acceptor || connection->wait_to_send_initiator || connection->wait_to_send_self; 333747ec646SMilanka Ringwald if (more_to_send){ 334747ec646SMilanka Ringwald l2cap_request_can_send_now_event(l2cap_cid); 335747ec646SMilanka Ringwald } 336747ec646SMilanka Ringwald } 337747ec646SMilanka Ringwald /* END: tracking can send now requests pro l2cap cid */ 338747ec646SMilanka Ringwald 339747ec646SMilanka Ringwald avdtp_connection_t * avdtp_create_connection(bd_addr_t remote_addr, avdtp_context_t * context){ 340747ec646SMilanka Ringwald avdtp_connection_t * connection = btstack_memory_avdtp_connection_get(); 3414567cc17SMilanka Ringwald if (!connection){ 3429900b7faSMilanka Ringwald log_error("Not enough memory to create connection"); 3434567cc17SMilanka Ringwald return NULL; 3444567cc17SMilanka Ringwald } 345747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 346b0d75c91SMilanka Ringwald connection->initiator_transaction_label = avdtp_get_next_initiator_transaction_label(context); 347af121d54SMilanka Ringwald connection->avdtp_cid = avdtp_get_next_avdtp_cid(context); 34863331bf4SMilanka Ringwald context->avdtp_cid = connection->avdtp_cid; 349*6535961aSMatthias Ringwald (void)memcpy(connection->remote_addr, remote_addr, 6); 350747ec646SMilanka Ringwald btstack_linked_list_add(&context->connections, (btstack_linked_item_t *) connection); 351747ec646SMilanka Ringwald return connection; 352747ec646SMilanka Ringwald } 353747ec646SMilanka Ringwald 354747ec646SMilanka 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){ 355747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = btstack_memory_avdtp_stream_endpoint_get(); 3564567cc17SMilanka Ringwald if (!stream_endpoint){ 3579900b7faSMilanka Ringwald log_error("Not enough memory to create stream endpoint"); 3584567cc17SMilanka Ringwald return NULL; 3594567cc17SMilanka Ringwald } 3604ccacc40SMilanka Ringwald stream_endpoint->sep.seid = avdtp_get_next_local_seid(context); 361747ec646SMilanka Ringwald stream_endpoint->sep.media_type = media_type; 362747ec646SMilanka Ringwald stream_endpoint->sep.type = sep_type; 363747ec646SMilanka Ringwald btstack_linked_list_add(&context->stream_endpoints, (btstack_linked_item_t *) stream_endpoint); 364747ec646SMilanka Ringwald return stream_endpoint; 365747ec646SMilanka Ringwald } 366747ec646SMilanka Ringwald 367747ec646SMilanka Ringwald 368747ec646SMilanka Ringwald static void handle_l2cap_data_packet_for_signaling_connection(avdtp_connection_t * connection, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 369747ec646SMilanka Ringwald int offset = avdtp_read_signaling_header(&connection->signaling_packet, packet, size); 370747ec646SMilanka Ringwald switch (connection->signaling_packet.message_type){ 371747ec646SMilanka Ringwald case AVDTP_CMD_MSG: 372747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm(connection, packet, size, offset, context); 373747ec646SMilanka Ringwald break; 374747ec646SMilanka Ringwald default: 375747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm(connection, packet, size, offset, context); 376747ec646SMilanka Ringwald break; 377747ec646SMilanka Ringwald } 378747ec646SMilanka Ringwald } 379747ec646SMilanka Ringwald 380692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 3812f6083d0SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(sdp_query_context->avdtp_cid, sdp_query_context); 3822f6083d0SMilanka Ringwald if (!connection) { 383355ac553SMilanka Ringwald log_error("SDP query, connection with 0x%02x cid not found", sdp_query_context->avdtp_cid); 3842f6083d0SMilanka Ringwald return; 3852f6083d0SMilanka Ringwald } 3862f6083d0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE) return; 3872f6083d0SMilanka Ringwald 388692c0605SMilanka Ringwald UNUSED(packet_type); 389692c0605SMilanka Ringwald UNUSED(channel); 390692c0605SMilanka Ringwald UNUSED(size); 391692c0605SMilanka Ringwald 392692c0605SMilanka Ringwald des_iterator_t des_list_it; 393692c0605SMilanka Ringwald des_iterator_t prot_it; 3941e1ae2bcSMilanka Ringwald uint8_t status; 395692c0605SMilanka Ringwald 396692c0605SMilanka Ringwald switch (hci_event_packet_get_type(packet)){ 397692c0605SMilanka Ringwald case SDP_EVENT_QUERY_ATTRIBUTE_VALUE: 398692c0605SMilanka Ringwald // Handle new SDP record 399692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_record_id(packet) != record_id) { 400692c0605SMilanka Ringwald record_id = sdp_event_query_attribute_byte_get_record_id(packet); 4018587e32cSMilanka Ringwald // log_info("SDP Record: Nr: %d", record_id); 402692c0605SMilanka Ringwald } 403692c0605SMilanka Ringwald 404692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_attribute_length(packet) <= attribute_value_buffer_size) { 405692c0605SMilanka Ringwald attribute_value[sdp_event_query_attribute_byte_get_data_offset(packet)] = sdp_event_query_attribute_byte_get_data(packet); 406692c0605SMilanka Ringwald 407692c0605SMilanka Ringwald if ((uint16_t)(sdp_event_query_attribute_byte_get_data_offset(packet)+1) == sdp_event_query_attribute_byte_get_attribute_length(packet)) { 408692c0605SMilanka Ringwald 409692c0605SMilanka Ringwald switch(sdp_event_query_attribute_byte_get_attribute_id(packet)) { 410692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_SERVICE_CLASS_ID_LIST: 411692c0605SMilanka Ringwald if (de_get_element_type(attribute_value) != DE_DES) break; 412692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 413692c0605SMilanka Ringwald uint8_t * element = des_iterator_get_element(&des_list_it); 414692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 415692c0605SMilanka Ringwald uint32_t uuid = de_get_uuid32(element); 416692c0605SMilanka Ringwald switch (uuid){ 417692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SOURCE: 4182f6083d0SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SOURCE) { 4192f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 420692c0605SMilanka Ringwald break; 421692c0605SMilanka Ringwald } 4228587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SOURCE protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 423eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 424692c0605SMilanka Ringwald break; 425692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SINK: 4265448c259SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SINK) { 4272f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 428692c0605SMilanka Ringwald break; 429692c0605SMilanka Ringwald } 4308587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SINK protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 431eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 432692c0605SMilanka Ringwald break; 433692c0605SMilanka Ringwald default: 434692c0605SMilanka Ringwald break; 435692c0605SMilanka Ringwald } 436692c0605SMilanka Ringwald } 437692c0605SMilanka Ringwald break; 438692c0605SMilanka Ringwald 439692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_PROTOCOL_DESCRIPTOR_LIST: { 4408587e32cSMilanka Ringwald // log_info("SDP Attribute: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet)); 441692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 442692c0605SMilanka Ringwald uint8_t *des_element; 443692c0605SMilanka Ringwald uint8_t *element; 444692c0605SMilanka Ringwald uint32_t uuid; 445692c0605SMilanka Ringwald 446692c0605SMilanka Ringwald if (des_iterator_get_type(&des_list_it) != DE_DES) continue; 447692c0605SMilanka Ringwald 448692c0605SMilanka Ringwald des_element = des_iterator_get_element(&des_list_it); 449692c0605SMilanka Ringwald des_iterator_init(&prot_it, des_element); 450692c0605SMilanka Ringwald element = des_iterator_get_element(&prot_it); 451692c0605SMilanka Ringwald 452692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 453692c0605SMilanka Ringwald 454692c0605SMilanka Ringwald uuid = de_get_uuid32(element); 45514fd128cSMatthias Ringwald des_iterator_next(&prot_it); 456692c0605SMilanka Ringwald switch (uuid){ 457692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_L2CAP: 458692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 4592f6083d0SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_l2cap_psm); 460692c0605SMilanka Ringwald break; 461692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_AVDTP: 462692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 4632f6083d0SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_version); 464692c0605SMilanka Ringwald break; 465692c0605SMilanka Ringwald default: 466692c0605SMilanka Ringwald break; 467692c0605SMilanka Ringwald } 468692c0605SMilanka Ringwald } 469692c0605SMilanka Ringwald } 470692c0605SMilanka Ringwald break; 471692c0605SMilanka Ringwald default: 472692c0605SMilanka Ringwald break; 473692c0605SMilanka Ringwald } 474692c0605SMilanka Ringwald } 475692c0605SMilanka Ringwald } else { 4768587e32cSMilanka 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)); 477692c0605SMilanka Ringwald } 478692c0605SMilanka Ringwald break; 479692c0605SMilanka Ringwald 480692c0605SMilanka Ringwald case SDP_EVENT_QUERY_COMPLETE: 4810d4a198eSMatthias Ringwald if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE){ 4820d4a198eSMatthias Ringwald // bail out, we must have had an incoming connection in the meantime; 4830d4a198eSMatthias Ringwald break; 4840d4a198eSMatthias Ringwald } 4851e1ae2bcSMilanka Ringwald status = sdp_event_query_complete_get_status(packet); 4862f6083d0SMilanka Ringwald if (status != ERROR_CODE_SUCCESS){ 4875448c259SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, status); 4882f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4892f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 490355ac553SMilanka Ringwald log_info("SDP query failed with status 0x%02x.", status); 4911e1ae2bcSMilanka Ringwald break; 4921e1ae2bcSMilanka Ringwald } 4932f6083d0SMilanka Ringwald if (!sdp_query_context->role_supported){ 4942f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4952f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 4962f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, SDP_SERVICE_NOT_FOUND); 497355ac553SMilanka Ringwald log_info("SDP query, remote device does not support required role."); 498974d4d6eSMilanka Ringwald break; 499974d4d6eSMilanka Ringwald } 5002f6083d0SMilanka Ringwald if (!sdp_query_context->avdtp_l2cap_psm) { 5012f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*)connection); 5022f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 5032f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, L2CAP_SERVICE_DOES_NOT_EXIST); 504355ac553SMilanka Ringwald log_info("SDP query, no l2cap psm found."); 5052f6083d0SMilanka Ringwald break; 5062f6083d0SMilanka Ringwald } 5072f6083d0SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 5082f6083d0SMilanka Ringwald l2cap_create_channel(sdp_query_context->packet_handler, connection->remote_addr, sdp_query_context->avdtp_l2cap_psm, l2cap_max_mtu(), NULL); 509692c0605SMilanka Ringwald break; 510692c0605SMilanka Ringwald } 511692c0605SMilanka Ringwald } 512692c0605SMilanka Ringwald 513692c0605SMilanka Ringwald 514747ec646SMilanka Ringwald void avdtp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 515747ec646SMilanka Ringwald bd_addr_t event_addr; 516747ec646SMilanka Ringwald uint16_t psm; 517747ec646SMilanka Ringwald uint16_t local_cid; 5181e1ae2bcSMilanka Ringwald uint8_t status; 5190d4a198eSMatthias Ringwald int accept_signaling_connection; 520747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = NULL; 521747ec646SMilanka Ringwald avdtp_connection_t * connection = NULL; 522747ec646SMilanka Ringwald btstack_linked_list_t * avdtp_connections = &context->connections; 523747ec646SMilanka Ringwald btstack_linked_list_t * stream_endpoints = &context->stream_endpoints; 524747ec646SMilanka Ringwald handle_media_data = context->handle_media_data; 5258587e32cSMilanka Ringwald // log_info("avdtp_packet_handler packet type %02x, event %02x ", packet_type, hci_event_packet_get_type(packet)); 526747ec646SMilanka Ringwald switch (packet_type) { 527747ec646SMilanka Ringwald case L2CAP_DATA_PACKET: 5289413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 529747ec646SMilanka Ringwald if (connection){ 530747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 531747ec646SMilanka Ringwald break; 532747ec646SMilanka Ringwald } 533747ec646SMilanka Ringwald 534747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 535747ec646SMilanka Ringwald if (!stream_endpoint){ 536747ec646SMilanka Ringwald if (!connection) break; 537747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 538747ec646SMilanka Ringwald break; 539747ec646SMilanka Ringwald } 540747ec646SMilanka Ringwald 5418c0f3635SMilanka Ringwald if (stream_endpoint->connection){ 5429413b167SMilanka Ringwald if (channel == stream_endpoint->connection->l2cap_signaling_cid){ 543a466d508SMilanka Ringwald int offset = avdtp_read_signaling_header(&stream_endpoint->connection->signaling_packet, packet, size); 544a466d508SMilanka Ringwald if (stream_endpoint->connection->signaling_packet.message_type == AVDTP_CMD_MSG){ 545a466d508SMilanka Ringwald avdtp_acceptor_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context); 546a466d508SMilanka Ringwald } else { 547a466d508SMilanka Ringwald avdtp_initiator_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context); 548a466d508SMilanka Ringwald } 549747ec646SMilanka Ringwald break; 550747ec646SMilanka Ringwald } 5518c0f3635SMilanka Ringwald } 552747ec646SMilanka Ringwald 553747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_media_cid){ 5548b097e29SMilanka Ringwald if (handle_media_data){ 555fd58c900SMilanka Ringwald (*handle_media_data)(avdtp_local_seid(stream_endpoint), packet, size); 5568b097e29SMilanka Ringwald } 557747ec646SMilanka Ringwald break; 558747ec646SMilanka Ringwald } 559747ec646SMilanka Ringwald 560747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_reporting_cid){ 561747ec646SMilanka Ringwald // TODO 5628587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for reporting: NOT IMPLEMENTED"); 563747ec646SMilanka Ringwald } else if (channel == stream_endpoint->l2cap_recovery_cid){ 564747ec646SMilanka Ringwald // TODO 5658587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for recovery: NOT IMPLEMENTED"); 566747ec646SMilanka Ringwald } else { 567747ec646SMilanka Ringwald log_error("avdtp packet handler L2CAP_DATA_PACKET: local cid 0x%02x not found", channel); 568747ec646SMilanka Ringwald } 569747ec646SMilanka Ringwald break; 570747ec646SMilanka Ringwald 571747ec646SMilanka Ringwald case HCI_EVENT_PACKET: 572747ec646SMilanka Ringwald switch (hci_event_packet_get_type(packet)) { 573747ec646SMilanka Ringwald case L2CAP_EVENT_INCOMING_CONNECTION: 574747ec646SMilanka Ringwald l2cap_event_incoming_connection_get_address(packet, event_addr); 575747ec646SMilanka Ringwald local_cid = l2cap_event_incoming_connection_get_local_cid(packet); 576747ec646SMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 577355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION, local cid 0x%02x ", local_cid); 5780d4a198eSMatthias Ringwald if (!connection){ 5790d4a198eSMatthias Ringwald // create connection struct for regular inconming connection request 580747ec646SMilanka Ringwald connection = avdtp_create_connection(event_addr, context); 5810d4a198eSMatthias Ringwald if (!connection){ 5820d4a198eSMatthias Ringwald log_error("Could not create connection, reject"); 5830d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 5840d4a198eSMatthias Ringwald break; 5850d4a198eSMatthias Ringwald } 5860d4a198eSMatthias Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 5870d4a198eSMatthias Ringwald } 5880d4a198eSMatthias Ringwald 5890d4a198eSMatthias Ringwald // connection exists, check states 5900d4a198eSMatthias Ringwald log_info("Connection state %d", connection->state); 591cf837c28SMatthias Ringwald accept_signaling_connection = 0; 5920d4a198eSMatthias Ringwald switch (connection->state){ 5930d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_IDLE: 5940d4a198eSMatthias Ringwald // regular incoming connection 5950d4a198eSMatthias Ringwald accept_signaling_connection = 1; 5960d4a198eSMatthias Ringwald break; 5970d4a198eSMatthias Ringwald case AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE: 5980d4a198eSMatthias Ringwald // incoming connection during sdp query, just accept it 5990d4a198eSMatthias Ringwald accept_signaling_connection = 1; 6000d4a198eSMatthias Ringwald break; 6010d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED: 6020d4a198eSMatthias Ringwald log_info("Reject incoming connection after creating outgoing"); 6030d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 6040d4a198eSMatthias Ringwald return; 6050d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_OPENED: 6060d4a198eSMatthias Ringwald // handled below 6070d4a198eSMatthias Ringwald break; 6080d4a198eSMatthias Ringwald case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED: 6090d4a198eSMatthias Ringwald log_info("Reject incoming connection during disconnect"); 6100d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 6110d4a198eSMatthias Ringwald return; 6120d4a198eSMatthias Ringwald } 6130d4a198eSMatthias Ringwald 6140d4a198eSMatthias Ringwald if (accept_signaling_connection){ 61563331bf4SMilanka Ringwald connection->is_initiator = 0; 616747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 6170d4a198eSMatthias Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION: role is_initiator %d", connection->is_initiator); 618355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION, connection %p, state connection %d, avdtp cid 0x%02x", connection, connection->state, connection->avdtp_cid); 619747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 620747ec646SMilanka Ringwald break; 621747ec646SMilanka Ringwald } 622747ec646SMilanka Ringwald 6230d4a198eSMatthias Ringwald // now, we're only dealing with media connections 6246b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 625747ec646SMilanka Ringwald if (!stream_endpoint) { 626355ac553SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION no streamendpoint found for seid %d", connection->local_seid); 6270d4a198eSMatthias Ringwald l2cap_decline_connection(local_cid); 628747ec646SMilanka Ringwald break; 629747ec646SMilanka Ringwald } 630747ec646SMilanka Ringwald 631355ac553SMilanka Ringwald log_info("Checking l2cap_media_cid %d, for local seid %d, state of stream endpoint %d, role is_initiator %d", stream_endpoint->l2cap_media_cid, connection->local_seid, stream_endpoint->state, connection->is_initiator); 632747ec646SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 6339900b7faSMilanka Ringwald if (connection->is_initiator){ 6349900b7faSMilanka Ringwald l2cap_decline_connection(local_cid); 6359900b7faSMilanka Ringwald break; 6369900b7faSMilanka Ringwald } 637747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 638747ec646SMilanka Ringwald break; 639747ec646SMilanka Ringwald } 6409900b7faSMilanka Ringwald l2cap_decline_connection(local_cid); 641747ec646SMilanka Ringwald break; 642747ec646SMilanka Ringwald 643747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_OPENED: 644a0b8a58cSMilanka Ringwald psm = l2cap_event_channel_opened_get_psm(packet); 64584e3541eSMilanka Ringwald if (psm != BLUETOOTH_PSM_AVDTP){ 646355ac553SMilanka Ringwald log_info("Unexpected PSM - Not implemented yet, avdtp sink: L2CAP_EVENT_CHANNEL_OPENED "); 647a0b8a58cSMilanka Ringwald return; 648a0b8a58cSMilanka Ringwald } 649a0b8a58cSMilanka Ringwald 6501e1ae2bcSMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 651747ec646SMilanka Ringwald // inform about new l2cap connection 652747ec646SMilanka Ringwald l2cap_event_channel_opened_get_address(packet, event_addr); 6537050d2caSMilanka Ringwald local_cid = l2cap_event_channel_opened_get_local_cid(packet); 654a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 655a0b8a58cSMilanka Ringwald 656355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED: status %d, cid 0x%02x , signaling connection %p ", status, local_cid, connection); 657a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 658a0b8a58cSMilanka Ringwald if (!connection){ 659355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED 2: status %d, signaling connection %p ", status, connection); 660a0b8a58cSMilanka Ringwald break; 661a0b8a58cSMilanka Ringwald } 662a0b8a58cSMilanka Ringwald 663a0b8a58cSMilanka Ringwald if (connection->l2cap_signaling_cid == 0) { 6641e1ae2bcSMilanka Ringwald if (status){ 665355ac553SMilanka Ringwald log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 66655ddebccSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 667a0b8a58cSMilanka Ringwald connection->l2cap_signaling_cid = 0; 6684ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, l2cap_event_channel_opened_get_status(packet)); 669747ec646SMilanka Ringwald break; 670747ec646SMilanka Ringwald } 671a0b8a58cSMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED) { 672355ac553SMilanka Ringwald log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 673a0b8a58cSMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, AVDTP_CONNECTION_IN_WRONG_STATE); 674a0b8a58cSMilanka Ringwald break; 67555ddebccSMilanka Ringwald } 6769413b167SMilanka Ringwald connection->l2cap_signaling_cid = local_cid; 6776b0ee1d0SMilanka Ringwald connection->local_seid = 0; 678747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_OPENED; 679355ac553SMilanka Ringwald log_info("AVDTP_SIGNALING_CONNECTION_OPENED, connection %p, l2cap_signaling_cid 0x%02x, avdtp_cid 0x%02x", connection, local_cid, connection->avdtp_cid); 6804ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, 0); 681747ec646SMilanka Ringwald break; 682747ec646SMilanka Ringwald } 683747ec646SMilanka Ringwald 6846b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 685747ec646SMilanka Ringwald if (!stream_endpoint){ 686355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED: stream_endpoint not found"); 687747ec646SMilanka Ringwald return; 688747ec646SMilanka Ringwald } 689a466d508SMilanka Ringwald 690a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 691a466d508SMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 692a466d508SMilanka Ringwald if (status){ 693355ac553SMilanka 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)); 694a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_IDLE; 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), status); 696a466d508SMilanka Ringwald break; 697a466d508SMilanka Ringwald } 698a466d508SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED){ 699355ac553SMilanka 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)); 700596b7fdcSMilanka 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); 701a466d508SMilanka Ringwald break; 702a466d508SMilanka Ringwald } 703a466d508SMilanka Ringwald 704a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 705a466d508SMilanka Ringwald stream_endpoint->connection = connection; 706a466d508SMilanka Ringwald stream_endpoint->l2cap_media_cid = l2cap_event_channel_opened_get_local_cid(packet); 707a466d508SMilanka Ringwald stream_endpoint->media_con_handle = l2cap_event_channel_opened_get_handle(packet); 708d1207cd8SMilanka Ringwald 709355ac553SMilanka 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)); 710596b7fdcSMilanka 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); 711a466d508SMilanka Ringwald return; 712a466d508SMilanka Ringwald } 713747ec646SMilanka Ringwald break; 714747ec646SMilanka Ringwald 715747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_CLOSED: 716747ec646SMilanka Ringwald local_cid = l2cap_event_channel_closed_get_local_cid(packet); 7179413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(local_cid, context); 7180bd7cb1fSMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(local_cid, context); 719485c0a4cSMilanka Ringwald log_info("Received L2CAP_EVENT_CHANNEL_CLOSED, cid 0x%2x, connection %p, stream_endpoint %p", local_cid, connection, stream_endpoint); 7200bd7cb1fSMilanka Ringwald 721a466d508SMilanka Ringwald if (stream_endpoint){ 722a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == local_cid){ 723a466d508SMilanka Ringwald connection = stream_endpoint->connection; 724596b7fdcSMilanka Ringwald if (connection) { 725596b7fdcSMilanka Ringwald avdtp_streaming_emit_connection_released(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint)); 726596b7fdcSMilanka Ringwald } 727485c0a4cSMilanka Ringwald avdtp_reset_stream_endpoint(stream_endpoint); 728a466d508SMilanka Ringwald if (connection && connection->disconnect){ 729a466d508SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 730a466d508SMilanka Ringwald } 731a466d508SMilanka Ringwald break; 732a466d508SMilanka Ringwald } 733a466d508SMilanka Ringwald if (stream_endpoint->l2cap_recovery_cid == local_cid){ 734355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_CLOSED recovery cid 0x%0x", local_cid); 735a466d508SMilanka Ringwald stream_endpoint->l2cap_recovery_cid = 0; 736a466d508SMilanka Ringwald break; 737a466d508SMilanka Ringwald } 738a466d508SMilanka Ringwald 739a466d508SMilanka Ringwald if (stream_endpoint->l2cap_reporting_cid == local_cid){ 740355ac553SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_CLOSED reporting cid 0x%0x", local_cid); 741a466d508SMilanka Ringwald stream_endpoint->l2cap_reporting_cid = 0; 742a466d508SMilanka Ringwald break; 743a466d508SMilanka Ringwald } 744a466d508SMilanka Ringwald } 745596b7fdcSMilanka Ringwald 746596b7fdcSMilanka Ringwald if (connection){ 747596b7fdcSMilanka Ringwald btstack_linked_list_iterator_t it; 748596b7fdcSMilanka Ringwald btstack_linked_list_iterator_init(&it, stream_endpoints); 749596b7fdcSMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 750596b7fdcSMilanka Ringwald avdtp_stream_endpoint_t * _stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 751596b7fdcSMilanka Ringwald if (_stream_endpoint->connection == connection){ 752485c0a4cSMilanka Ringwald avdtp_reset_stream_endpoint(_stream_endpoint); 753596b7fdcSMilanka Ringwald } 754596b7fdcSMilanka Ringwald } 755485c0a4cSMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 756485c0a4cSMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 757485c0a4cSMilanka Ringwald btstack_linked_list_remove(avdtp_connections, (btstack_linked_item_t*) connection); 758596b7fdcSMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 759596b7fdcSMilanka Ringwald break; 760596b7fdcSMilanka Ringwald } 761596b7fdcSMilanka Ringwald 762747ec646SMilanka Ringwald break; 763747ec646SMilanka Ringwald 764747ec646SMilanka Ringwald case HCI_EVENT_DISCONNECTION_COMPLETE: 765747ec646SMilanka Ringwald break; 766747ec646SMilanka Ringwald 767747ec646SMilanka Ringwald case L2CAP_EVENT_CAN_SEND_NOW: 7689413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 769747ec646SMilanka Ringwald if (!connection) { 770747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 771747ec646SMilanka Ringwald if (!stream_endpoint->connection) break; 772747ec646SMilanka Ringwald connection = stream_endpoint->connection; 773747ec646SMilanka Ringwald } 774747ec646SMilanka Ringwald avdtp_handle_can_send_now(connection, channel, context); 775747ec646SMilanka Ringwald break; 776747ec646SMilanka Ringwald default: 777355ac553SMilanka Ringwald log_info("Unknown HCI event type %02x", hci_event_packet_get_type(packet)); 778747ec646SMilanka Ringwald break; 779747ec646SMilanka Ringwald } 780747ec646SMilanka Ringwald break; 781747ec646SMilanka Ringwald 782747ec646SMilanka Ringwald default: 783747ec646SMilanka Ringwald // other packet type 784747ec646SMilanka Ringwald break; 785747ec646SMilanka Ringwald } 786747ec646SMilanka Ringwald } 787747ec646SMilanka Ringwald 7884ccacc40SMilanka Ringwald uint8_t avdtp_disconnect(uint16_t avdtp_cid, avdtp_context_t * context){ 7894ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 7902ad6b656SMilanka Ringwald if (!connection) return AVDTP_CONNECTION_DOES_NOT_EXIST; 791a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_IDLE){ 792a466d508SMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 793a466d508SMilanka Ringwald return ERROR_CODE_SUCCESS; 794a466d508SMilanka Ringwald } 795a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED) return ERROR_CODE_SUCCESS; 796747ec646SMilanka Ringwald 797747ec646SMilanka Ringwald connection->disconnect = 1; 7989413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 7994ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 800747ec646SMilanka Ringwald } 801747ec646SMilanka Ringwald 8026b0ee1d0SMilanka Ringwald uint8_t avdtp_open_stream(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, avdtp_context_t * context){ 8034ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 804747ec646SMilanka Ringwald if (!connection){ 8058587e32cSMilanka Ringwald log_error("avdtp_media_connect: no connection for signaling cid 0x%02x found", avdtp_cid); 8064ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 807747ec646SMilanka Ringwald } 808747ec646SMilanka Ringwald 809747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) { 8108587e32cSMilanka Ringwald log_error("avdtp_media_connect: wrong connection state %d", connection->state); 8114ccacc40SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 812747ec646SMilanka Ringwald } 813747ec646SMilanka Ringwald 8146b0ee1d0SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 815747ec646SMilanka Ringwald if (!stream_endpoint) { 8166b0ee1d0SMilanka Ringwald log_error("avdtp_media_connect: no stream_endpoint with seid %d found", local_seid); 8174ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 818747ec646SMilanka Ringwald } 819747ec646SMilanka Ringwald 820485c0a4cSMilanka Ringwald if (stream_endpoint->remote_sep.seid != remote_seid){ 821485c0a4cSMilanka Ringwald log_error("avdtp_media_connect: no remote sep with seid %d registered with the stream endpoint", remote_seid); 822485c0a4cSMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 823485c0a4cSMilanka Ringwald } 824485c0a4cSMilanka Ringwald 8254ccacc40SMilanka Ringwald if (stream_endpoint->state < AVDTP_STREAM_ENDPOINT_CONFIGURED) return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 826747ec646SMilanka Ringwald 827747ec646SMilanka Ringwald connection->initiator_transaction_label++; 8289413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8296b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 830747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_OPEN_STREAM; 831747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W2_REQUEST_OPEN_STREAM; 8329413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8334ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 834747ec646SMilanka Ringwald } 835747ec646SMilanka Ringwald 8364ccacc40SMilanka Ringwald uint8_t avdtp_start_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8374ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 83846e6b063SMilanka Ringwald if (!connection){ 8394ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8404ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 84146e6b063SMilanka Ringwald } 8425cfe7f4cSMilanka Ringwald 8434ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8444ccacc40SMilanka Ringwald if (!stream_endpoint) { 8454ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no stream_endpoint with seid %d found", local_seid); 8464ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8474ccacc40SMilanka Ringwald } 8484ccacc40SMilanka Ringwald 8494ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8504ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8514ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8524ccacc40SMilanka Ringwald } 8534ccacc40SMilanka Ringwald 854485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint)){ 855485c0a4cSMilanka Ringwald log_error("avdtp_media_connect: no remote sep registered with the stream endpoint"); 856485c0a4cSMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8574ccacc40SMilanka Ringwald } 8584ccacc40SMilanka Ringwald 85960ec20d0SMilanka Ringwald stream_endpoint->start_stream = 1; 8604ccacc40SMilanka Ringwald connection->local_seid = local_seid; 861c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 8629413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8634ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 864747ec646SMilanka Ringwald } 865747ec646SMilanka Ringwald 8664ccacc40SMilanka Ringwald uint8_t avdtp_stop_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8674ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 868747ec646SMilanka Ringwald if (!connection){ 8694ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8704ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 871747ec646SMilanka Ringwald } 8724ccacc40SMilanka Ringwald 8734ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8744ccacc40SMilanka Ringwald if (!stream_endpoint) { 8754ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no stream_endpoint with seid %d found", local_seid); 8764ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8774ccacc40SMilanka Ringwald } 8784ccacc40SMilanka Ringwald 8794ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8804ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8814ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8824ccacc40SMilanka Ringwald } 883485c0a4cSMilanka Ringwald 884485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->stop_stream){ 885485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 886485c0a4cSMilanka Ringwald } 8874ccacc40SMilanka Ringwald 88860ec20d0SMilanka Ringwald stream_endpoint->stop_stream = 1; 8894ccacc40SMilanka Ringwald connection->local_seid = local_seid; 890c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 8919413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8924ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 893747ec646SMilanka Ringwald } 894747ec646SMilanka Ringwald 8954ccacc40SMilanka Ringwald uint8_t avdtp_abort_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8964ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 89760ec20d0SMilanka Ringwald if (!connection){ 8984ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8994ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 900747ec646SMilanka Ringwald } 9014ccacc40SMilanka Ringwald 9024ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 9034ccacc40SMilanka Ringwald if (!stream_endpoint) { 9044ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no stream_endpoint with seid %d found", local_seid); 9054ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 9064ccacc40SMilanka Ringwald } 9074ccacc40SMilanka Ringwald 9084ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 9094ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no media connection for stream_endpoint with seid %d found", local_seid); 9104ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 9114ccacc40SMilanka Ringwald } 912485c0a4cSMilanka Ringwald 913485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->abort_stream){ 914485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 915485c0a4cSMilanka Ringwald } 9164ccacc40SMilanka Ringwald 91760ec20d0SMilanka Ringwald stream_endpoint->abort_stream = 1; 9186b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 919c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 9209413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 9214ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 922747ec646SMilanka Ringwald } 923747ec646SMilanka Ringwald 9244ccacc40SMilanka Ringwald uint8_t avdtp_suspend_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 9254ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 926747ec646SMilanka Ringwald if (!connection){ 9274ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 9284ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 92960ec20d0SMilanka Ringwald } 9304ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 9314ccacc40SMilanka Ringwald if (!stream_endpoint) { 9324ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no stream_endpoint with seid %d found", local_seid); 9334ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 9344ccacc40SMilanka Ringwald } 9354ccacc40SMilanka Ringwald 9364ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 9374ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no media connection for stream_endpoint with seid %d found", local_seid); 9384ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 9394ccacc40SMilanka Ringwald } 940485c0a4cSMilanka Ringwald 941485c0a4cSMilanka Ringwald if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->suspend_stream){ 942485c0a4cSMilanka Ringwald return ERROR_CODE_SUCCESS; 943485c0a4cSMilanka Ringwald } 9444ccacc40SMilanka Ringwald 94560ec20d0SMilanka Ringwald stream_endpoint->suspend_stream = 1; 9464ccacc40SMilanka Ringwald connection->local_seid = local_seid; 947c580323eSMilanka Ringwald connection->remote_seid = stream_endpoint->remote_sep.seid; 9489413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 9494ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 950747ec646SMilanka Ringwald } 951747ec646SMilanka Ringwald 9529974aee0SMilanka Ringwald uint8_t avdtp_discover_stream_endpoints(uint16_t avdtp_cid, avdtp_context_t * context){ 9534ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 954747ec646SMilanka Ringwald if (!connection){ 9558587e32cSMilanka Ringwald log_error("avdtp_discover_stream_endpoints: no connection for signaling cid 0x%02x found", avdtp_cid); 9569974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 9579974aee0SMilanka Ringwald } 9580e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 959c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9609974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 961747ec646SMilanka Ringwald } 962ec3d71e3SMilanka Ringwald 963747ec646SMilanka Ringwald connection->initiator_transaction_label++; 964747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_DISCOVER_SEPS; 9659974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 966747ec646SMilanka Ringwald } 967747ec646SMilanka Ringwald 968747ec646SMilanka Ringwald 9699974aee0SMilanka Ringwald uint8_t avdtp_get_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9704ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 971747ec646SMilanka Ringwald if (!connection){ 9729900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9739974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 974747ec646SMilanka Ringwald } 9750e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 976c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9779974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9789974aee0SMilanka Ringwald } 9799974aee0SMilanka Ringwald 980747ec646SMilanka Ringwald connection->initiator_transaction_label++; 981747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CAPABILITIES; 9829413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9839974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 984747ec646SMilanka Ringwald } 985747ec646SMilanka Ringwald 986747ec646SMilanka Ringwald 9879974aee0SMilanka Ringwald uint8_t avdtp_get_all_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9884ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 989747ec646SMilanka Ringwald if (!connection){ 9909900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9919974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 992747ec646SMilanka Ringwald } 9930e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 994c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 9959974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9969974aee0SMilanka Ringwald } 9979974aee0SMilanka Ringwald 998747ec646SMilanka Ringwald connection->initiator_transaction_label++; 999747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_ALL_CAPABILITIES; 10009413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10019974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1002747ec646SMilanka Ringwald } 1003747ec646SMilanka Ringwald 10049974aee0SMilanka Ringwald uint8_t avdtp_get_configuration(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 10054ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1006747ec646SMilanka Ringwald if (!connection){ 10079900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10089974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1009747ec646SMilanka Ringwald } 10100e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1011c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 10129974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10139974aee0SMilanka Ringwald } 10149974aee0SMilanka Ringwald 1015747ec646SMilanka Ringwald connection->initiator_transaction_label++; 1016747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CONFIGURATION; 10179413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10189974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1019747ec646SMilanka Ringwald } 1020747ec646SMilanka Ringwald 10219974aee0SMilanka 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){ 10224ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1023747ec646SMilanka Ringwald if (!connection){ 10249900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 10259974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 1026747ec646SMilanka Ringwald } 10270e588213SMatthias Ringwald if ((connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) || 1028c1ab6cc1SMatthias Ringwald (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE)) { 1029485c0a4cSMilanka Ringwald log_error("connection in wrong state, %d, initiator state %d", connection->state, connection->initiator_connection_state); 10309974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10319974aee0SMilanka 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 } 1042bdbc3ef6SMilanka Ringwald connection->active_stream_endpoint = (void*) stream_endpoint; 1043d8b859a2SMilanka Ringwald connection->is_configuration_initiated_locally = 1; 104463331bf4SMilanka Ringwald connection->is_initiator = 1; 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; 1057*6535961aSMatthias Ringwald (void)memcpy(stream_endpoint->media_codec_sbc_info, 1058*6535961aSMatthias 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