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" 47747ec646SMilanka Ringwald #include "avdtp.h" 48747ec646SMilanka Ringwald #include "avdtp_util.h" 49747ec646SMilanka Ringwald #include "avdtp_acceptor.h" 50747ec646SMilanka Ringwald #include "avdtp_initiator.h" 51747ec646SMilanka Ringwald 52692c0605SMilanka Ringwald static int record_id = -1; 53692c0605SMilanka Ringwald static uint8_t attribute_value[1000]; 54692c0605SMilanka Ringwald static const unsigned int attribute_value_buffer_size = sizeof(attribute_value); 55747ec646SMilanka Ringwald 56692c0605SMilanka Ringwald typedef struct { 57692c0605SMilanka Ringwald avdtp_connection_t * connection; 58692c0605SMilanka Ringwald btstack_packet_handler_t avdtp_callback; 59692c0605SMilanka Ringwald avdtp_sep_type_t query_role; 60692c0605SMilanka Ringwald btstack_packet_handler_t packet_handler; 61692c0605SMilanka Ringwald } avdtp_sdp_query_context_t; 62692c0605SMilanka Ringwald 63692c0605SMilanka Ringwald static avdtp_sdp_query_context_t sdp_query_context; 644ccacc40SMilanka Ringwald static uint16_t avdtp_cid_counter = 0; 65747ec646SMilanka Ringwald 66747ec646SMilanka Ringwald static void (*handle_media_data)(avdtp_stream_endpoint_t * stream_endpoint, uint8_t *packet, uint16_t size); 67692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size); 68692c0605SMilanka Ringwald 69b0d75c91SMilanka Ringwald static uint16_t avdtp_get_next_initiator_transaction_label(avdtp_context_t * context){ 70b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter++; 71b0d75c91SMilanka Ringwald if (context->initiator_transaction_id_counter == 0){ 72b0d75c91SMilanka Ringwald context->initiator_transaction_id_counter = 1; 73b0d75c91SMilanka Ringwald } 74b0d75c91SMilanka Ringwald return context->initiator_transaction_id_counter; 75b0d75c91SMilanka Ringwald } 76b0d75c91SMilanka Ringwald 774ccacc40SMilanka Ringwald static uint16_t avdtp_get_next_avdtp_cid(){ 784ccacc40SMilanka Ringwald avdtp_cid_counter++; 794ccacc40SMilanka Ringwald if (avdtp_cid_counter == 0){ 804ccacc40SMilanka Ringwald avdtp_cid_counter = 1; 814ccacc40SMilanka Ringwald } 824ccacc40SMilanka Ringwald return avdtp_cid_counter; 834ccacc40SMilanka Ringwald } 844ccacc40SMilanka Ringwald 854ccacc40SMilanka Ringwald static uint16_t avdtp_get_next_local_seid(avdtp_context_t * context){ 864ccacc40SMilanka Ringwald context->stream_endpoints_id_counter++; 874ccacc40SMilanka Ringwald if (context->stream_endpoints_id_counter == 0){ 884ccacc40SMilanka Ringwald context->stream_endpoints_id_counter = 1; 894ccacc40SMilanka Ringwald } 904ccacc40SMilanka Ringwald return context->stream_endpoints_id_counter; 914ccacc40SMilanka Ringwald } 924ccacc40SMilanka Ringwald 934ccacc40SMilanka Ringwald uint8_t avdtp_connect(bd_addr_t remote, avdtp_sep_type_t query_role, avdtp_context_t * avdtp_context, uint16_t * avdtp_cid){ 94692c0605SMilanka Ringwald sdp_query_context.connection = NULL; 95692c0605SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_bd_addr(remote, avdtp_context); 96692c0605SMilanka Ringwald if (!connection){ 97692c0605SMilanka Ringwald connection = avdtp_create_connection(remote, avdtp_context); 98692c0605SMilanka Ringwald } 990bd7cb1fSMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_IDLE){ 1000bd7cb1fSMilanka Ringwald log_error("avdtp_connect: sink in wrong state,"); 1014ccacc40SMilanka Ringwald return BTSTACK_MEMORY_ALLOC_FAILED; 1020bd7cb1fSMilanka Ringwald } 1034ccacc40SMilanka Ringwald 1044ccacc40SMilanka Ringwald if (!avdtp_cid) return L2CAP_LOCAL_CID_DOES_NOT_EXIST; 1054ccacc40SMilanka Ringwald 106b0d75c91SMilanka Ringwald *avdtp_cid = connection->avdtp_cid; 107692c0605SMilanka Ringwald connection->state = AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE; 108692c0605SMilanka Ringwald sdp_query_context.connection = connection; 109692c0605SMilanka Ringwald sdp_query_context.query_role = query_role; 110692c0605SMilanka Ringwald sdp_query_context.avdtp_callback = avdtp_context->avdtp_callback; 111692c0605SMilanka Ringwald sdp_query_context.packet_handler = avdtp_context->packet_handler; 112692c0605SMilanka Ringwald 113692c0605SMilanka Ringwald sdp_client_query_uuid16(&avdtp_handle_sdp_client_query_result, remote, BLUETOOTH_PROTOCOL_AVDTP); 1144ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 115692c0605SMilanka Ringwald } 116747ec646SMilanka Ringwald 117747ec646SMilanka Ringwald void avdtp_register_media_transport_category(avdtp_stream_endpoint_t * stream_endpoint){ 118747ec646SMilanka Ringwald if (!stream_endpoint){ 119747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 120747ec646SMilanka Ringwald return; 121747ec646SMilanka Ringwald } 122747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MEDIA_TRANSPORT, 1); 123747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 124747ec646SMilanka Ringwald } 125747ec646SMilanka Ringwald 126747ec646SMilanka Ringwald void avdtp_register_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 127747ec646SMilanka Ringwald if (!stream_endpoint){ 128747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 129747ec646SMilanka Ringwald return; 130747ec646SMilanka Ringwald } 131747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_REPORTING, 1); 132747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 133747ec646SMilanka Ringwald } 134747ec646SMilanka Ringwald 135747ec646SMilanka Ringwald void avdtp_register_delay_reporting_category(avdtp_stream_endpoint_t * stream_endpoint){ 136747ec646SMilanka Ringwald if (!stream_endpoint){ 137747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 138747ec646SMilanka Ringwald return; 139747ec646SMilanka Ringwald } 140747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_DELAY_REPORTING, 1); 141747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 142747ec646SMilanka Ringwald } 143747ec646SMilanka Ringwald 144747ec646SMilanka Ringwald void avdtp_register_recovery_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t maximum_recovery_window_size, uint8_t maximum_number_media_packets){ 145747ec646SMilanka Ringwald if (!stream_endpoint){ 146747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 147747ec646SMilanka Ringwald return; 148747ec646SMilanka Ringwald } 149747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_RECOVERY, 1); 150747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 151747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.recovery_type = 0x01; // 0x01 = RFC2733 152747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_recovery_window_size = maximum_recovery_window_size; 153747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.recovery.maximum_number_media_packets = maximum_number_media_packets; 154747ec646SMilanka Ringwald } 155747ec646SMilanka Ringwald 156747ec646SMilanka 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){ 157747ec646SMilanka Ringwald if (!stream_endpoint){ 158747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 159747ec646SMilanka Ringwald return; 160747ec646SMilanka Ringwald } 161747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_CONTENT_PROTECTION, 1); 162747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 163747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type = cp_type; 164747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type_value = cp_type_value; 165747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.content_protection.cp_type_value_len = cp_type_value_len; 166747ec646SMilanka Ringwald } 167747ec646SMilanka Ringwald 168747ec646SMilanka Ringwald void avdtp_register_header_compression_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t back_ch, uint8_t media, uint8_t recovery){ 169747ec646SMilanka Ringwald if (!stream_endpoint){ 170747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 171747ec646SMilanka Ringwald return; 172747ec646SMilanka Ringwald } 173747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_HEADER_COMPRESSION, 1); 174747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 175747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.back_ch = back_ch; 176747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.media = media; 177747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.header_compression.recovery = recovery; 178747ec646SMilanka Ringwald } 179747ec646SMilanka Ringwald 18078d08d09SMilanka 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){ 181747ec646SMilanka Ringwald if (!stream_endpoint){ 182747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 183747ec646SMilanka Ringwald return; 184747ec646SMilanka Ringwald } 185747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MEDIA_CODEC, 1); 186747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 187747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_type = media_type; 188747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_type = media_codec_type; 189747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information = media_codec_info; 190747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.media_codec.media_codec_information_len = media_codec_info_len; 191747ec646SMilanka Ringwald } 192747ec646SMilanka Ringwald 193747ec646SMilanka Ringwald void avdtp_register_multiplexing_category(avdtp_stream_endpoint_t * stream_endpoint, uint8_t fragmentation){ 194747ec646SMilanka Ringwald if (!stream_endpoint){ 195747ec646SMilanka Ringwald log_error("avdtp_register_media_transport_category: stream endpoint with given seid is not registered"); 196747ec646SMilanka Ringwald return; 197747ec646SMilanka Ringwald } 198747ec646SMilanka Ringwald uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_MULTIPLEXING, 1); 199747ec646SMilanka Ringwald stream_endpoint->sep.registered_service_categories = bitmap; 200747ec646SMilanka Ringwald stream_endpoint->sep.capabilities.multiplexing_mode.fragmentation = fragmentation; 201747ec646SMilanka Ringwald } 202747ec646SMilanka Ringwald 203747ec646SMilanka Ringwald 204747ec646SMilanka Ringwald /* START: tracking can send now requests pro l2cap cid */ 205747ec646SMilanka Ringwald void avdtp_handle_can_send_now(avdtp_connection_t * connection, uint16_t l2cap_cid, avdtp_context_t * context){ 206747ec646SMilanka Ringwald if (connection->wait_to_send_acceptor){ 207747ec646SMilanka Ringwald connection->wait_to_send_acceptor = 0; 208747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm_run(connection, context); 209747ec646SMilanka Ringwald } else if (connection->wait_to_send_initiator){ 210747ec646SMilanka Ringwald connection->wait_to_send_initiator = 0; 211747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm_run(connection, context); 212747ec646SMilanka Ringwald } else if (connection->wait_to_send_self){ 213747ec646SMilanka Ringwald connection->wait_to_send_self = 0; 214747ec646SMilanka Ringwald if (connection->disconnect){ 215747ec646SMilanka Ringwald btstack_linked_list_iterator_t it; 216747ec646SMilanka Ringwald btstack_linked_list_iterator_init(&it, &context->stream_endpoints); 217747ec646SMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 218747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 219747ec646SMilanka Ringwald if (stream_endpoint->connection == connection){ 220747ec646SMilanka Ringwald if (stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_OPENED && stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED){ 221747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_DISCONNECTED; 2229413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 223747ec646SMilanka Ringwald l2cap_disconnect(stream_endpoint->l2cap_media_cid, 0); 224747ec646SMilanka Ringwald return; 225747ec646SMilanka Ringwald } 226747ec646SMilanka Ringwald } 227747ec646SMilanka Ringwald } 228747ec646SMilanka Ringwald connection->disconnect = 0; 229747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED; 2309413b167SMilanka Ringwald l2cap_disconnect(connection->l2cap_signaling_cid, 0); 231747ec646SMilanka Ringwald return; 232747ec646SMilanka Ringwald } 233747ec646SMilanka Ringwald } 234747ec646SMilanka Ringwald 235747ec646SMilanka Ringwald // re-register 236747ec646SMilanka Ringwald int more_to_send = connection->wait_to_send_acceptor || connection->wait_to_send_initiator || connection->wait_to_send_self; 237747ec646SMilanka Ringwald if (more_to_send){ 238747ec646SMilanka Ringwald l2cap_request_can_send_now_event(l2cap_cid); 239747ec646SMilanka Ringwald } 240747ec646SMilanka Ringwald } 241747ec646SMilanka Ringwald /* END: tracking can send now requests pro l2cap cid */ 242747ec646SMilanka Ringwald 243747ec646SMilanka Ringwald avdtp_connection_t * avdtp_create_connection(bd_addr_t remote_addr, avdtp_context_t * context){ 244747ec646SMilanka Ringwald avdtp_connection_t * connection = btstack_memory_avdtp_connection_get(); 245747ec646SMilanka Ringwald memset(connection, 0, sizeof(avdtp_connection_t)); 246747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 247b0d75c91SMilanka Ringwald connection->initiator_transaction_label = avdtp_get_next_initiator_transaction_label(context); 248b0d75c91SMilanka Ringwald connection->avdtp_cid = avdtp_get_next_avdtp_cid(); 249747ec646SMilanka Ringwald memcpy(connection->remote_addr, remote_addr, 6); 250747ec646SMilanka Ringwald btstack_linked_list_add(&context->connections, (btstack_linked_item_t *) connection); 251747ec646SMilanka Ringwald return connection; 252747ec646SMilanka Ringwald } 253747ec646SMilanka Ringwald 254747ec646SMilanka 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){ 255747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = btstack_memory_avdtp_stream_endpoint_get(); 256747ec646SMilanka Ringwald memset(stream_endpoint, 0, sizeof(avdtp_stream_endpoint_t)); 2574ccacc40SMilanka Ringwald stream_endpoint->sep.seid = avdtp_get_next_local_seid(context); 258747ec646SMilanka Ringwald stream_endpoint->sep.media_type = media_type; 259747ec646SMilanka Ringwald stream_endpoint->sep.type = sep_type; 260747ec646SMilanka Ringwald btstack_linked_list_add(&context->stream_endpoints, (btstack_linked_item_t *) stream_endpoint); 261747ec646SMilanka Ringwald return stream_endpoint; 262747ec646SMilanka Ringwald } 263747ec646SMilanka Ringwald 264747ec646SMilanka Ringwald 265747ec646SMilanka Ringwald static void handle_l2cap_data_packet_for_signaling_connection(avdtp_connection_t * connection, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 266747ec646SMilanka Ringwald int offset = avdtp_read_signaling_header(&connection->signaling_packet, packet, size); 267747ec646SMilanka Ringwald switch (connection->signaling_packet.message_type){ 268747ec646SMilanka Ringwald case AVDTP_CMD_MSG: 269747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm(connection, packet, size, offset, context); 270747ec646SMilanka Ringwald break; 271747ec646SMilanka Ringwald default: 272747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm(connection, packet, size, offset, context); 273747ec646SMilanka Ringwald break; 274747ec646SMilanka Ringwald } 275747ec646SMilanka Ringwald } 276747ec646SMilanka Ringwald 277747ec646SMilanka Ringwald static void stream_endpoint_state_machine(avdtp_connection_t * connection, avdtp_stream_endpoint_t * stream_endpoint, uint8_t packet_type, uint8_t event, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 278747ec646SMilanka Ringwald uint16_t local_cid; 2794ccacc40SMilanka Ringwald uint8_t status; 280747ec646SMilanka Ringwald switch (packet_type){ 281747ec646SMilanka Ringwald case L2CAP_DATA_PACKET:{ 282747ec646SMilanka Ringwald int offset = avdtp_read_signaling_header(&connection->signaling_packet, packet, size); 283747ec646SMilanka Ringwald if (connection->signaling_packet.message_type == AVDTP_CMD_MSG){ 284747ec646SMilanka Ringwald avdtp_acceptor_stream_config_subsm(connection, packet, size, offset, context); 285747ec646SMilanka Ringwald } else { 286747ec646SMilanka Ringwald avdtp_initiator_stream_config_subsm(connection, packet, size, offset, context); 287747ec646SMilanka Ringwald } 288747ec646SMilanka Ringwald break; 289747ec646SMilanka Ringwald } 290747ec646SMilanka Ringwald case HCI_EVENT_PACKET: 291747ec646SMilanka Ringwald switch (event){ 292747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_OPENED: 293747ec646SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 2944ccacc40SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED){ 29534b22aacSMilanka 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)); 2964ccacc40SMilanka Ringwald avdtp_streaming_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint), AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE); 2974ccacc40SMilanka Ringwald break; 2984ccacc40SMilanka Ringwald } 2994ccacc40SMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet); 3004ccacc40SMilanka Ringwald if (status){ 30134b22aacSMilanka 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)); 3024ccacc40SMilanka Ringwald avdtp_streaming_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint), status); 3034ccacc40SMilanka Ringwald break; 3044ccacc40SMilanka Ringwald } 305747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED; 306747ec646SMilanka Ringwald stream_endpoint->connection = connection; 307747ec646SMilanka Ringwald stream_endpoint->l2cap_media_cid = l2cap_event_channel_opened_get_local_cid(packet); 30857f317a1SMilanka Ringwald stream_endpoint->media_con_handle = l2cap_event_channel_opened_get_handle(packet); 3094ccacc40SMilanka Ringwald 3106b0ee1d0SMilanka 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)); 3114ccacc40SMilanka Ringwald avdtp_streaming_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint), avdtp_remote_seid(stream_endpoint), 0); 312747ec646SMilanka Ringwald break; 313747ec646SMilanka Ringwald } 314747ec646SMilanka Ringwald break; 315747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_CLOSED: 316747ec646SMilanka Ringwald local_cid = l2cap_event_channel_closed_get_local_cid(packet); 317747ec646SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == local_cid){ 318b0d75c91SMilanka Ringwald avdtp_streaming_emit_connection_released(context->avdtp_callback, stream_endpoint->connection->avdtp_cid, avdtp_local_seid(stream_endpoint)); 319747ec646SMilanka Ringwald stream_endpoint->l2cap_media_cid = 0; 320747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_IDLE; 321747ec646SMilanka Ringwald stream_endpoint->acceptor_config_state = AVDTP_ACCEPTOR_STREAM_CONFIG_IDLE; 322747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_STREAM_CONFIG_IDLE; 323747ec646SMilanka Ringwald stream_endpoint->remote_sep_index = 0; 324747ec646SMilanka Ringwald break; 325747ec646SMilanka Ringwald } 326747ec646SMilanka Ringwald if (stream_endpoint->l2cap_recovery_cid == local_cid){ 327747ec646SMilanka Ringwald log_info(" -> L2CAP_EVENT_CHANNEL_CLOSED recovery cid 0x%0x", local_cid); 328747ec646SMilanka Ringwald stream_endpoint->l2cap_recovery_cid = 0; 329747ec646SMilanka Ringwald break; 330747ec646SMilanka Ringwald } 331747ec646SMilanka Ringwald 332747ec646SMilanka Ringwald if (stream_endpoint->l2cap_reporting_cid == local_cid){ 333747ec646SMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_CLOSED reporting cid 0x%0x", local_cid); 334747ec646SMilanka Ringwald stream_endpoint->l2cap_reporting_cid = 0; 335747ec646SMilanka Ringwald break; 336747ec646SMilanka Ringwald } 337747ec646SMilanka Ringwald break; 338747ec646SMilanka Ringwald default: 339747ec646SMilanka Ringwald break; 340747ec646SMilanka Ringwald } 341747ec646SMilanka Ringwald break; 342747ec646SMilanka Ringwald default: 343747ec646SMilanka Ringwald break; 344747ec646SMilanka Ringwald } 345747ec646SMilanka Ringwald } 346747ec646SMilanka Ringwald 347692c0605SMilanka Ringwald static void avdtp_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 348692c0605SMilanka Ringwald UNUSED(packet_type); 349692c0605SMilanka Ringwald UNUSED(channel); 350692c0605SMilanka Ringwald UNUSED(size); 351692c0605SMilanka Ringwald 352692c0605SMilanka Ringwald des_iterator_t des_list_it; 353692c0605SMilanka Ringwald des_iterator_t prot_it; 354692c0605SMilanka Ringwald uint16_t avdtp_l2cap_psm = 0; 355692c0605SMilanka Ringwald uint16_t avdtp_version = 0; 356eddf49b7SMatthias Ringwald // uint32_t avdtp_remote_uuid = 0; 357692c0605SMilanka Ringwald 358692c0605SMilanka Ringwald if (!sdp_query_context.connection) return; 359692c0605SMilanka Ringwald 360692c0605SMilanka Ringwald switch (hci_event_packet_get_type(packet)){ 361692c0605SMilanka Ringwald case SDP_EVENT_QUERY_ATTRIBUTE_VALUE: 362692c0605SMilanka Ringwald // Handle new SDP record 363692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_record_id(packet) != record_id) { 364692c0605SMilanka Ringwald record_id = sdp_event_query_attribute_byte_get_record_id(packet); 3658587e32cSMilanka Ringwald // log_info("SDP Record: Nr: %d", record_id); 366692c0605SMilanka Ringwald } 367692c0605SMilanka Ringwald 368692c0605SMilanka Ringwald if (sdp_event_query_attribute_byte_get_attribute_length(packet) <= attribute_value_buffer_size) { 369692c0605SMilanka Ringwald attribute_value[sdp_event_query_attribute_byte_get_data_offset(packet)] = sdp_event_query_attribute_byte_get_data(packet); 370692c0605SMilanka Ringwald 371692c0605SMilanka Ringwald if ((uint16_t)(sdp_event_query_attribute_byte_get_data_offset(packet)+1) == sdp_event_query_attribute_byte_get_attribute_length(packet)) { 372692c0605SMilanka Ringwald 373692c0605SMilanka Ringwald switch(sdp_event_query_attribute_byte_get_attribute_id(packet)) { 374692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_SERVICE_CLASS_ID_LIST: 375692c0605SMilanka Ringwald if (de_get_element_type(attribute_value) != DE_DES) break; 376692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 377692c0605SMilanka Ringwald uint8_t * element = des_iterator_get_element(&des_list_it); 378692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 379692c0605SMilanka Ringwald uint32_t uuid = de_get_uuid32(element); 380692c0605SMilanka Ringwald switch (uuid){ 381692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SOURCE: 382692c0605SMilanka Ringwald if (sdp_query_context.query_role != AVDTP_SOURCE) { 383692c0605SMilanka Ringwald sdp_query_context.connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 3844ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context.avdtp_callback, sdp_query_context.connection->avdtp_cid, sdp_query_context.connection->remote_addr, SDP_SERVICE_NOT_FOUND); 385692c0605SMilanka Ringwald break; 386692c0605SMilanka Ringwald } 3878587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SOURCE protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 388eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 389692c0605SMilanka Ringwald break; 390692c0605SMilanka Ringwald case BLUETOOTH_SERVICE_CLASS_AUDIO_SINK: 391692c0605SMilanka Ringwald if (sdp_query_context.query_role != AVDTP_SINK) { 392692c0605SMilanka Ringwald sdp_query_context.connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 3934ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context.avdtp_callback, sdp_query_context.connection->avdtp_cid, sdp_query_context.connection->remote_addr, SDP_SERVICE_NOT_FOUND); 394692c0605SMilanka Ringwald break; 395692c0605SMilanka Ringwald } 3968587e32cSMilanka Ringwald // log_info("SDP Attribute 0x%04x: AVDTP SINK protocol UUID: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet), uuid); 397eddf49b7SMatthias Ringwald // avdtp_remote_uuid = uuid; 398692c0605SMilanka Ringwald break; 399692c0605SMilanka Ringwald default: 400692c0605SMilanka Ringwald break; 401692c0605SMilanka Ringwald } 402692c0605SMilanka Ringwald } 403692c0605SMilanka Ringwald break; 404692c0605SMilanka Ringwald 405692c0605SMilanka Ringwald case BLUETOOTH_ATTRIBUTE_PROTOCOL_DESCRIPTOR_LIST: { 4068587e32cSMilanka Ringwald // log_info("SDP Attribute: 0x%04x", sdp_event_query_attribute_byte_get_attribute_id(packet)); 407692c0605SMilanka Ringwald 408692c0605SMilanka Ringwald for (des_iterator_init(&des_list_it, attribute_value); des_iterator_has_more(&des_list_it); des_iterator_next(&des_list_it)) { 409692c0605SMilanka Ringwald uint8_t *des_element; 410692c0605SMilanka Ringwald uint8_t *element; 411692c0605SMilanka Ringwald uint32_t uuid; 412692c0605SMilanka Ringwald 413692c0605SMilanka Ringwald if (des_iterator_get_type(&des_list_it) != DE_DES) continue; 414692c0605SMilanka Ringwald 415692c0605SMilanka Ringwald des_element = des_iterator_get_element(&des_list_it); 416692c0605SMilanka Ringwald des_iterator_init(&prot_it, des_element); 417692c0605SMilanka Ringwald element = des_iterator_get_element(&prot_it); 418692c0605SMilanka Ringwald 419692c0605SMilanka Ringwald if (de_get_element_type(element) != DE_UUID) continue; 420692c0605SMilanka Ringwald 421692c0605SMilanka Ringwald uuid = de_get_uuid32(element); 422692c0605SMilanka Ringwald switch (uuid){ 423692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_L2CAP: 424692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 425692c0605SMilanka Ringwald des_iterator_next(&prot_it); 426692c0605SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &avdtp_l2cap_psm); 427692c0605SMilanka Ringwald break; 428692c0605SMilanka Ringwald case BLUETOOTH_PROTOCOL_AVDTP: 429692c0605SMilanka Ringwald if (!des_iterator_has_more(&prot_it)) continue; 430692c0605SMilanka Ringwald des_iterator_next(&prot_it); 431692c0605SMilanka Ringwald de_element_get_uint16(des_iterator_get_element(&prot_it), &avdtp_version); 432692c0605SMilanka Ringwald break; 433692c0605SMilanka Ringwald default: 434692c0605SMilanka Ringwald break; 435692c0605SMilanka Ringwald } 436692c0605SMilanka Ringwald } 4378587e32cSMilanka Ringwald if (!avdtp_l2cap_psm) { 4388587e32cSMilanka Ringwald sdp_query_context.connection->state = AVDTP_SIGNALING_CONNECTION_IDLE; 4394ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(sdp_query_context.avdtp_callback, sdp_query_context.connection->avdtp_cid, sdp_query_context.connection->remote_addr, L2CAP_SERVICE_DOES_NOT_EXIST); 4408587e32cSMilanka Ringwald break; 4418587e32cSMilanka Ringwald } 442692c0605SMilanka Ringwald sdp_query_context.connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 443692c0605SMilanka Ringwald l2cap_create_channel(sdp_query_context.packet_handler, sdp_query_context.connection->remote_addr, avdtp_l2cap_psm, l2cap_max_mtu(), NULL); 444692c0605SMilanka Ringwald } 445692c0605SMilanka Ringwald break; 446692c0605SMilanka Ringwald default: 447692c0605SMilanka Ringwald break; 448692c0605SMilanka Ringwald } 449692c0605SMilanka Ringwald } 450692c0605SMilanka Ringwald } else { 4518587e32cSMilanka 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)); 452692c0605SMilanka Ringwald } 453692c0605SMilanka Ringwald break; 454692c0605SMilanka Ringwald 455692c0605SMilanka Ringwald case SDP_EVENT_QUERY_COMPLETE: 4568587e32cSMilanka Ringwald log_info("General query done with status %d.", sdp_event_query_complete_get_status(packet)); 457692c0605SMilanka Ringwald break; 458692c0605SMilanka Ringwald } 459692c0605SMilanka Ringwald } 460692c0605SMilanka Ringwald 461692c0605SMilanka Ringwald 462747ec646SMilanka Ringwald void avdtp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size, avdtp_context_t * context){ 463747ec646SMilanka Ringwald bd_addr_t event_addr; 464747ec646SMilanka Ringwald hci_con_handle_t con_handle; 465747ec646SMilanka Ringwald uint16_t psm; 466747ec646SMilanka Ringwald uint16_t local_cid; 467747ec646SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = NULL; 468747ec646SMilanka Ringwald avdtp_connection_t * connection = NULL; 469747ec646SMilanka Ringwald btstack_linked_list_t * avdtp_connections = &context->connections; 470747ec646SMilanka Ringwald btstack_linked_list_t * stream_endpoints = &context->stream_endpoints; 471747ec646SMilanka Ringwald handle_media_data = context->handle_media_data; 4728587e32cSMilanka Ringwald // log_info("avdtp_packet_handler packet type %02x, event %02x ", packet_type, hci_event_packet_get_type(packet)); 473747ec646SMilanka Ringwald switch (packet_type) { 474747ec646SMilanka Ringwald case L2CAP_DATA_PACKET: 4759413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 476747ec646SMilanka Ringwald if (connection){ 477747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 478747ec646SMilanka Ringwald break; 479747ec646SMilanka Ringwald } 480747ec646SMilanka Ringwald 481747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 482747ec646SMilanka Ringwald if (!stream_endpoint){ 483747ec646SMilanka Ringwald if (!connection) break; 484747ec646SMilanka Ringwald handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context); 485747ec646SMilanka Ringwald break; 486747ec646SMilanka Ringwald } 487747ec646SMilanka Ringwald 4889413b167SMilanka Ringwald if (channel == stream_endpoint->connection->l2cap_signaling_cid){ 489747ec646SMilanka Ringwald stream_endpoint_state_machine(stream_endpoint->connection, stream_endpoint, L2CAP_DATA_PACKET, 0, packet, size, context); 490747ec646SMilanka Ringwald break; 491747ec646SMilanka Ringwald } 492747ec646SMilanka Ringwald 493747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_media_cid){ 4948b097e29SMilanka Ringwald if (handle_media_data){ 495747ec646SMilanka Ringwald (*handle_media_data)(stream_endpoint, packet, size); 4968b097e29SMilanka Ringwald } 497747ec646SMilanka Ringwald break; 498747ec646SMilanka Ringwald } 499747ec646SMilanka Ringwald 500747ec646SMilanka Ringwald if (channel == stream_endpoint->l2cap_reporting_cid){ 501747ec646SMilanka Ringwald // TODO 5028587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for reporting: NOT IMPLEMENTED"); 503747ec646SMilanka Ringwald } else if (channel == stream_endpoint->l2cap_recovery_cid){ 504747ec646SMilanka Ringwald // TODO 5058587e32cSMilanka Ringwald log_info("L2CAP_DATA_PACKET for recovery: NOT IMPLEMENTED"); 506747ec646SMilanka Ringwald } else { 507747ec646SMilanka Ringwald log_error("avdtp packet handler L2CAP_DATA_PACKET: local cid 0x%02x not found", channel); 508747ec646SMilanka Ringwald } 509747ec646SMilanka Ringwald break; 510747ec646SMilanka Ringwald 511747ec646SMilanka Ringwald case HCI_EVENT_PACKET: 512747ec646SMilanka Ringwald switch (hci_event_packet_get_type(packet)) { 513747ec646SMilanka Ringwald case L2CAP_EVENT_INCOMING_CONNECTION: 514747ec646SMilanka Ringwald l2cap_event_incoming_connection_get_address(packet, event_addr); 515747ec646SMilanka Ringwald local_cid = l2cap_event_incoming_connection_get_local_cid(packet); 516747ec646SMilanka Ringwald 517747ec646SMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 51849dda273SMilanka Ringwald 51949dda273SMilanka Ringwald if (!connection || connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED){ 520747ec646SMilanka Ringwald connection = avdtp_create_connection(event_addr, context); 521747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED; 5228587e32cSMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION, connection %p, state connection %d", connection, connection->state); 523747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 524747ec646SMilanka Ringwald break; 525747ec646SMilanka Ringwald } 526747ec646SMilanka Ringwald 5276b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 528747ec646SMilanka Ringwald if (!stream_endpoint) { 5296b0ee1d0SMilanka Ringwald log_info("L2CAP_EVENT_INCOMING_CONNECTION no streamendpoint found for seid %d", connection->local_seid); 530747ec646SMilanka Ringwald break; 531747ec646SMilanka Ringwald } 532747ec646SMilanka Ringwald 533747ec646SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 534747ec646SMilanka Ringwald if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED) break; 535747ec646SMilanka Ringwald l2cap_accept_connection(local_cid); 536747ec646SMilanka Ringwald break; 537747ec646SMilanka Ringwald } 538747ec646SMilanka Ringwald break; 539747ec646SMilanka Ringwald 540747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_OPENED: 541747ec646SMilanka Ringwald // inform about new l2cap connection 542747ec646SMilanka Ringwald l2cap_event_channel_opened_get_address(packet, event_addr); 5437050d2caSMilanka Ringwald local_cid = l2cap_event_channel_opened_get_local_cid(packet); 544747ec646SMilanka Ringwald if (l2cap_event_channel_opened_get_status(packet)){ 5454ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, l2cap_event_channel_opened_get_status(packet)); 546747ec646SMilanka Ringwald log_error("L2CAP connection to connection %s failed. status code 0x%02x", 547747ec646SMilanka Ringwald bd_addr_to_str(event_addr), l2cap_event_channel_opened_get_status(packet)); 548747ec646SMilanka Ringwald break; 549747ec646SMilanka Ringwald } 550747ec646SMilanka Ringwald psm = l2cap_event_channel_opened_get_psm(packet); 551235946f1SMatthias Ringwald if (psm != BLUETOOTH_PROTOCOL_AVDTP){ 552747ec646SMilanka Ringwald log_error("unexpected PSM - Not implemented yet, avdtp sink: L2CAP_EVENT_CHANNEL_OPENED"); 553747ec646SMilanka Ringwald return; 554747ec646SMilanka Ringwald } 555747ec646SMilanka Ringwald 556747ec646SMilanka Ringwald con_handle = l2cap_event_channel_opened_get_handle(packet); 557747ec646SMilanka Ringwald 5586b0ee1d0SMilanka Ringwald // log_info("L2CAP_EVENT_CHANNEL_OPENED: Channel successfully opened: %s, handle 0x%02x, psm 0x%02x, local cid 0x%02x, remote cid 0x%02x", 5596b0ee1d0SMilanka Ringwald // bd_addr_to_str(event_addr), con_handle, psm, local_cid, l2cap_event_channel_opened_get_remote_cid(packet)); 560747ec646SMilanka Ringwald 561235946f1SMatthias Ringwald if (psm != BLUETOOTH_PROTOCOL_AVDTP) break; 562747ec646SMilanka Ringwald 563747ec646SMilanka Ringwald connection = avdtp_connection_for_bd_addr(event_addr, context); 564747ec646SMilanka Ringwald if (!connection) break; 565747ec646SMilanka Ringwald 5669413b167SMilanka Ringwald if (connection->l2cap_signaling_cid == 0) { 567747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED) break; 5689413b167SMilanka Ringwald connection->l2cap_signaling_cid = local_cid; 5696b0ee1d0SMilanka Ringwald connection->local_seid = 0; 570747ec646SMilanka Ringwald connection->state = AVDTP_SIGNALING_CONNECTION_OPENED; 5716b0ee1d0SMilanka Ringwald log_info(" -> AVDTP_SIGNALING_CONNECTION_OPENED, connection %p, avdtp_cid 0x%02x", connection, connection->avdtp_cid); 5724ccacc40SMilanka Ringwald avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, 0); 573747ec646SMilanka Ringwald break; 574747ec646SMilanka Ringwald } 575747ec646SMilanka Ringwald 5766b0ee1d0SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context); 577747ec646SMilanka Ringwald if (!stream_endpoint){ 5788587e32cSMilanka Ringwald log_info("L2CAP_EVENT_CHANNEL_OPENED: stream_endpoint not found"); 579747ec646SMilanka Ringwald return; 580747ec646SMilanka Ringwald } 581747ec646SMilanka Ringwald stream_endpoint_state_machine(connection, stream_endpoint, HCI_EVENT_PACKET, L2CAP_EVENT_CHANNEL_OPENED, packet, size, context); 582747ec646SMilanka Ringwald break; 583747ec646SMilanka Ringwald 584747ec646SMilanka Ringwald case L2CAP_EVENT_CHANNEL_CLOSED: 585747ec646SMilanka Ringwald local_cid = l2cap_event_channel_closed_get_local_cid(packet); 5869413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(local_cid, context); 5870bd7cb1fSMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(local_cid, context); 58834b22aacSMilanka Ringwald 5890bd7cb1fSMilanka Ringwald if (stream_endpoint){ 5900bd7cb1fSMilanka Ringwald stream_endpoint_state_machine(connection, stream_endpoint, HCI_EVENT_PACKET, L2CAP_EVENT_CHANNEL_CLOSED, packet, size, context); 5910bd7cb1fSMilanka Ringwald break; 5920bd7cb1fSMilanka Ringwald } 5930bd7cb1fSMilanka Ringwald 594747ec646SMilanka Ringwald if (connection){ 59534b22aacSMilanka Ringwald avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid); 596747ec646SMilanka Ringwald btstack_linked_list_remove(avdtp_connections, (btstack_linked_item_t*) connection); 597747ec646SMilanka Ringwald btstack_linked_list_iterator_t it; 598747ec646SMilanka Ringwald btstack_linked_list_iterator_init(&it, stream_endpoints); 599747ec646SMilanka Ringwald while (btstack_linked_list_iterator_has_next(&it)){ 600747ec646SMilanka Ringwald avdtp_stream_endpoint_t * _stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it); 601ac98fce4SMilanka Ringwald 602ac98fce4SMilanka Ringwald if (_stream_endpoint->connection == connection){ 603ac98fce4SMilanka Ringwald avdtp_initialize_stream_endpoint(_stream_endpoint); 604ac98fce4SMilanka Ringwald } 605747ec646SMilanka Ringwald } 606671c66c1SMatthias Ringwald btstack_memory_avdtp_connection_free(connection); 607747ec646SMilanka Ringwald break; 608747ec646SMilanka Ringwald } 609747ec646SMilanka Ringwald break; 610747ec646SMilanka Ringwald 611747ec646SMilanka Ringwald case HCI_EVENT_DISCONNECTION_COMPLETE: 612747ec646SMilanka Ringwald break; 613747ec646SMilanka Ringwald 614747ec646SMilanka Ringwald case L2CAP_EVENT_CAN_SEND_NOW: 6159413b167SMilanka Ringwald connection = avdtp_connection_for_l2cap_signaling_cid(channel, context); 616747ec646SMilanka Ringwald if (!connection) { 617747ec646SMilanka Ringwald stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context); 618747ec646SMilanka Ringwald if (!stream_endpoint->connection) break; 619747ec646SMilanka Ringwald connection = stream_endpoint->connection; 620747ec646SMilanka Ringwald } 621747ec646SMilanka Ringwald avdtp_handle_can_send_now(connection, channel, context); 622747ec646SMilanka Ringwald break; 623747ec646SMilanka Ringwald default: 6248587e32cSMilanka Ringwald log_info("unknown HCI event type %02x", hci_event_packet_get_type(packet)); 625747ec646SMilanka Ringwald break; 626747ec646SMilanka Ringwald } 627747ec646SMilanka Ringwald break; 628747ec646SMilanka Ringwald 629747ec646SMilanka Ringwald default: 630747ec646SMilanka Ringwald // other packet type 631747ec646SMilanka Ringwald break; 632747ec646SMilanka Ringwald } 633747ec646SMilanka Ringwald } 634747ec646SMilanka Ringwald 6354ccacc40SMilanka Ringwald uint8_t avdtp_disconnect(uint16_t avdtp_cid, avdtp_context_t * context){ 6364ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 6374ccacc40SMilanka Ringwald if (!connection) return BTSTACK_MEMORY_ALLOC_FAILED; 6384ccacc40SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_IDLE) return AVDTP_CONNECTION_IN_WRONG_STATE; 6394ccacc40SMilanka Ringwald if (connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED) return AVDTP_CONNECTION_IN_WRONG_STATE; 640747ec646SMilanka Ringwald 641747ec646SMilanka Ringwald connection->disconnect = 1; 6429413b167SMilanka Ringwald avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid); 6434ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 644747ec646SMilanka Ringwald } 645747ec646SMilanka Ringwald 6466b0ee1d0SMilanka Ringwald uint8_t avdtp_open_stream(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, avdtp_context_t * context){ 6474ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 648747ec646SMilanka Ringwald if (!connection){ 6498587e32cSMilanka Ringwald log_error("avdtp_media_connect: no connection for signaling cid 0x%02x found", avdtp_cid); 6504ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 651747ec646SMilanka Ringwald } 6526b0ee1d0SMilanka Ringwald if (avdtp_find_remote_sep(connection, remote_seid) == 0xFF){ 6536b0ee1d0SMilanka Ringwald log_error("avdtp_media_connect: no remote sep for seid %d found", remote_seid); 6544ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 655747ec646SMilanka Ringwald } 656747ec646SMilanka Ringwald 657747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) { 6588587e32cSMilanka Ringwald log_error("avdtp_media_connect: wrong connection state %d", connection->state); 6594ccacc40SMilanka Ringwald return AVDTP_CONNECTION_IN_WRONG_STATE; 660747ec646SMilanka Ringwald } 661747ec646SMilanka Ringwald 6626b0ee1d0SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 663747ec646SMilanka Ringwald if (!stream_endpoint) { 6646b0ee1d0SMilanka Ringwald log_error("avdtp_media_connect: no stream_endpoint with seid %d found", local_seid); 6654ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 666747ec646SMilanka Ringwald } 667747ec646SMilanka Ringwald 6684ccacc40SMilanka Ringwald if (stream_endpoint->state < AVDTP_STREAM_ENDPOINT_CONFIGURED) return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 6694ccacc40SMilanka Ringwald if (stream_endpoint->remote_sep_index == AVDTP_INVALID_SEP_INDEX) return AVDTP_SEID_DOES_NOT_EXIST; 670747ec646SMilanka Ringwald 671747ec646SMilanka Ringwald connection->initiator_transaction_label++; 6729413b167SMilanka Ringwald connection->remote_seid = remote_seid; 6736b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 674747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_OPEN_STREAM; 675747ec646SMilanka Ringwald stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W2_REQUEST_OPEN_STREAM; 6769413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 6774ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 678747ec646SMilanka Ringwald } 679747ec646SMilanka Ringwald 6804ccacc40SMilanka Ringwald uint8_t avdtp_start_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 6814ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 68246e6b063SMilanka Ringwald if (!connection){ 6834ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 6844ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 68546e6b063SMilanka Ringwald } 6865cfe7f4cSMilanka Ringwald 6874ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 6884ccacc40SMilanka Ringwald if (!stream_endpoint) { 6894ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no stream_endpoint with seid %d found", local_seid); 6904ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 6914ccacc40SMilanka Ringwald } 6924ccacc40SMilanka Ringwald 6934ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 6944ccacc40SMilanka Ringwald log_error("avdtp_start_stream: no media connection for stream_endpoint with seid %d found", local_seid); 6954ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 6964ccacc40SMilanka Ringwald } 6974ccacc40SMilanka Ringwald 6984ccacc40SMilanka Ringwald if (stream_endpoint->remote_sep_index == AVDTP_INVALID_SEP_INDEX || stream_endpoint->start_stream){ 6994ccacc40SMilanka Ringwald return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE; 7004ccacc40SMilanka Ringwald } 7014ccacc40SMilanka Ringwald 70260ec20d0SMilanka Ringwald stream_endpoint->start_stream = 1; 7034ccacc40SMilanka Ringwald connection->local_seid = local_seid; 7049413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 7054ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 706747ec646SMilanka Ringwald } 707747ec646SMilanka Ringwald 7084ccacc40SMilanka Ringwald uint8_t avdtp_stop_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 7094ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 710747ec646SMilanka Ringwald if (!connection){ 7114ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 7124ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 713747ec646SMilanka Ringwald } 7144ccacc40SMilanka Ringwald 7154ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 7164ccacc40SMilanka Ringwald if (!stream_endpoint) { 7174ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no stream_endpoint with seid %d found", local_seid); 7184ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 7194ccacc40SMilanka Ringwald } 7204ccacc40SMilanka Ringwald 7214ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 7224ccacc40SMilanka Ringwald log_error("avdtp_stop_stream: no media connection for stream_endpoint with seid %d found", local_seid); 7234ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 7244ccacc40SMilanka Ringwald } 7254ccacc40SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF || stream_endpoint->stop_stream) return ERROR_CODE_SUCCESS; 7264ccacc40SMilanka Ringwald 72760ec20d0SMilanka Ringwald stream_endpoint->stop_stream = 1; 7284ccacc40SMilanka Ringwald connection->local_seid = local_seid; 7299413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 7304ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 731747ec646SMilanka Ringwald } 732747ec646SMilanka Ringwald 7334ccacc40SMilanka Ringwald uint8_t avdtp_abort_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 7344ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 73560ec20d0SMilanka Ringwald if (!connection){ 7364ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 7374ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 738747ec646SMilanka Ringwald } 7394ccacc40SMilanka Ringwald 7404ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 7414ccacc40SMilanka Ringwald if (!stream_endpoint) { 7424ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no stream_endpoint with seid %d found", local_seid); 7434ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 7444ccacc40SMilanka Ringwald } 7454ccacc40SMilanka Ringwald 7464ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 7474ccacc40SMilanka Ringwald log_error("avdtp_abort_stream: no media connection for stream_endpoint with seid %d found", local_seid); 7484ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 7494ccacc40SMilanka Ringwald } 7506b0ee1d0SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF || stream_endpoint->abort_stream) return ERROR_CODE_SUCCESS; 7514ccacc40SMilanka Ringwald 75260ec20d0SMilanka Ringwald stream_endpoint->abort_stream = 1; 7536b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 7549413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 7554ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 756747ec646SMilanka Ringwald } 757747ec646SMilanka Ringwald 7584ccacc40SMilanka Ringwald uint8_t avdtp_suspend_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){ 7594ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 760747ec646SMilanka Ringwald if (!connection){ 7614ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no connection for signaling cid 0x%02x found", avdtp_cid); 7624ccacc40SMilanka Ringwald return AVDTP_CONNECTION_DOES_NOT_EXIST; 76360ec20d0SMilanka Ringwald } 7644ccacc40SMilanka Ringwald 7654ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context); 7664ccacc40SMilanka Ringwald if (!stream_endpoint) { 7674ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no stream_endpoint with seid %d found", local_seid); 7684ccacc40SMilanka Ringwald return AVDTP_SEID_DOES_NOT_EXIST; 7694ccacc40SMilanka Ringwald } 7704ccacc40SMilanka Ringwald 7714ccacc40SMilanka Ringwald if (stream_endpoint->l2cap_media_cid == 0){ 7724ccacc40SMilanka Ringwald log_error("avdtp_suspend_stream: no media connection for stream_endpoint with seid %d found", local_seid); 7734ccacc40SMilanka Ringwald return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST; 7744ccacc40SMilanka Ringwald } 7756b0ee1d0SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF || stream_endpoint->suspend_stream) return ERROR_CODE_SUCCESS; 7764ccacc40SMilanka Ringwald 77760ec20d0SMilanka Ringwald stream_endpoint->suspend_stream = 1; 7784ccacc40SMilanka Ringwald connection->local_seid = local_seid; 7799413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 7804ccacc40SMilanka Ringwald return ERROR_CODE_SUCCESS; 781747ec646SMilanka Ringwald } 782747ec646SMilanka Ringwald 783f9bca1f3SMilanka Ringwald void avdtp_discover_stream_endpoints(uint16_t avdtp_cid, avdtp_context_t * context){ 7844ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 785747ec646SMilanka Ringwald if (!connection){ 7868587e32cSMilanka Ringwald log_error("avdtp_discover_stream_endpoints: no connection for signaling cid 0x%02x found", avdtp_cid); 787747ec646SMilanka Ringwald return; 788747ec646SMilanka Ringwald } 789747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) return; 790747ec646SMilanka Ringwald 791747ec646SMilanka Ringwald switch (connection->initiator_connection_state){ 792747ec646SMilanka Ringwald case AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE: 793747ec646SMilanka Ringwald connection->initiator_transaction_label++; 794747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_DISCOVER_SEPS; 7959413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 796747ec646SMilanka Ringwald break; 797747ec646SMilanka Ringwald default: 7988587e32cSMilanka Ringwald log_error("avdtp_discover_stream_endpoints: wrong state"); 799747ec646SMilanka Ringwald break; 800747ec646SMilanka Ringwald } 801747ec646SMilanka Ringwald } 802747ec646SMilanka Ringwald 803747ec646SMilanka Ringwald 8046b0ee1d0SMilanka Ringwald void avdtp_get_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 8054ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 806747ec646SMilanka Ringwald if (!connection){ 8074ccacc40SMilanka Ringwald log_error("avdtp_get_capabilities: no connection for AVDTP cid 0x%02x found", avdtp_cid); 808747ec646SMilanka Ringwald return; 809747ec646SMilanka Ringwald } 810747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) return; 811747ec646SMilanka Ringwald if (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) return; 812747ec646SMilanka Ringwald connection->initiator_transaction_label++; 813747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CAPABILITIES; 8149413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8159413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 816747ec646SMilanka Ringwald } 817747ec646SMilanka Ringwald 818747ec646SMilanka Ringwald 8196b0ee1d0SMilanka Ringwald void avdtp_get_all_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 8204ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 821747ec646SMilanka Ringwald if (!connection){ 8224ccacc40SMilanka Ringwald log_error("avdtp_get_all_capabilities: no connection for AVDTP cid 0x%02x found", avdtp_cid); 823747ec646SMilanka Ringwald return; 824747ec646SMilanka Ringwald } 825747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) return; 826747ec646SMilanka Ringwald if (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) return; 827747ec646SMilanka Ringwald connection->initiator_transaction_label++; 828747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_ALL_CAPABILITIES; 8299413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8309413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 831747ec646SMilanka Ringwald } 832747ec646SMilanka Ringwald 8336b0ee1d0SMilanka Ringwald void avdtp_get_configuration(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){ 8344ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 835747ec646SMilanka Ringwald if (!connection){ 8364ccacc40SMilanka Ringwald log_error("avdtp_get_configuration: no connection for AVDTP cid 0x%02x found", avdtp_cid); 837747ec646SMilanka Ringwald return; 838747ec646SMilanka Ringwald } 839747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) return; 840747ec646SMilanka Ringwald if (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) return; 841747ec646SMilanka Ringwald connection->initiator_transaction_label++; 842747ec646SMilanka Ringwald connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CONFIGURATION; 8439413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8449413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 845747ec646SMilanka Ringwald } 846747ec646SMilanka Ringwald 8476b0ee1d0SMilanka Ringwald void 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){ 8484ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 849747ec646SMilanka Ringwald if (!connection){ 8504ccacc40SMilanka Ringwald log_error("avdtp_set_configuration: no connection for AVDTP cid 0x%02x found", avdtp_cid); 851747ec646SMilanka Ringwald return; 852747ec646SMilanka Ringwald } 853747ec646SMilanka Ringwald if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) return; 854747ec646SMilanka Ringwald if (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) return; 855747ec646SMilanka Ringwald 8564ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 857747ec646SMilanka Ringwald if (!stream_endpoint) { 8584ccacc40SMilanka Ringwald log_error("avdtp_set_configuration: no initiator stream endpoint for seid %d", local_seid); 859747ec646SMilanka Ringwald return; 860747ec646SMilanka Ringwald } 861747ec646SMilanka Ringwald 862747ec646SMilanka Ringwald connection->initiator_transaction_label++; 8639413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8644ccacc40SMilanka Ringwald connection->local_seid = local_seid; 865*f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 866*f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 867747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION; 8689413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 869747ec646SMilanka Ringwald } 870747ec646SMilanka Ringwald 8716b0ee1d0SMilanka Ringwald void 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){ 8724ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 873747ec646SMilanka Ringwald if (!connection){ 8744ccacc40SMilanka Ringwald log_error("avdtp_reconfigure: no connection for AVDTP cid 0x%02x found", avdtp_cid); 875747ec646SMilanka Ringwald return; 876747ec646SMilanka Ringwald } 877747ec646SMilanka Ringwald //TODO: if opened only app capabilities, enable reconfigure for not opened 878747ec646SMilanka Ringwald if (connection->state < AVDTP_SIGNALING_CONNECTION_OPENED) return; 879747ec646SMilanka Ringwald if (connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) return; 8809e42cfccSMilanka Ringwald 8814ccacc40SMilanka Ringwald avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context); 88278d08d09SMilanka Ringwald if (!stream_endpoint) { 8834ccacc40SMilanka Ringwald log_error("avdtp_reconfigure: no initiator stream endpoint for seid %d", local_seid); 88478d08d09SMilanka Ringwald return; 88578d08d09SMilanka Ringwald } 88678d08d09SMilanka Ringwald 88778d08d09SMilanka Ringwald if (stream_endpoint->remote_sep_index == 0xFF){ 8888587e32cSMilanka Ringwald log_error("avdtp_reconfigure: no associated remote sep"); 88978d08d09SMilanka Ringwald return; 89078d08d09SMilanka Ringwald } 89178d08d09SMilanka Ringwald 892747ec646SMilanka Ringwald connection->initiator_transaction_label++; 8939413b167SMilanka Ringwald connection->remote_seid = remote_seid; 8946b0ee1d0SMilanka Ringwald connection->local_seid = local_seid; 895*f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration_bitmap = configured_services_bitmap; 896*f53d6fa7SMilanka Ringwald stream_endpoint->remote_configuration = configuration; 897747ec646SMilanka Ringwald stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_RECONFIGURE_STREAM_WITH_SEID; 898*f53d6fa7SMilanka Ringwald printf("AVDTP_INITIATOR_W2_RECONFIGURE_STREAM_WITH_SEID \n"); 8999413b167SMilanka Ringwald avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid); 900747ec646SMilanka Ringwald } 901747ec646SMilanka Ringwald 902f9bca1f3SMilanka Ringwald uint8_t avdtp_remote_seps_num(uint16_t avdtp_cid, avdtp_context_t * context){ 9034ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 904747ec646SMilanka Ringwald if (!connection){ 9054ccacc40SMilanka Ringwald log_error("avdtp_suspend: no connection for AVDTP cid 0x%02x found", avdtp_cid); 9069e42cfccSMilanka Ringwald return 0; 9079e42cfccSMilanka Ringwald } 9089e42cfccSMilanka Ringwald return connection->remote_seps_num; 9099e42cfccSMilanka Ringwald } 9109e42cfccSMilanka Ringwald 911f9bca1f3SMilanka Ringwald avdtp_sep_t * avdtp_remote_sep(uint16_t avdtp_cid, uint8_t index, avdtp_context_t * context){ 9124ccacc40SMilanka Ringwald avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context); 9139e42cfccSMilanka Ringwald if (!connection){ 9144ccacc40SMilanka Ringwald log_error("avdtp_suspend: no connection for AVDTP cid 0x%02x found", avdtp_cid); 9159e42cfccSMilanka Ringwald return NULL; 9169e42cfccSMilanka Ringwald } 9179e42cfccSMilanka Ringwald return &connection->remote_seps[index]; 9189e42cfccSMilanka Ringwald } 91978d08d09SMilanka Ringwald 920106b2cc4SMilanka 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){ 921106b2cc4SMilanka Ringwald UNUSED(packet_size); 922106b2cc4SMilanka Ringwald if (storage_size < 4) { 9238587e32cSMilanka Ringwald log_error("storage must have 4 bytes"); 924747ec646SMilanka Ringwald return; 925747ec646SMilanka Ringwald } 926106b2cc4SMilanka Ringwald uint8_t sampling_frequency = avdtp_choose_sbc_sampling_frequency(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_sampling_frequency_bitmap(packet)); 927106b2cc4SMilanka Ringwald uint8_t channel_mode = avdtp_choose_sbc_channel_mode(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_channel_mode_bitmap(packet)); 928106b2cc4SMilanka Ringwald uint8_t block_length = avdtp_choose_sbc_block_length(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_block_length_bitmap(packet)); 929106b2cc4SMilanka Ringwald uint8_t subbands = avdtp_choose_sbc_subbands(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_subbands_bitmap(packet)); 930106b2cc4SMilanka Ringwald 931106b2cc4SMilanka Ringwald uint8_t allocation_method = avdtp_choose_sbc_allocation_method(stream_endpoint, avdtp_subevent_signaling_media_codec_sbc_capability_get_allocation_method_bitmap(packet)); 932106b2cc4SMilanka 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)); 933106b2cc4SMilanka 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)); 934106b2cc4SMilanka Ringwald 93578d08d09SMilanka Ringwald config_storage[0] = (sampling_frequency << 4) | channel_mode; 93678d08d09SMilanka Ringwald config_storage[1] = (block_length << 4) | (subbands << 2) | allocation_method; 93778d08d09SMilanka Ringwald config_storage[2] = min_bitpool_value; 93878d08d09SMilanka Ringwald config_storage[3] = max_bitpool_value; 939a0ed80bdSMilanka Ringwald 940a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration_bitmap = store_bit16(stream_endpoint->remote_configuration_bitmap, AVDTP_MEDIA_CODEC, 1); 941a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_type = AVDTP_AUDIO; 942a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_codec_type = AVDTP_CODEC_SBC; 943a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_codec_information_len = storage_size; 944a0ed80bdSMilanka Ringwald stream_endpoint->remote_configuration.media_codec.media_codec_information = config_storage; 94578d08d09SMilanka Ringwald } 94678d08d09SMilanka Ringwald 94778d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_channel_mode(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_channel_mode_bitmap){ 94878d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 94978d08d09SMilanka Ringwald uint8_t channel_mode_bitmap = (media_codec[0] & 0x0F) & remote_channel_mode_bitmap; 95078d08d09SMilanka Ringwald 95178d08d09SMilanka Ringwald uint8_t channel_mode = AVDTP_SBC_STEREO; 95278d08d09SMilanka Ringwald if (channel_mode_bitmap & AVDTP_SBC_JOINT_STEREO){ 95378d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_JOINT_STEREO; 95478d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_STEREO){ 95578d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_STEREO; 95678d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_DUAL_CHANNEL){ 95778d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_DUAL_CHANNEL; 95878d08d09SMilanka Ringwald } else if (channel_mode_bitmap & AVDTP_SBC_MONO){ 95978d08d09SMilanka Ringwald channel_mode = AVDTP_SBC_MONO; 96078d08d09SMilanka Ringwald } 96178d08d09SMilanka Ringwald return channel_mode; 96278d08d09SMilanka Ringwald } 96378d08d09SMilanka Ringwald 96478d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_allocation_method(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_allocation_method_bitmap){ 96578d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 96678d08d09SMilanka Ringwald uint8_t allocation_method_bitmap = (media_codec[1] & 0x03) & remote_allocation_method_bitmap; 96778d08d09SMilanka Ringwald 96878d08d09SMilanka Ringwald uint8_t allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 96978d08d09SMilanka Ringwald if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS){ 97078d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS; 97178d08d09SMilanka Ringwald } else if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_SNR){ 97278d08d09SMilanka Ringwald allocation_method = AVDTP_SBC_ALLOCATION_METHOD_SNR; 97378d08d09SMilanka Ringwald } 97478d08d09SMilanka Ringwald return allocation_method; 97578d08d09SMilanka Ringwald } 97678d08d09SMilanka Ringwald 977bd1ecb8aSMilanka Ringwald uint8_t avdtp_stream_endpoint_seid(avdtp_stream_endpoint_t * stream_endpoint){ 978bd1ecb8aSMilanka Ringwald if (!stream_endpoint) return 0; 979bd1ecb8aSMilanka Ringwald return stream_endpoint->sep.seid; 980bd1ecb8aSMilanka Ringwald } 98178d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_subbands(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_subbands_bitmap){ 98278d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 98378d08d09SMilanka Ringwald uint8_t subbands_bitmap = ((media_codec[1] >> 2) & 0x03) & remote_subbands_bitmap; 98478d08d09SMilanka Ringwald 98578d08d09SMilanka Ringwald uint8_t subbands = AVDTP_SBC_SUBBANDS_8; 98678d08d09SMilanka Ringwald if (subbands_bitmap & AVDTP_SBC_SUBBANDS_8){ 98778d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_8; 98878d08d09SMilanka Ringwald } else if (subbands_bitmap & AVDTP_SBC_SUBBANDS_4){ 98978d08d09SMilanka Ringwald subbands = AVDTP_SBC_SUBBANDS_4; 99078d08d09SMilanka Ringwald } 99178d08d09SMilanka Ringwald return subbands; 99278d08d09SMilanka Ringwald } 99378d08d09SMilanka Ringwald 99478d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_block_length(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_block_length_bitmap){ 99578d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 99678d08d09SMilanka Ringwald uint8_t block_length_bitmap = (media_codec[1] >> 4) & remote_block_length_bitmap; 99778d08d09SMilanka Ringwald 99878d08d09SMilanka Ringwald uint8_t block_length = AVDTP_SBC_BLOCK_LENGTH_16; 99978d08d09SMilanka Ringwald if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_16){ 100078d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_16; 100178d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_12){ 100278d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_12; 100378d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_8){ 100478d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_8; 100578d08d09SMilanka Ringwald } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_4){ 100678d08d09SMilanka Ringwald block_length = AVDTP_SBC_BLOCK_LENGTH_4; 100778d08d09SMilanka Ringwald } 100878d08d09SMilanka Ringwald return block_length; 100978d08d09SMilanka Ringwald } 101078d08d09SMilanka Ringwald 101178d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_sampling_frequency(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_sampling_frequency_bitmap){ 101278d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 101378d08d09SMilanka Ringwald uint8_t sampling_frequency_bitmap = (media_codec[0] >> 4) & remote_sampling_frequency_bitmap; 101478d08d09SMilanka Ringwald 101578d08d09SMilanka Ringwald uint8_t sampling_frequency = AVDTP_SBC_44100; 101678d08d09SMilanka Ringwald if (sampling_frequency_bitmap & AVDTP_SBC_48000){ 101778d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_48000; 101878d08d09SMilanka Ringwald } else if (sampling_frequency_bitmap & AVDTP_SBC_44100){ 101978d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_44100; 102078d08d09SMilanka Ringwald } else if (sampling_frequency_bitmap & AVDTP_SBC_32000){ 102178d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_32000; 102278d08d09SMilanka Ringwald } else if (sampling_frequency_bitmap & AVDTP_SBC_16000){ 102378d08d09SMilanka Ringwald sampling_frequency = AVDTP_SBC_16000; 102478d08d09SMilanka Ringwald } 102578d08d09SMilanka Ringwald return sampling_frequency; 102678d08d09SMilanka Ringwald } 102778d08d09SMilanka Ringwald 102878d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_max_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_max_bitpool_value){ 102978d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 103078d08d09SMilanka Ringwald return btstack_min(media_codec[3], remote_max_bitpool_value); 103178d08d09SMilanka Ringwald } 103278d08d09SMilanka Ringwald 103378d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_min_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_min_bitpool_value){ 103478d08d09SMilanka Ringwald uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information; 103578d08d09SMilanka Ringwald return btstack_max(media_codec[2], remote_min_bitpool_value); 1036747ec646SMilanka Ringwald } 1037