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 38ab2c6ae4SMatthias Ringwald #define __BTSTACK_FILE__ "avdtp.c" 39ab2c6ae4SMatthias Ringwald 40747ec646SMilanka Ringwald 41747ec646SMilanka Ringwald #include <stdint.h> 42747ec646SMilanka Ringwald #include <stdio.h> 43747ec646SMilanka Ringwald #include <stdlib.h> 44747ec646SMilanka Ringwald #include <string.h> 45747ec646SMilanka Ringwald 46747ec646SMilanka Ringwald #include "btstack.h" 474cb889a5SMilanka Ringwald #include "classic/avdtp.h" 484cb889a5SMilanka Ringwald #include "classic/avdtp_util.h" 494cb889a5SMilanka Ringwald #include "classic/avdtp_acceptor.h" 504cb889a5SMilanka Ringwald #include "classic/avdtp_initiator.h" 51747ec646SMilanka Ringwald 52d8b859a2SMilanka Ringwald #define CONFIGURATION_TIMEOUT_MS 300 53d8b859a2SMilanka Ringwald 54692c0605SMilanka Ringwald static int record_id = -1; 55692c0605SMilanka Ringwald static uint8_t attribute_value[1000]; 56692c0605SMilanka Ringwald static const unsigned int attribute_value_buffer_size = sizeof(attribute_value); 57747ec646SMilanka Ringwald 582f6083d0SMilanka Ringwald // typedef struct { 592f6083d0SMilanka Ringwald // btstack_linked_list_t * avdtp_connections; 602f6083d0SMilanka Ringwald // avdtp_connection_t * connection; 612f6083d0SMilanka Ringwald // btstack_packet_handler_t avdtp_callback; 622f6083d0SMilanka Ringwald // avdtp_sep_type_t query_role; 632f6083d0SMilanka Ringwald // btstack_packet_handler_t packet_handler; 642f6083d0SMilanka Ringwald // uint16_t avdtp_l2cap_psm; 652f6083d0SMilanka Ringwald // uint16_t avdtp_version; 662f6083d0SMilanka Ringwald // uint8_t role_supported; 672f6083d0SMilanka Ringwald // } avdtp_sdp_query_context_t; 68692c0605SMilanka Ringwald 692f6083d0SMilanka Ringwald static avdtp_context_t * sdp_query_context; 7063331bf4SMilanka Ringwald static uint16_t avdtp_cid_counter = 0x55; 71747ec646SMilanka Ringwald 72fd58c900SMilanka Ringwald static void (*handle_media_data)(uint8_t local_seid, uint8_t *packet, uint16_t size); 73692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size); 74692c0605SMilanka Ringwald 75d8b859a2SMilanka Ringwald void avdtp_configuration_timeout_handler(btstack_timer_source_t * timer){ 76d8b859a2SMilanka Ringwald avdtp_connection_t * connection = (avdtp_connection_t *) btstack_run_loop_get_timer_context(timer); 77d8b859a2SMilanka Ringwald if (!connection){ 78d8b859a2SMilanka Ringwald log_error("Context of avdtp_configuration_timeout_handler is NULL"); 79d8b859a2SMilanka Ringwald return; 80d8b859a2SMilanka Ringwald } 81bdbc3ef6SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t*) connection->active_stream_endpoint; 82bdbc3ef6SMilanka Ringwald if (!stream_endpoint) { 83bdbc3ef6SMilanka Ringwald log_error("avdtp_configuration_timeout_handler: no initiator stream endpoint for seid %d", connection->local_seid); 84bdbc3ef6SMilanka Ringwald return; 85bdbc3ef6SMilanka Ringwald } 86bdbc3ef6SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_CONFIGURATION_SUBSTATEMACHINE) return; 87d8b859a2SMilanka Ringwald connection->is_configuration_initiated_locally = 1; 88bdbc3ef6SMilanka Ringwald connection->is_initiator = 1; 89bdbc3ef6SMilanka Ringwald connection->initiator_transaction_label++; 90bdbc3ef6SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION; 91d8b859a2SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 92d8b859a2SMilanka Ringwald } 93d8b859a2SMilanka Ringwald 94d8b859a2SMilanka Ringwald void avdtp_configuration_timer_start(avdtp_connection_t * connection){ 95bdbc3ef6SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t*) connection->active_stream_endpoint; 96bdbc3ef6SMilanka Ringwald if (!stream_endpoint) { 97bdbc3ef6SMilanka Ringwald log_error("avdtp_configuration_timeout_handler: no initiator stream endpoint for seid %d", connection->local_seid); 98bdbc3ef6SMilanka Ringwald return; 99bdbc3ef6SMilanka Ringwald } 100bdbc3ef6SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_CONFIGURATION_SUBSTATEMACHINE) return; 101bdbc3ef6SMilanka Ringwald 102d8b859a2SMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 103d8b859a2SMilanka Ringwald btstack_run_loop_set_timer_handler(&connection->configuration_timer, avdtp_configuration_timeout_handler); 104d8b859a2SMilanka Ringwald btstack_run_loop_set_timer_context(&connection->configuration_timer, connection); 105d8b859a2SMilanka Ringwald btstack_run_loop_set_timer(&connection->configuration_timer, CONFIGURATION_TIMEOUT_MS); 106d8b859a2SMilanka Ringwald btstack_run_loop_add_timer(&connection->configuration_timer); 107d8b859a2SMilanka Ringwald } 108d8b859a2SMilanka Ringwald 109d8b859a2SMilanka Ringwald void avdtp_configuration_timer_stop(avdtp_connection_t * connection){ 110d8b859a2SMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 111d8b859a2SMilanka Ringwald } 112d8b859a2SMilanka Ringwald 113b0d75c91SMilanka Ringwald static uint16_t avdtp_get_next_initiator_transaction_label(avdtp_context_t * context){ 114b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter++; 115b0d75c91SMilanka Ringwald if (context->initiator_transaction_id_counter == 0){ 116b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter = 1; 117b0d75c91SMilanka Ringwald } 118b0d75c91SMilanka Ringwald return context->initiator_transaction_id_counter; 119b0d75c91SMilanka Ringwald } 120b0d75c91SMilanka Ringwald 121b4202ff1SMatthias Ringwald static uint16_t avdtp_get_next_avdtp_cid(void){ 1224ccacc40SMilanka Ringwald avdtp_cid_counter++; 1234ccacc40SMilanka Ringwald if (avdtp_cid_counter == 0){ 1244ccacc40SMilanka Ringwald avdtp_cid_counter = 1; 1254ccacc40SMilanka Ringwald } 1264ccacc40SMilanka Ringwald return avdtp_cid_counter; 1274ccacc40SMilanka Ringwald } 1284ccacc40SMilanka Ringwald 1294ccacc40SMilanka Ringwald static uint16_t avdtp_get_next_local_seid(avdtp_context_t * context){ 1304ccacc40SMilanka Ringwald context->stream_endpoints_id_counter++; 1314ccacc40SMilanka Ringwald if (context->stream_endpoints_id_counter == 0){ 1324ccacc40SMilanka Ringwald context->stream_endpoints_id_counter = 1; 1334ccacc40SMilanka Ringwald } 1344ccacc40SMilanka Ringwald return context->stream_endpoints_id_counter; 1354ccacc40SMilanka Ringwald } 1364ccacc40SMilanka Ringwald 1374ccacc40SMilanka Ringwald uint8_t avdtp_connect(bd_addr_t remote, avdtp_sep_type_t query_role, avdtp_context_t * avdtp_context, uint16_t * avdtp_cid){ 1382f6083d0SMilanka Ringwald sdp_query_context = avdtp_context; 139692c0605SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_bd_addr(remote, avdtp_context); 140692c0605SMilanka Ringwald if (!connection){ 141692c0605SMilanka Ringwald connection = avdtp_create_connection(remote, avdtp_context); 1424567cc17SMilanka Ringwald if (!connection){ 143*9900b7faSMilanka Ringwald log_error("Not enough memory to create connection."); 1444567cc17SMilanka Ringwald return BTSTACK_MEMORY_ALLOC_FAILED; 1454567cc17SMilanka Ringwald } 146692c0605SMilanka Ringwald } 1474ccacc40SMilanka Ringwald 1485448c259SMilanka Ringwald *avdtp_cid = connection->avdtp_cid; 1491e1ae2bcSMilanka Ringwald if (!avdtp_cid) { 1501e1ae2bcSMilanka Ringwald return L2CAP_LOCAL_CID_DOES_NOT_EXIST; 1511e1ae2bcSMilanka Ringwald } 1525448c259SMilanka Ringwald avdtp_context->avdtp_cid = connection->avdtp_cid; 1535448c259SMilanka Ringwald 1545448c259SMilanka Ringwald uint8_t err; 1555448c259SMilanka Ringwald switch (connection->state){ 1565448c259SMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_IDLE: 1575448c259SMilanka Ringwald connection->state = AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE; 15863331bf4SMilanka Ringwald connection->is_initiator = 1; 1595448c259SMilanka Ringwald sdp_query_context = avdtp_context; 1602f6083d0SMilanka Ringwald avdtp_context->avdtp_l2cap_psm = 0; 1612f6083d0SMilanka Ringwald avdtp_context->avdtp_version = 0; 1622f6083d0SMilanka Ringwald avdtp_context->query_role = query_role; 1635448c259SMilanka Ringwald err = sdp_client_query_uuid16(&avdtp_handle_sdp_client_query_result, remote, BLUETOOTH_PROTOCOL_AVDTP); 1641e1ae2bcSMilanka Ringwald if (err != ERROR_CODE_SUCCESS){ 1651e1ae2bcSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 1662f6083d0SMilanka Ringwald btstack_linked_list_remove(&avdtp_context->connections, (btstack_linked_item_t*) connection); 1672f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 1685448c259SMilanka Ringwald } 1691e1ae2bcSMilanka Ringwald return err; 170a0b8a58cSMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_OPENED:{ 171a0b8a58cSMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_signaling_cid(connection->l2cap_signaling_cid, avdtp_context); 172a0b8a58cSMilanka Ringwald if (stream_endpoint){ 173a0b8a58cSMilanka 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); 174a0b8a58cSMilanka Ringwald break; 175a0b8a58cSMilanka Ringwald } 176596b7fdcSMilanka Ringwald avdtp_signaling_emit_connection_established(avdtp_context->avdtp_callback, connection->avdtp_cid, connection->remote_addr, ERROR_CODE_SUCCESS); 1775448c259SMilanka Ringwald break; 178a0b8a58cSMilanka Ringwald } 1795448c259SMilanka Ringwald default: 1805448c259SMilanka Ringwald log_error("avdtp_connect: sink in wrong state"); 1815448c259SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 1825448c259SMilanka Ringwald 1831e1ae2bcSMilanka Ringwald } 1844ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 185692c0605SMilanka Ringwald } 186747ec646SMilanka Ringwald 187747ec646SMilanka Ringwald void avdtp_register_media_transport_category(avdtp_stream_endpoint_t * stream_endpoint){ 188747ec646SMilanka Ringwald if (!stream_endpoint){ 189*9900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 190747ec646SMilanka Ringwald return; 191747ec646SMilanka Ringwald } 192747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MEDIA_TRANSPORT, 1); 193747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 194747ec646SMilanka Ringwald } 195747ec646SMilanka Ringwald 196747ec646SMilanka Ringwald void avdtp_register_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 197747ec646SMilanka Ringwald if (!stream_endpoint){ 198*9900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 199747ec646SMilanka Ringwald return; 200747ec646SMilanka Ringwald } 201747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_REPORTING, 1); 202747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 203747ec646SMilanka Ringwald } 204747ec646SMilanka Ringwald 205747ec646SMilanka Ringwald void avdtp_register_delay_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 206747ec646SMilanka Ringwald if (!stream_endpoint){ 207*9900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 208747ec646SMilanka Ringwald return; 209747ec646SMilanka Ringwald } 210747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_DELAY_REPORTING, 1); 211747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 212747ec646SMilanka Ringwald } 213747ec646SMilanka Ringwald 214747ec646SMilanka Ringwald void avdtp_register_recovery_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t maximum_recovery_window_size, uint8_t maximum_number_media_packets){ 215747ec646SMilanka Ringwald if (!stream_endpoint){ 216*9900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 217747ec646SMilanka Ringwald return; 218747ec646SMilanka Ringwald } 219747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_RECOVERY, 1); 220747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 221747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.recovery_type = 0x01; // 0x01 = RFC2733 222747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_recovery_window_size = maximum_recovery_window_size; 223747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_number_media_packets = maximum_number_media_packets; 224747ec646SMilanka Ringwald } 225747ec646SMilanka Ringwald 226747ec646SMilanka 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){ 227747ec646SMilanka Ringwald if (!stream_endpoint){ 228*9900b7faSMilanka 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_CONTENT_PROTECTION, 1); 232747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 233747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type = cp_type; 23467ae582dSMilanka Ringwald memcpy(stream_endpoint->sep.capabilities.content_protection.cp_type_value, cp_type_value, btstack_min(cp_type_value_len, AVDTP_MAX_CONTENT_PROTECTION_TYPE_VALUE_LEN)); 23567ae582dSMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type_value_len = btstack_min(cp_type_value_len, AVDTP_MAX_CONTENT_PROTECTION_TYPE_VALUE_LEN); 236747ec646SMilanka Ringwald } 237747ec646SMilanka Ringwald 238747ec646SMilanka Ringwald void avdtp_register_header_compression_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t back_ch, uint8_t media, uint8_t recovery){ 239747ec646SMilanka Ringwald if (!stream_endpoint){ 240*9900b7faSMilanka 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_HEADER_COMPRESSION, 1); 244747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 245747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.back_ch = back_ch; 246747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.media = media; 247747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.recovery = recovery; 248747ec646SMilanka Ringwald } 249747ec646SMilanka Ringwald 25078d08d09SMilanka 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){ 251747ec646SMilanka Ringwald if (!stream_endpoint){ 252*9900b7faSMilanka 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_MEDIA_CODEC, 1); 256747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 257747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_type = media_type; 258747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_type = media_codec_type; 259747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information = media_codec_info; 260747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information_len = media_codec_info_len; 261747ec646SMilanka Ringwald } 262747ec646SMilanka Ringwald 263747ec646SMilanka Ringwald void avdtp_register_multiplexing_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t fragmentation){ 264747ec646SMilanka Ringwald if (!stream_endpoint){ 265*9900b7faSMilanka Ringwald log_error("Stream endpoint with given seid is not registered."); 266747ec646SMilanka Ringwald return; 267747ec646SMilanka Ringwald } 268747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MULTIPLEXING, 1); 269747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 270747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.multiplexing_mode.fragmentation = fragmentation; 271747ec646SMilanka Ringwald } 272747ec646SMilanka Ringwald 273747ec646SMilanka Ringwald 274747ec646SMilanka Ringwald /* START: tracking can send now requests pro l2cap cid */ 275747ec646SMilanka Ringwald void avdtp_handle_can_send_now(avdtp_connection_t * connection, uint16_t l2cap_cid, avdtp_context_t * context){ 276747ec646SMilanka Ringwald if (connection->wait_to_send_acceptor){ 277747ec646SMilanka Ringwald connection->wait_to_send_acceptor = 0; 278747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm_run(connection, context); 279747ec646SMilanka Ringwald } else if (connection->wait_to_send_initiator){ 280747ec646SMilanka Ringwald connection->wait_to_send_initiator = 0; 281747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm_run(connection, context); 282747ec646SMilanka Ringwald } else if (connection->wait_to_send_self){ 283747ec646SMilanka Ringwald connection->wait_to_send_self = 0; 284747ec646SMilanka Ringwald if (connection->disconnect){ 285747ec646SMilanka Ringwald btstack_linked_list_iterator_t it; 286747ec646SMilanka Ringwald btstack_linked_list_iterator_init(&it, &context->stream_endpoints); 287747ec646SMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 288747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 289747ec646SMilanka Ringwald if (stream_endpoint->connection == connection){ 290747ec646SMilanka Ringwald if (stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_OPENED && stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED){ 291747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED; 2929413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 293747ec646SMilanka Ringwald l2cap_disconnect(stream_endpoint->l2cap_media_cid, 0); 294747ec646SMilanka Ringwald return; 295747ec646SMilanka Ringwald } 296747ec646SMilanka Ringwald } 297747ec646SMilanka Ringwald } 298747ec646SMilanka Ringwald connection->disconnect = 0; 299747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED; 3009413b167SMilanka Ringwald l2cap_disconnect(connection->l2cap_signaling_cid, 0); 301747ec646SMilanka Ringwald return; 302747ec646SMilanka Ringwald } 303747ec646SMilanka Ringwald } 304747ec646SMilanka Ringwald 305747ec646SMilanka Ringwald // re-register 306747ec646SMilanka Ringwald int more_to_send = connection->wait_to_send_acceptor || connection->wait_to_send_initiator || connection->wait_to_send_self; 307747ec646SMilanka Ringwald if (more_to_send){ 308747ec646SMilanka Ringwald l2cap_request_can_send_now_event(l2cap_cid); 309747ec646SMilanka Ringwald } 310747ec646SMilanka Ringwald } 311747ec646SMilanka Ringwald /* END: tracking can send now requests pro l2cap cid */ 312747ec646SMilanka Ringwald 313747ec646SMilanka Ringwald avdtp_connection_t * avdtp_create_connection(bd_addr_t remote_addr, avdtp_context_t * context){ 314747ec646SMilanka Ringwald avdtp_connection_t * connection = btstack_memory_avdtp_connection_get(); 3154567cc17SMilanka Ringwald if (!connection){ 316*9900b7faSMilanka Ringwald log_error("Not enough memory to create connection"); 3174567cc17SMilanka Ringwald return NULL; 3184567cc17SMilanka Ringwald } 319747ec646SMilanka Ringwald memset(connection, 0, sizeof(avdtp_connection_t)); 320747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 321b0d75c91SMilanka Ringwald connection->initiator_transaction_label = avdtp_get_next_initiator_transaction_label(context); 322b0d75c91SMilanka Ringwald connection->avdtp_cid = avdtp_get_next_avdtp_cid(); 32363331bf4SMilanka Ringwald context->avdtp_cid = connection->avdtp_cid; 324747ec646SMilanka Ringwald memcpy(connection->remote_addr, remote_addr, 6); 325747ec646SMilanka Ringwald btstack_linked_list_add(&context->connections, (btstack_linked_item_t *) connection); 326747ec646SMilanka Ringwald return connection; 327747ec646SMilanka Ringwald } 328747ec646SMilanka Ringwald 329747ec646SMilanka 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){ 330747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = btstack_memory_avdtp_stream_endpoint_get(); 3314567cc17SMilanka Ringwald if (!stream_endpoint){ 332*9900b7faSMilanka Ringwald log_error("Not enough memory to create stream endpoint"); 3334567cc17SMilanka Ringwald return NULL; 3344567cc17SMilanka Ringwald } 335747ec646SMilanka Ringwald memset(stream_endpoint, 0, sizeof(avdtp_stream_endpoint_t)); 3364ccacc40SMilanka Ringwald stream_endpoint->sep.seid = avdtp_get_next_local_seid(context); 337747ec646SMilanka Ringwald stream_endpoint->sep.media_type = media_type; 338747ec646SMilanka Ringwald stream_endpoint->sep.type = sep_type; 339747ec646SMilanka Ringwald btstack_linked_list_add(&context->stream_endpoints, (btstack_linked_item_t *) stream_endpoint); 340747ec646SMilanka Ringwald return stream_endpoint; 341747ec646SMilanka Ringwald } 342747ec646SMilanka Ringwald 343747ec646SMilanka Ringwald 344747ec646SMilanka Ringwald static void handle_l2cap_data_packet_for_signaling_connection(avdtp_connection_t * connection, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 345747ec646SMilanka Ringwald int offset = avdtp_read_signaling_header(&connection->signaling_packet, packet, size); 346747ec646SMilanka Ringwald switch (connection->signaling_packet.message_type){ 347747ec646SMilanka Ringwald case AVDTP_CMD_MSG: 348747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm(connection, packet, size, offset, context); 349747ec646SMilanka Ringwald break; 350747ec646SMilanka Ringwald default: 351747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm(connection, packet, size, offset, context); 352747ec646SMilanka Ringwald break; 353747ec646SMilanka Ringwald } 354747ec646SMilanka Ringwald } 355747ec646SMilanka Ringwald 356692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 3572f6083d0SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(sdp_query_context->avdtp_cid, sdp_query_context); 3582f6083d0SMilanka Ringwald if (!connection) { 359*9900b7faSMilanka Ringwald log_error("Ndp query, connection with 0x%02x cid not found", sdp_query_context->avdtp_cid); 3602f6083d0SMilanka Ringwald return; 3612f6083d0SMilanka Ringwald } 3622f6083d0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE) return; 3632f6083d0SMilanka Ringwald 364692c0605SMilanka Ringwald UNUSED(packet_type); 365692c0605SMilanka Ringwald UNUSED(channel); 366692c0605SMilanka Ringwald UNUSED(size); 367692c0605SMilanka Ringwald 368692c0605SMilanka Ringwald des_iterator_t des_list_it; 369692c0605SMilanka Ringwald des_iterator_t prot_it; 3701e1ae2bcSMilanka Ringwald uint8_t status; 371692c0605SMilanka Ringwald 372692c0605SMilanka Ringwald switch (hci_event_packet_get_type(packet)){ 373692c0605SMilanka Ringwald case SDP_EVENT_QUERY_ATTRIBUTE_VALUE: 374692c0605SMilanka Ringwald // Handle new SDP record 375692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_record_id(packet) != record_id) { 376692c0605SMilanka Ringwald record_id = sdp_event_query_attribute_byte_get_record_id(packet); 3778587e32cSMilanka Ringwald // log_info("SDP Record: Nr: %d", record_id); 378692c0605SMilanka Ringwald } 379692c0605SMilanka Ringwald 380692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_attribute_length(packet) <= attribute_value_buffer_size) { 381692c0605SMilanka Ringwald attribute_value[sdp_event_query_attribute_byte_get_data_offset(packet)] = sdp_event_query_attribute_byte_get_data(packet); 382692c0605SMilanka Ringwald 383692c0605SMilanka Ringwald if ((uint16_t)(sdp_event_query_attribute_byte_get_data_offset(packet)+1) == sdp_event_query_attribute_byte_get_attribute_length(packet)) { 384692c0605SMilanka Ringwald 385692c0605SMilanka Ringwald switch(sdp_event_query_attribute_byte_get_attribute_id(packet)) { 386692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_SERVICE_CLASS_ID_LIST: 387692c0605SMilanka Ringwald if (de_get_element_type(attribute_value) != DE_DES) break; 388692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 389692c0605SMilanka Ringwald uint8_t * element = des_iterator_get_element(&des_list_it); 390692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 391692c0605SMilanka Ringwald uint32_t uuid = de_get_uuid32(element); 392692c0605SMilanka Ringwald switch (uuid){ 393692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SOURCE: 3942f6083d0SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SOURCE) { 3952f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 396692c0605SMilanka Ringwald break; 397692c0605SMilanka Ringwald } 3988587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SOURCE protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 399eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 400692c0605SMilanka Ringwald break; 401692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SINK: 4025448c259SMilanka Ringwald if (sdp_query_context->query_role == AVDTP_SINK) { 4032f6083d0SMilanka Ringwald sdp_query_context->role_supported = 1; 404692c0605SMilanka Ringwald break; 405692c0605SMilanka Ringwald } 4068587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SINK protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 407eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 408692c0605SMilanka Ringwald break; 409692c0605SMilanka Ringwald default: 410692c0605SMilanka Ringwald break; 411692c0605SMilanka Ringwald } 412692c0605SMilanka Ringwald } 413692c0605SMilanka Ringwald break; 414692c0605SMilanka Ringwald 415692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_PROTOCOL_DESCRIPTOR_LIST: { 4168587e32cSMilanka Ringwald // log_info("SDP Attribute: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet)); 417692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 418692c0605SMilanka Ringwald uint8_t *des_element; 419692c0605SMilanka Ringwald uint8_t *element; 420692c0605SMilanka Ringwald uint32_t uuid; 421692c0605SMilanka Ringwald 422692c0605SMilanka Ringwald if (des_iterator_get_type(&des_list_it) != DE_DES) continue; 423692c0605SMilanka Ringwald 424692c0605SMilanka Ringwald des_element = des_iterator_get_element(&des_list_it); 425692c0605SMilanka Ringwald des_iterator_init(&prot_it, des_element); 426692c0605SMilanka Ringwald element = des_iterator_get_element(&prot_it); 427692c0605SMilanka Ringwald 428692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 429692c0605SMilanka Ringwald 430692c0605SMilanka Ringwald uuid = de_get_uuid32(element); 431692c0605SMilanka Ringwald switch (uuid){ 432692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_L2CAP: 433692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 434692c0605SMilanka Ringwald des_iterator_next(&prot_it); 4352f6083d0SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_l2cap_psm); 436692c0605SMilanka Ringwald break; 437692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_AVDTP: 438692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 439692c0605SMilanka Ringwald des_iterator_next(&prot_it); 4402f6083d0SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_version); 441692c0605SMilanka Ringwald break; 442692c0605SMilanka Ringwald default: 443692c0605SMilanka Ringwald break; 444692c0605SMilanka Ringwald } 445692c0605SMilanka Ringwald } 446692c0605SMilanka Ringwald } 447692c0605SMilanka Ringwald break; 448692c0605SMilanka Ringwald default: 449692c0605SMilanka Ringwald break; 450692c0605SMilanka Ringwald } 451692c0605SMilanka Ringwald } 452692c0605SMilanka Ringwald } else { 4538587e32cSMilanka 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)); 454692c0605SMilanka Ringwald } 455692c0605SMilanka Ringwald break; 456692c0605SMilanka Ringwald 457692c0605SMilanka Ringwald case SDP_EVENT_QUERY_COMPLETE: 4581e1ae2bcSMilanka Ringwald status = sdp_event_query_complete_get_status(packet); 4592f6083d0SMilanka Ringwald if (status != ERROR_CODE_SUCCESS){ 4605448c259SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, status); 4612f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4622f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 463*9900b7faSMilanka Ringwald log_info("NDP query failed with status 0x%02x.", status); 4641e1ae2bcSMilanka Ringwald break; 4651e1ae2bcSMilanka Ringwald } 4662f6083d0SMilanka Ringwald if (!sdp_query_context->role_supported){ 4672f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection); 4682f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 4692f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, SDP_SERVICE_NOT_FOUND); 470*9900b7faSMilanka Ringwald log_info("NDP query, remote device does not support required role."); 471974d4d6eSMilanka Ringwald break; 472974d4d6eSMilanka Ringwald } 4732f6083d0SMilanka Ringwald if (!sdp_query_context->avdtp_l2cap_psm) { 4742f6083d0SMilanka Ringwald btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*)connection); 4752f6083d0SMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 4762f6083d0SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, L2CAP_SERVICE_DOES_NOT_EXIST); 477*9900b7faSMilanka Ringwald log_info("NDP query, no l2cap psm found."); 4782f6083d0SMilanka Ringwald break; 4792f6083d0SMilanka Ringwald } 4802f6083d0SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 4812f6083d0SMilanka Ringwald l2cap_create_channel(sdp_query_context->packet_handler, connection->remote_addr, sdp_query_context->avdtp_l2cap_psm, l2cap_max_mtu(), NULL); 482692c0605SMilanka Ringwald break; 483692c0605SMilanka Ringwald } 484692c0605SMilanka Ringwald } 485692c0605SMilanka Ringwald 486692c0605SMilanka Ringwald 487747ec646SMilanka Ringwald void avdtp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 488747ec646SMilanka Ringwald bd_addr_t event_addr; 489747ec646SMilanka Ringwald uint16_t psm; 490747ec646SMilanka Ringwald uint16_t local_cid; 4911e1ae2bcSMilanka Ringwald uint8_t status; 492747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = NULL; 493747ec646SMilanka Ringwald avdtp_connection_t * connection = NULL; 494747ec646SMilanka Ringwald btstack_linked_list_t * avdtp_connections = &context->connections; 495747ec646SMilanka Ringwald btstack_linked_list_t * stream_endpoints = &context->stream_endpoints; 496747ec646SMilanka Ringwald handle_media_data = context->handle_media_data; 4978587e32cSMilanka Ringwald // log_info("avdtp_packet_handler packet type %02x, event %02x ", packet_type, hci_event_packet_get_type(packet)); 498747ec646SMilanka Ringwald switch (packet_type) { 499747ec646SMilanka Ringwald case L2CAP_DATA_PACKET: 5009413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 501747ec646SMilanka Ringwald if (connection){ 502747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 503747ec646SMilanka Ringwald break; 504747ec646SMilanka Ringwald } 505747ec646SMilanka Ringwald 506747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 507747ec646SMilanka Ringwald if (!stream_endpoint){ 508747ec646SMilanka Ringwald if (!connection) break; 509747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 510747ec646SMilanka Ringwald break; 511747ec646SMilanka Ringwald } 512747ec646SMilanka Ringwald 5138c0f3635SMilanka Ringwald if (stream_endpoint->connection){ 5149413b167SMilanka Ringwald if (channel == stream_endpoint->connection->l2cap_signaling_cid){ 515a466d508SMilanka Ringwald int offset = avdtp_read_signaling_header(&stream_endpoint->connection->signaling_packet, packet, size); 516a466d508SMilanka Ringwald if (stream_endpoint->connection->signaling_packet.message_type == AVDTP_CMD_MSG){ 517a466d508SMilanka Ringwald avdtp_acceptor_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context); 518a466d508SMilanka Ringwald } else { 519a466d508SMilanka Ringwald avdtp_initiator_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context); 520a466d508SMilanka Ringwald } 521747ec646SMilanka Ringwald break; 522747ec646SMilanka Ringwald } 5238c0f3635SMilanka Ringwald } 524747ec646SMilanka Ringwald 525747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_media_cid){ 5268b097e29SMilanka Ringwald if (handle_media_data){ 527fd58c900SMilanka Ringwald (*handle_media_data)(avdtp_local_seid(stream_endpoint), packet, size); 5288b097e29SMilanka Ringwald } 529747ec646SMilanka Ringwald break; 530747ec646SMilanka Ringwald } 531747ec646SMilanka Ringwald 532747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_reporting_cid){ 533747ec646SMilanka Ringwald // TODO 5348587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for reporting: NOT IMPLEMENTED"); 535747ec646SMilanka Ringwald } else if (channel == stream_endpoint->l2cap_recovery_cid){ 536747ec646SMilanka Ringwald // TODO 5378587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for recovery: NOT IMPLEMENTED"); 538747ec646SMilanka Ringwald } else { 539747ec646SMilanka Ringwald log_error("avdtp packet handler L2CAP_DATA_PACKET: local cid 0x%02x not found", channel); 540747ec646SMilanka Ringwald } 541747ec646SMilanka Ringwald break; 542747ec646SMilanka Ringwald 543747ec646SMilanka Ringwald case HCI_EVENT_PACKET: 544747ec646SMilanka Ringwald switch (hci_event_packet_get_type(packet)) { 545747ec646SMilanka Ringwald case L2CAP_EVENT_INCOMING_CONNECTION: 546747ec646SMilanka Ringwald l2cap_event_incoming_connection_get_address(packet, event_addr); 547747ec646SMilanka Ringwald local_cid = l2cap_event_incoming_connection_get_local_cid(packet); 548747ec646SMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 549*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_INCOMING_CONNECTION, local cid 0x%02x ", local_cid); 550*9900b7faSMilanka Ringwald if (connection){ 551*9900b7faSMilanka Ringwald log_info("Nonnection state %d", connection->state); 552*9900b7faSMilanka Ringwald } 55349dda273SMilanka Ringwald if (!connection || connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED){ 554747ec646SMilanka Ringwald connection = avdtp_create_connection(event_addr, context); 55563331bf4SMilanka Ringwald connection->is_initiator = 0; 556*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_INCOMING_CONNECTION: role is_initiator %d", connection->is_initiator); 557747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 558*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_INCOMING_CONNECTION, connection %p, state connection %d, avdtp cid 0x%02x", connection, connection->state, connection->avdtp_cid); 559747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 560747ec646SMilanka Ringwald break; 561747ec646SMilanka Ringwald } 562747ec646SMilanka Ringwald 5636b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 564747ec646SMilanka Ringwald if (!stream_endpoint) { 565*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_INCOMING_CONNECTION no streamendpoint found for seid %d", connection->local_seid); 566747ec646SMilanka Ringwald break; 567747ec646SMilanka Ringwald } 568747ec646SMilanka Ringwald 569*9900b7faSMilanka Ringwald log_info("Nhecking 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); 570747ec646SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 571*9900b7faSMilanka Ringwald if (connection->is_initiator){ 572*9900b7faSMilanka Ringwald l2cap_decline_connection(local_cid); 573*9900b7faSMilanka Ringwald break; 574*9900b7faSMilanka Ringwald } 575747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 576747ec646SMilanka Ringwald break; 577747ec646SMilanka Ringwald } 578*9900b7faSMilanka Ringwald l2cap_decline_connection(local_cid); 579747ec646SMilanka Ringwald break; 580747ec646SMilanka Ringwald 581747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_OPENED: 582a0b8a58cSMilanka Ringwald psm = l2cap_event_channel_opened_get_psm(packet); 583a0b8a58cSMilanka Ringwald if (psm != BLUETOOTH_PROTOCOL_AVDTP){ 584*9900b7faSMilanka Ringwald log_info("Nnexpected PSM - Not implemented yet, avdtp sink: L2CAP_EVENT_CHANNEL_OPENED "); 585a0b8a58cSMilanka Ringwald return; 586a0b8a58cSMilanka Ringwald } 587a0b8a58cSMilanka Ringwald 5881e1ae2bcSMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 589747ec646SMilanka Ringwald // inform about new l2cap connection 590747ec646SMilanka Ringwald l2cap_event_channel_opened_get_address(packet, event_addr); 5917050d2caSMilanka Ringwald local_cid = l2cap_event_channel_opened_get_local_cid(packet); 592a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 593a0b8a58cSMilanka Ringwald 594*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_CHANNEL_OPENED: status %d, cid 0x%02x , signaling connection %p ", status, local_cid, connection); 595a0b8a58cSMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 596a0b8a58cSMilanka Ringwald if (!connection){ 597*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_CHANNEL_OPENED 2: status %d, signaling connection %p ", status, connection); 598a0b8a58cSMilanka Ringwald break; 599a0b8a58cSMilanka Ringwald } 600a0b8a58cSMilanka Ringwald 601a0b8a58cSMilanka Ringwald if (connection->l2cap_signaling_cid == 0) { 6021e1ae2bcSMilanka Ringwald if (status){ 603*9900b7faSMilanka Ringwald log_info("N2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 60455ddebccSMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 605a0b8a58cSMilanka Ringwald connection->l2cap_signaling_cid = 0; 6064ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, l2cap_event_channel_opened_get_status(packet)); 607747ec646SMilanka Ringwald break; 608747ec646SMilanka Ringwald } 609a0b8a58cSMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED) { 610*9900b7faSMilanka Ringwald log_info("N2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status); 611a0b8a58cSMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, AVDTP_CONNECTION_IN_WRONG_STATE); 612a0b8a58cSMilanka Ringwald break; 61355ddebccSMilanka Ringwald } 6149413b167SMilanka Ringwald connection->l2cap_signaling_cid = local_cid; 6156b0ee1d0SMilanka Ringwald connection->local_seid = 0; 616747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_OPENED; 617*9900b7faSMilanka Ringwald log_info("NVDTP_SIGNALING_CONNECTION_OPENED, connection %p, l2cap_signaling_cid 0x%02x, avdtp_cid 0x%02x", connection, local_cid, connection->avdtp_cid); 6184ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, 0); 619747ec646SMilanka Ringwald break; 620747ec646SMilanka Ringwald } 621747ec646SMilanka Ringwald 6226b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 623747ec646SMilanka Ringwald if (!stream_endpoint){ 624*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_CHANNEL_OPENED: stream_endpoint not found"); 625747ec646SMilanka Ringwald return; 626747ec646SMilanka Ringwald } 627a466d508SMilanka Ringwald 628a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 629a466d508SMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 630a466d508SMilanka Ringwald if (status){ 631*9900b7faSMilanka Ringwald log_info("NVDTP_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)); 632a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_IDLE; 633596b7fdcSMilanka 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); 634a466d508SMilanka Ringwald break; 635a466d508SMilanka Ringwald } 636a466d508SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED){ 637*9900b7faSMilanka Ringwald log_info("NVDTP_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)); 638596b7fdcSMilanka 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); 639a466d508SMilanka Ringwald break; 640a466d508SMilanka Ringwald } 641a466d508SMilanka Ringwald 642a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 643a466d508SMilanka Ringwald stream_endpoint->connection = connection; 644a466d508SMilanka Ringwald stream_endpoint->l2cap_media_cid = l2cap_event_channel_opened_get_local_cid(packet); 645a466d508SMilanka Ringwald stream_endpoint->media_con_handle = l2cap_event_channel_opened_get_handle(packet); 646d1207cd8SMilanka Ringwald 647*9900b7faSMilanka Ringwald log_info("NVDTP_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)); 648596b7fdcSMilanka 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); 649a466d508SMilanka Ringwald return; 650a466d508SMilanka Ringwald } 651747ec646SMilanka Ringwald break; 652747ec646SMilanka Ringwald 653747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_CLOSED: 654747ec646SMilanka Ringwald local_cid = l2cap_event_channel_closed_get_local_cid(packet); 6559413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(local_cid, context); 6560bd7cb1fSMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(local_cid, context); 657*9900b7faSMilanka Ringwald log_info("Neceived L2CAP_EVENT_CHANNEL_CLOSED, cid 0x%2x, connection %p, stream_endpoint %p", local_cid, connection, stream_endpoint); 6580bd7cb1fSMilanka Ringwald 659a466d508SMilanka Ringwald if (stream_endpoint){ 660a466d508SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == local_cid){ 661a466d508SMilanka Ringwald connection = stream_endpoint->connection; 662596b7fdcSMilanka Ringwald if (connection) { 663596b7fdcSMilanka Ringwald avdtp_streaming_emit_connection_released(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint)); 664596b7fdcSMilanka Ringwald } 665a466d508SMilanka Ringwald stream_endpoint->l2cap_media_cid = 0; 666a466d508SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_IDLE; 667a466d508SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_STREAM_CONFIG_IDLE; 668a466d508SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_STREAM_CONFIG_IDLE; 669a466d508SMilanka Ringwald stream_endpoint->remote_sep_index = 0; 670a466d508SMilanka Ringwald if (connection && connection->disconnect){ 671a466d508SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 672a466d508SMilanka Ringwald } 673a466d508SMilanka Ringwald break; 674a466d508SMilanka Ringwald } 675a466d508SMilanka Ringwald if (stream_endpoint->l2cap_recovery_cid == local_cid){ 676*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_CHANNEL_CLOSED recovery cid 0x%0x", local_cid); 677a466d508SMilanka Ringwald stream_endpoint->l2cap_recovery_cid = 0; 678a466d508SMilanka Ringwald break; 679a466d508SMilanka Ringwald } 680a466d508SMilanka Ringwald 681a466d508SMilanka Ringwald if (stream_endpoint->l2cap_reporting_cid == local_cid){ 682*9900b7faSMilanka Ringwald log_info("N2CAP_EVENT_CHANNEL_CLOSED reporting cid 0x%0x", local_cid); 683a466d508SMilanka Ringwald stream_endpoint->l2cap_reporting_cid = 0; 684a466d508SMilanka Ringwald break; 685a466d508SMilanka Ringwald } 686a466d508SMilanka Ringwald } 687596b7fdcSMilanka Ringwald 688596b7fdcSMilanka Ringwald if (connection){ 689bdbc3ef6SMilanka Ringwald btstack_run_loop_remove_timer(&connection->configuration_timer); 690596b7fdcSMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 691bdbc3ef6SMilanka Ringwald 692596b7fdcSMilanka Ringwald btstack_linked_list_remove(avdtp_connections, (btstack_linked_item_t*) connection); 693596b7fdcSMilanka Ringwald btstack_linked_list_iterator_t it; 694596b7fdcSMilanka Ringwald btstack_linked_list_iterator_init(&it, stream_endpoints); 695596b7fdcSMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 696596b7fdcSMilanka Ringwald avdtp_stream_endpoint_t * _stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 697596b7fdcSMilanka Ringwald 698596b7fdcSMilanka Ringwald if (_stream_endpoint->connection == connection){ 699596b7fdcSMilanka Ringwald avdtp_initialize_stream_endpoint(_stream_endpoint); 700596b7fdcSMilanka Ringwald } 701596b7fdcSMilanka Ringwald } 702596b7fdcSMilanka Ringwald btstack_memory_avdtp_connection_free(connection); 703596b7fdcSMilanka Ringwald break; 704596b7fdcSMilanka Ringwald } 705596b7fdcSMilanka Ringwald 706747ec646SMilanka Ringwald break; 707747ec646SMilanka Ringwald 708747ec646SMilanka Ringwald case HCI_EVENT_DISCONNECTION_COMPLETE: 709747ec646SMilanka Ringwald break; 710747ec646SMilanka Ringwald 711747ec646SMilanka Ringwald case L2CAP_EVENT_CAN_SEND_NOW: 7129413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 713747ec646SMilanka Ringwald if (!connection) { 714747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 715747ec646SMilanka Ringwald if (!stream_endpoint->connection) break; 716747ec646SMilanka Ringwald connection = stream_endpoint->connection; 717747ec646SMilanka Ringwald } 718747ec646SMilanka Ringwald avdtp_handle_can_send_now(connection, channel, context); 719747ec646SMilanka Ringwald break; 720747ec646SMilanka Ringwald default: 721*9900b7faSMilanka Ringwald log_info("Nnknown HCI event type %02x", hci_event_packet_get_type(packet)); 722747ec646SMilanka Ringwald break; 723747ec646SMilanka Ringwald } 724747ec646SMilanka Ringwald break; 725747ec646SMilanka Ringwald 726747ec646SMilanka Ringwald default: 727747ec646SMilanka Ringwald // other packet type 728747ec646SMilanka Ringwald break; 729747ec646SMilanka Ringwald } 730747ec646SMilanka Ringwald } 731747ec646SMilanka Ringwald 7324ccacc40SMilanka Ringwald uint8_t avdtp_disconnect(uint16_t avdtp_cid, avdtp_context_t * context){ 7334ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 7344ccacc40SMilanka Ringwald if (!connection) return BTSTACK_MEMORY_ALLOC_FAILED; 735a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_IDLE){ 736a466d508SMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 737a466d508SMilanka Ringwald return ERROR_CODE_SUCCESS; 738a466d508SMilanka Ringwald } 739a466d508SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED) return ERROR_CODE_SUCCESS; 740747ec646SMilanka Ringwald 741747ec646SMilanka Ringwald connection->disconnect = 1; 7429413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 7434ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 744747ec646SMilanka Ringwald } 745747ec646SMilanka Ringwald 7466b0ee1d0SMilanka Ringwald uint8_t avdtp_open_stream(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, avdtp_context_t * context){ 7474ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 748747ec646SMilanka Ringwald if (!connection){ 7498587e32cSMilanka Ringwald log_error("avdtp_media_connect: no connection for signaling cid 0x%02x found", avdtp_cid); 7504ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 751747ec646SMilanka Ringwald } 7526b0ee1d0SMilanka Ringwald if (avdtp_find_remote_sep(connection, remote_seid) == 0xFF){ 7536b0ee1d0SMilanka Ringwald log_error("avdtp_media_connect: no remote sep for seid %d found", remote_seid); 7544ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 755747ec646SMilanka Ringwald } 756747ec646SMilanka Ringwald 757747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) { 7588587e32cSMilanka Ringwald log_error("avdtp_media_connect: wrong connection state %d", connection->state); 7594ccacc40SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 760747ec646SMilanka Ringwald } 761747ec646SMilanka Ringwald 7626b0ee1d0SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 763747ec646SMilanka Ringwald if (!stream_endpoint) { 7646b0ee1d0SMilanka Ringwald log_error("avdtp_media_connect: no stream_endpoint with seid %d found", local_seid); 7654ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 766747ec646SMilanka Ringwald } 767747ec646SMilanka Ringwald 7684ccacc40SMilanka Ringwald if (stream_endpoint->state < AVDTP_STREAM_ENDPOINT_CONFIGURED) return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 7694ccacc40SMilanka Ringwald if (stream_endpoint->remote_sep_index == AVDTP_INVALID_SEP_INDEX) return AVDTP_SEID_DOES_NOT_EXIST; 770747ec646SMilanka Ringwald 771747ec646SMilanka Ringwald connection->initiator_transaction_label++; 7729413b167SMilanka Ringwald connection->remote_seid = remote_seid; 7736b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 774747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_OPEN_STREAM; 775747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W2_REQUEST_OPEN_STREAM; 7769413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 7774ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 778747ec646SMilanka Ringwald } 779747ec646SMilanka Ringwald 7804ccacc40SMilanka Ringwald uint8_t avdtp_start_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 7814ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 78246e6b063SMilanka Ringwald if (!connection){ 7834ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 7844ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 78546e6b063SMilanka Ringwald } 7865cfe7f4cSMilanka Ringwald 7874ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 7884ccacc40SMilanka Ringwald if (!stream_endpoint) { 7894ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no stream_endpoint with seid %d found", local_seid); 7904ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 7914ccacc40SMilanka Ringwald } 7924ccacc40SMilanka Ringwald 7934ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 7944ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no media connection for stream_endpoint with seid %d found", local_seid); 7954ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 7964ccacc40SMilanka Ringwald } 7974ccacc40SMilanka Ringwald 7984ccacc40SMilanka Ringwald if (stream_endpoint->remote_sep_index == AVDTP_INVALID_SEP_INDEX || stream_endpoint->start_stream){ 7994ccacc40SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 8004ccacc40SMilanka Ringwald } 8014ccacc40SMilanka Ringwald 80260ec20d0SMilanka Ringwald stream_endpoint->start_stream = 1; 8034ccacc40SMilanka Ringwald connection->local_seid = local_seid; 8049413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8054ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 806747ec646SMilanka Ringwald } 807747ec646SMilanka Ringwald 8084ccacc40SMilanka Ringwald uint8_t avdtp_stop_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8094ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 810747ec646SMilanka Ringwald if (!connection){ 8114ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8124ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 813747ec646SMilanka Ringwald } 8144ccacc40SMilanka Ringwald 8154ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8164ccacc40SMilanka Ringwald if (!stream_endpoint) { 8174ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no stream_endpoint with seid %d found", local_seid); 8184ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8194ccacc40SMilanka Ringwald } 8204ccacc40SMilanka Ringwald 8214ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8224ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8234ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8244ccacc40SMilanka Ringwald } 8254ccacc40SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF || stream_endpoint->stop_stream) return ERROR_CODE_SUCCESS; 8264ccacc40SMilanka Ringwald 82760ec20d0SMilanka Ringwald stream_endpoint->stop_stream = 1; 8284ccacc40SMilanka Ringwald connection->local_seid = local_seid; 8299413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8304ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 831747ec646SMilanka Ringwald } 832747ec646SMilanka Ringwald 8334ccacc40SMilanka Ringwald uint8_t avdtp_abort_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8344ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 83560ec20d0SMilanka Ringwald if (!connection){ 8364ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8374ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 838747ec646SMilanka Ringwald } 8394ccacc40SMilanka Ringwald 8404ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8414ccacc40SMilanka Ringwald if (!stream_endpoint) { 8424ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no stream_endpoint with seid %d found", local_seid); 8434ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8444ccacc40SMilanka Ringwald } 8454ccacc40SMilanka Ringwald 8464ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8474ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8484ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8494ccacc40SMilanka Ringwald } 8506b0ee1d0SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF || stream_endpoint->abort_stream) return ERROR_CODE_SUCCESS; 8514ccacc40SMilanka Ringwald 85260ec20d0SMilanka Ringwald stream_endpoint->abort_stream = 1; 8536b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 8549413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8554ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 856747ec646SMilanka Ringwald } 857747ec646SMilanka Ringwald 8584ccacc40SMilanka Ringwald uint8_t avdtp_suspend_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 8594ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 860747ec646SMilanka Ringwald if (!connection){ 8614ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 8624ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 86360ec20d0SMilanka Ringwald } 8644ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 8654ccacc40SMilanka Ringwald if (!stream_endpoint) { 8664ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no stream_endpoint with seid %d found", local_seid); 8674ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 8684ccacc40SMilanka Ringwald } 8694ccacc40SMilanka Ringwald 8704ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 8714ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no media connection for stream_endpoint with seid %d found", local_seid); 8724ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 8734ccacc40SMilanka Ringwald } 8746b0ee1d0SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF || stream_endpoint->suspend_stream) return ERROR_CODE_SUCCESS; 8754ccacc40SMilanka Ringwald 87660ec20d0SMilanka Ringwald stream_endpoint->suspend_stream = 1; 8774ccacc40SMilanka Ringwald connection->local_seid = local_seid; 8789413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 8794ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 880747ec646SMilanka Ringwald } 881747ec646SMilanka Ringwald 8829974aee0SMilanka Ringwald uint8_t avdtp_discover_stream_endpoints(uint16_t avdtp_cid, avdtp_context_t * context){ 8834ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 884747ec646SMilanka Ringwald if (!connection){ 8858587e32cSMilanka Ringwald log_error("avdtp_discover_stream_endpoints: no connection for signaling cid 0x%02x found", avdtp_cid); 8869974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 8879974aee0SMilanka Ringwald } 8889974aee0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED || 8899974aee0SMilanka Ringwald connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) { 8909974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 891747ec646SMilanka Ringwald } 892ec3d71e3SMilanka Ringwald 893747ec646SMilanka Ringwald connection->initiator_transaction_label++; 894747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_DISCOVER_SEPS; 8959974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 896747ec646SMilanka Ringwald } 897747ec646SMilanka Ringwald 898747ec646SMilanka Ringwald 8999974aee0SMilanka Ringwald uint8_t avdtp_get_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9004ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 901747ec646SMilanka Ringwald if (!connection){ 902*9900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9039974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 904747ec646SMilanka Ringwald } 9059974aee0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED || 9069974aee0SMilanka Ringwald connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) { 9079974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9089974aee0SMilanka Ringwald } 9099974aee0SMilanka Ringwald 910747ec646SMilanka Ringwald connection->initiator_transaction_label++; 911747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CAPABILITIES; 9129413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9139974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 914747ec646SMilanka Ringwald } 915747ec646SMilanka Ringwald 916747ec646SMilanka Ringwald 9179974aee0SMilanka Ringwald uint8_t avdtp_get_all_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9184ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 919747ec646SMilanka Ringwald if (!connection){ 920*9900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9219974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 922747ec646SMilanka Ringwald } 9239974aee0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED || 9249974aee0SMilanka Ringwald connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) { 9259974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9269974aee0SMilanka Ringwald } 9279974aee0SMilanka Ringwald 928747ec646SMilanka Ringwald connection->initiator_transaction_label++; 929747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_ALL_CAPABILITIES; 9309413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9319974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 932747ec646SMilanka Ringwald } 933747ec646SMilanka Ringwald 9349974aee0SMilanka Ringwald uint8_t avdtp_get_configuration(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 9354ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 936747ec646SMilanka Ringwald if (!connection){ 937*9900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9389974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 939747ec646SMilanka Ringwald } 9409974aee0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED || 9419974aee0SMilanka Ringwald connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) { 9429974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9439974aee0SMilanka Ringwald } 9449974aee0SMilanka Ringwald 945747ec646SMilanka Ringwald connection->initiator_transaction_label++; 946747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CONFIGURATION; 9479413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9489974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 949747ec646SMilanka Ringwald } 950747ec646SMilanka Ringwald 9519974aee0SMilanka 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){ 9524ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 953747ec646SMilanka Ringwald if (!connection){ 954*9900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9559974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 956747ec646SMilanka Ringwald } 9579974aee0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED || 9589974aee0SMilanka Ringwald connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) { 9599974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 9609974aee0SMilanka Ringwald } 961747ec646SMilanka Ringwald 9624ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 963747ec646SMilanka Ringwald if (!stream_endpoint) { 964*9900b7faSMilanka Ringwald log_error("No initiator stream endpoint for seid %d", local_seid); 9659974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST; 966747ec646SMilanka Ringwald } 967417b4996SMilanka Ringwald if (stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_CONFIGURED){ 968*9900b7faSMilanka Ringwald log_error("Ntream endpoint seid %d in wrong state", local_seid); 969417b4996SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 970417b4996SMilanka Ringwald } 971bdbc3ef6SMilanka Ringwald connection->active_stream_endpoint = (void*) stream_endpoint; 972d8b859a2SMilanka Ringwald connection->is_configuration_initiated_locally = 1; 97363331bf4SMilanka Ringwald connection->is_initiator = 1; 974747ec646SMilanka Ringwald 975747ec646SMilanka Ringwald connection->initiator_transaction_label++; 9769413b167SMilanka Ringwald connection->remote_seid = remote_seid; 9774ccacc40SMilanka Ringwald connection->local_seid = local_seid; 978f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 979f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 980747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION; 981ffa6c160SMilanka Ringwald 982ffa6c160SMilanka Ringwald // cache media codec information for SBC 983ffa6c160SMilanka Ringwald stream_endpoint->media_codec_type = configuration.media_codec.media_codec_type; 984ffa6c160SMilanka Ringwald if (configuration.media_codec.media_codec_type == AVDTP_CODEC_SBC){ 985ffa6c160SMilanka Ringwald stream_endpoint->media_type = configuration.media_codec.media_type; 986ffa6c160SMilanka Ringwald memcpy(stream_endpoint->media_codec_sbc_info, configuration.media_codec.media_codec_information, 4); 987ffa6c160SMilanka Ringwald } 9889974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 989747ec646SMilanka Ringwald } 990747ec646SMilanka Ringwald 9919974aee0SMilanka 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){ 9924ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 993747ec646SMilanka Ringwald if (!connection){ 994*9900b7faSMilanka Ringwald log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid); 9959974aee0SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 996747ec646SMilanka Ringwald } 997747ec646SMilanka Ringwald //TODO: if opened only app capabilities, enable reconfigure for not opened 9989974aee0SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED || 9999974aee0SMilanka Ringwald connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) { 10009974aee0SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 10019974aee0SMilanka Ringwald } 10029e42cfccSMilanka Ringwald 10034ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 100478d08d09SMilanka Ringwald if (!stream_endpoint) { 10054ccacc40SMilanka Ringwald log_error("avdtp_reconfigure: no initiator stream endpoint for seid %d", local_seid); 10069974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST; 100778d08d09SMilanka Ringwald } 100878d08d09SMilanka Ringwald 100978d08d09SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF){ 10108587e32cSMilanka Ringwald log_error("avdtp_reconfigure: no associated remote sep"); 10119974aee0SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 101278d08d09SMilanka Ringwald } 1013747ec646SMilanka Ringwald connection->initiator_transaction_label++; 10149413b167SMilanka Ringwald connection->remote_seid = remote_seid; 10156b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 1016f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 1017f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 1018747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_RECONFIGURE_STREAM_WITH_SEID; 10199974aee0SMilanka Ringwald return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 1020747ec646SMilanka Ringwald } 1021747ec646SMilanka Ringwald 1022f9bca1f3SMilanka Ringwald uint8_t avdtp_remote_seps_num(uint16_t avdtp_cid, avdtp_context_t * context){ 10234ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 1024747ec646SMilanka Ringwald if (!connection){ 10254ccacc40SMilanka Ringwald log_error("avdtp_suspend: no connection for AVDTP cid 0x%02x found", avdtp_cid); 10269e42cfccSMilanka Ringwald return 0; 10279e42cfccSMilanka Ringwald } 10289e42cfccSMilanka Ringwald return connection->remote_seps_num; 10299e42cfccSMilanka Ringwald } 10309e42cfccSMilanka Ringwald 1031f9bca1f3SMilanka Ringwald avdtp_sep_t * avdtp_remote_sep(uint16_t avdtp_cid, uint8_t index, avdtp_context_t * context){ 10324ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 10339e42cfccSMilanka Ringwald if (!connection){ 10344ccacc40SMilanka Ringwald log_error("avdtp_suspend: no connection for AVDTP cid 0x%02x found", avdtp_cid); 10359e42cfccSMilanka Ringwald return NULL; 10369e42cfccSMilanka Ringwald } 1037578eda3eSMilanka Ringwald if (index == AVDTP_INVALID_SEP_INDEX) return NULL; 1038578eda3eSMilanka Ringwald if (index >= connection->remote_seps_num || index >= AVDTP_MAX_NUM_SEPS) return NULL; 10399e42cfccSMilanka Ringwald return &connection->remote_seps[index]; 10409e42cfccSMilanka Ringwald } 104178d08d09SMilanka Ringwald 1042106b2cc4SMilanka Ringwald void avdtp_initialize_sbc_configuration_storage(avdtp_stream_endpoint_t * stream_endpoint, uint8_t * config_storage, uint16_t storage_size, uint8_t * packet, uint16_t packet_size){ 1043106b2cc4SMilanka Ringwald UNUSED(packet_size); 1044106b2cc4SMilanka Ringwald if (storage_size < 4) { 10458587e32cSMilanka Ringwald log_error("storage must have 4 bytes"); 1046747ec646SMilanka Ringwald return; 1047747ec646SMilanka Ringwald } 1048106b2cc4SMilanka Ringwald uint8_t sampling_frequency = avdtp_choose_sbc_sampling_frequency(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_sampling_frequency_bitmap(packet)); 1049106b2cc4SMilanka Ringwald uint8_t channel_mode = avdtp_choose_sbc_channel_mode(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_channel_mode_bitmap(packet)); 1050106b2cc4SMilanka Ringwald uint8_t block_length = avdtp_choose_sbc_block_length(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_block_length_bitmap(packet)); 1051106b2cc4SMilanka Ringwald uint8_t subbands = avdtp_choose_sbc_subbands(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_subbands_bitmap(packet)); 1052106b2cc4SMilanka Ringwald 1053106b2cc4SMilanka Ringwald uint8_t allocation_method = avdtp_choose_sbc_allocation_method(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_allocation_method_bitmap(packet)); 1054106b2cc4SMilanka Ringwald uint8_t max_bitpool_value = avdtp_choose_sbc_max_bitpool_value(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_max_bitpool_value(packet)); 1055106b2cc4SMilanka Ringwald uint8_t min_bitpool_value = avdtp_choose_sbc_min_bitpool_value(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_min_bitpool_value(packet)); 1056106b2cc4SMilanka Ringwald 105778d08d09SMilanka Ringwald config_storage[0] = (sampling_frequency << 4) | channel_mode; 105878d08d09SMilanka Ringwald config_storage[1] = (block_length << 4) | (subbands << 2) | allocation_method; 105978d08d09SMilanka Ringwald config_storage[2] = min_bitpool_value; 106078d08d09SMilanka Ringwald config_storage[3] = max_bitpool_value; 1061a0ed80bdSMilanka Ringwald 1062a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration_bitmap = store_bit16(stream_endpoint->remote_configuration_bitmap, AVDTP_MEDIA_CODEC, 1); 1063a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_type = AVDTP_AUDIO; 1064a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_codec_type = AVDTP_CODEC_SBC; 1065a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_codec_information_len = storage_size; 1066a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_codec_information = config_storage; 106778d08d09SMilanka Ringwald } 106878d08d09SMilanka Ringwald 106978d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_channel_mode(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_channel_mode_bitmap){ 107078d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 107178d08d09SMilanka Ringwald uint8_t channel_mode_bitmap = (media_codec[0] & 0x0F) & remote_channel_mode_bitmap; 107278d08d09SMilanka Ringwald 107378d08d09SMilanka Ringwald uint8_t channel_mode = AVDTP_SBC_STEREO; 107478d08d09SMilanka Ringwald if (channel_mode_bitmap & AVDTP_SBC_JOINT_STEREO){ 107578d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_JOINT_STEREO; 107678d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_STEREO){ 107778d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_STEREO; 107878d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_DUAL_CHANNEL){ 107978d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_DUAL_CHANNEL; 108078d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_MONO){ 108178d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_MONO; 108278d08d09SMilanka Ringwald } 108378d08d09SMilanka Ringwald return channel_mode; 108478d08d09SMilanka Ringwald } 108578d08d09SMilanka Ringwald 108678d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_allocation_method(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_allocation_method_bitmap){ 108778d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 108878d08d09SMilanka Ringwald uint8_t allocation_method_bitmap = (media_codec[1] & 0x03) & remote_allocation_method_bitmap; 108978d08d09SMilanka Ringwald 109078d08d09SMilanka Ringwald uint8_t allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 109178d08d09SMilanka Ringwald if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS){ 109278d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 109378d08d09SMilanka Ringwald } else if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_SNR){ 109478d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_SNR; 109578d08d09SMilanka Ringwald } 109678d08d09SMilanka Ringwald return allocation_method; 109778d08d09SMilanka Ringwald } 109878d08d09SMilanka Ringwald 1099bd1ecb8aSMilanka Ringwald uint8_t avdtp_stream_endpoint_seid(avdtp_stream_endpoint_t * stream_endpoint){ 1100bd1ecb8aSMilanka Ringwald if (!stream_endpoint) return 0; 1101bd1ecb8aSMilanka Ringwald return stream_endpoint->sep.seid; 1102bd1ecb8aSMilanka Ringwald } 110378d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_subbands(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_subbands_bitmap){ 110467ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 110578d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 110678d08d09SMilanka Ringwald uint8_t subbands_bitmap = ((media_codec[1] >> 2) & 0x03) & remote_subbands_bitmap; 110778d08d09SMilanka Ringwald 110878d08d09SMilanka Ringwald uint8_t subbands = AVDTP_SBC_SUBBANDS_8; 110978d08d09SMilanka Ringwald if (subbands_bitmap & AVDTP_SBC_SUBBANDS_8){ 111078d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_8; 111178d08d09SMilanka Ringwald } else if (subbands_bitmap & AVDTP_SBC_SUBBANDS_4){ 111278d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_4; 111378d08d09SMilanka Ringwald } 111478d08d09SMilanka Ringwald return subbands; 111578d08d09SMilanka Ringwald } 111678d08d09SMilanka Ringwald 111778d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_block_length(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_block_length_bitmap){ 111867ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 111978d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 112078d08d09SMilanka Ringwald uint8_t block_length_bitmap = (media_codec[1] >> 4) & remote_block_length_bitmap; 112178d08d09SMilanka Ringwald 112278d08d09SMilanka Ringwald uint8_t block_length = AVDTP_SBC_BLOCK_LENGTH_16; 112378d08d09SMilanka Ringwald if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_16){ 112478d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_16; 112578d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_12){ 112678d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_12; 112778d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_8){ 112878d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_8; 112978d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_4){ 113078d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_4; 113178d08d09SMilanka Ringwald } 113278d08d09SMilanka Ringwald return block_length; 113378d08d09SMilanka Ringwald } 113478d08d09SMilanka Ringwald 113578d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_sampling_frequency(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_sampling_frequency_bitmap){ 113667ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 113778d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 113878d08d09SMilanka Ringwald uint8_t sampling_frequency_bitmap = (media_codec[0] >> 4) & remote_sampling_frequency_bitmap; 113978d08d09SMilanka Ringwald 114078d08d09SMilanka Ringwald uint8_t sampling_frequency = AVDTP_SBC_44100; 114178d08d09SMilanka Ringwald if (sampling_frequency_bitmap & AVDTP_SBC_48000){ 114278d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_48000; 114378d08d09SMilanka Ringwald } else if (sampling_frequency_bitmap & AVDTP_SBC_44100){ 114478d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_44100; 114578d08d09SMilanka Ringwald } else if (sampling_frequency_bitmap & AVDTP_SBC_32000){ 114678d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_32000; 114778d08d09SMilanka Ringwald } else if (sampling_frequency_bitmap & AVDTP_SBC_16000){ 114878d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_16000; 114978d08d09SMilanka Ringwald } 115078d08d09SMilanka Ringwald return sampling_frequency; 115178d08d09SMilanka Ringwald } 115278d08d09SMilanka Ringwald 115378d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_max_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_max_bitpool_value){ 115467ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 115578d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 115678d08d09SMilanka Ringwald return btstack_min(media_codec[3], remote_max_bitpool_value); 115778d08d09SMilanka Ringwald } 115878d08d09SMilanka Ringwald 115978d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_min_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_min_bitpool_value){ 116067ae582dSMilanka Ringwald if (!stream_endpoint) return 0; 116178d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 116278d08d09SMilanka Ringwald return btstack_max(media_codec[2], remote_min_bitpool_value); 1163747ec646SMilanka Ringwald } 1164