xref: /btstack/src/classic/avdtp.c (revision cf837c2859b2a8bade88e018aee91d666895ab3c)
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){
1439900b7faSMilanka 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){
1899900b7faSMilanka 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){
1989900b7faSMilanka 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){
2079900b7faSMilanka 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){
2169900b7faSMilanka 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){
2289900b7faSMilanka Ringwald         log_error("Stream endpoint with given seid is not registered.");
229747ec646SMilanka Ringwald         return;
230747ec646SMilanka Ringwald     }
231747ec646SMilanka Ringwald     uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_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){
2409900b7faSMilanka Ringwald         log_error("Stream endpoint with given seid is not registered.");
241747ec646SMilanka Ringwald         return;
242747ec646SMilanka Ringwald     }
243747ec646SMilanka Ringwald     uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_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){
2529900b7faSMilanka Ringwald         log_error("Stream endpoint with given seid is not registered.");
253747ec646SMilanka Ringwald         return;
254747ec646SMilanka Ringwald     }
255747ec646SMilanka Ringwald     uint16_t bitmap = store_bit16(stream_endpoint->sep.registered_service_categories, AVDTP_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){
2659900b7faSMilanka 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){
3169900b7faSMilanka 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){
3329900b7faSMilanka 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) {
359355ac553SMilanka Ringwald         log_error("SDP 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);
43114fd128cSMatthias Ringwald                                     des_iterator_next(&prot_it);
432692c0605SMilanka Ringwald                                     switch (uuid){
433692c0605SMilanka Ringwald                                         case BLUETOOTH_PROTOCOL_L2CAP:
434692c0605SMilanka Ringwald                                             if (!des_iterator_has_more(&prot_it)) continue;
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;
4392f6083d0SMilanka Ringwald                                             de_element_get_uint16(des_iterator_get_element(&prot_it), &sdp_query_context->avdtp_version);
440692c0605SMilanka Ringwald                                             break;
441692c0605SMilanka Ringwald                                         default:
442692c0605SMilanka Ringwald                                             break;
443692c0605SMilanka Ringwald                                     }
444692c0605SMilanka Ringwald                                 }
445692c0605SMilanka Ringwald                             }
446692c0605SMilanka Ringwald                             break;
447692c0605SMilanka Ringwald                         default:
448692c0605SMilanka Ringwald                             break;
449692c0605SMilanka Ringwald                     }
450692c0605SMilanka Ringwald                 }
451692c0605SMilanka Ringwald             } else {
4528587e32cSMilanka 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));
453692c0605SMilanka Ringwald             }
454692c0605SMilanka Ringwald             break;
455692c0605SMilanka Ringwald 
456692c0605SMilanka Ringwald         case SDP_EVENT_QUERY_COMPLETE:
4570d4a198eSMatthias Ringwald             if (connection->state != AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE){
4580d4a198eSMatthias Ringwald                 // bail out, we must have had an incoming connection in the meantime;
4590d4a198eSMatthias Ringwald                 break;
4600d4a198eSMatthias Ringwald             }
4611e1ae2bcSMilanka Ringwald             status = sdp_event_query_complete_get_status(packet);
4622f6083d0SMilanka Ringwald             if (status != ERROR_CODE_SUCCESS){
4635448c259SMilanka Ringwald                 avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, status);
4642f6083d0SMilanka Ringwald                 btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection);
4652f6083d0SMilanka Ringwald                 btstack_memory_avdtp_connection_free(connection);
466355ac553SMilanka Ringwald                 log_info("SDP query failed with status 0x%02x.", status);
4671e1ae2bcSMilanka Ringwald                 break;
4681e1ae2bcSMilanka Ringwald             }
4692f6083d0SMilanka Ringwald             if (!sdp_query_context->role_supported){
4702f6083d0SMilanka Ringwald                 btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*) connection);
4712f6083d0SMilanka Ringwald                 btstack_memory_avdtp_connection_free(connection);
4722f6083d0SMilanka Ringwald                 avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, SDP_SERVICE_NOT_FOUND);
473355ac553SMilanka Ringwald                 log_info("SDP query, remote device does not support required role.");
474974d4d6eSMilanka Ringwald                 break;
475974d4d6eSMilanka Ringwald             }
4762f6083d0SMilanka Ringwald             if (!sdp_query_context->avdtp_l2cap_psm) {
4772f6083d0SMilanka Ringwald                 btstack_linked_list_remove(&sdp_query_context->connections, (btstack_linked_item_t*)connection);
4782f6083d0SMilanka Ringwald                 btstack_memory_avdtp_connection_free(connection);
4792f6083d0SMilanka Ringwald                 avdtp_signaling_emit_connection_established(sdp_query_context->avdtp_callback, sdp_query_context->avdtp_cid, connection->remote_addr, L2CAP_SERVICE_DOES_NOT_EXIST);
480355ac553SMilanka Ringwald                 log_info("SDP query, no l2cap psm found.");
4812f6083d0SMilanka Ringwald                 break;
4822f6083d0SMilanka Ringwald             }
4832f6083d0SMilanka Ringwald             connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED;
4842f6083d0SMilanka Ringwald             l2cap_create_channel(sdp_query_context->packet_handler, connection->remote_addr, sdp_query_context->avdtp_l2cap_psm, l2cap_max_mtu(), NULL);
485692c0605SMilanka Ringwald             break;
486692c0605SMilanka Ringwald     }
487692c0605SMilanka Ringwald }
488692c0605SMilanka Ringwald 
489692c0605SMilanka Ringwald 
490747ec646SMilanka Ringwald void avdtp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size, avdtp_context_t * context){
491747ec646SMilanka Ringwald     bd_addr_t event_addr;
492747ec646SMilanka Ringwald     uint16_t psm;
493747ec646SMilanka Ringwald     uint16_t local_cid;
4941e1ae2bcSMilanka Ringwald     uint8_t  status;
4950d4a198eSMatthias Ringwald     int accept_signaling_connection;
496747ec646SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = NULL;
497747ec646SMilanka Ringwald     avdtp_connection_t * connection = NULL;
498747ec646SMilanka Ringwald     btstack_linked_list_t * avdtp_connections = &context->connections;
499747ec646SMilanka Ringwald     btstack_linked_list_t * stream_endpoints =  &context->stream_endpoints;
500747ec646SMilanka Ringwald     handle_media_data = context->handle_media_data;
5018587e32cSMilanka Ringwald     // log_info("avdtp_packet_handler packet type %02x, event %02x ", packet_type, hci_event_packet_get_type(packet));
502747ec646SMilanka Ringwald     switch (packet_type) {
503747ec646SMilanka Ringwald         case L2CAP_DATA_PACKET:
5049413b167SMilanka Ringwald             connection = avdtp_connection_for_l2cap_signaling_cid(channel, context);
505747ec646SMilanka Ringwald             if (connection){
506747ec646SMilanka Ringwald                 handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context);
507747ec646SMilanka Ringwald                 break;
508747ec646SMilanka Ringwald             }
509747ec646SMilanka Ringwald 
510747ec646SMilanka Ringwald             stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context);
511747ec646SMilanka Ringwald             if (!stream_endpoint){
512747ec646SMilanka Ringwald                 if (!connection) break;
513747ec646SMilanka Ringwald                 handle_l2cap_data_packet_for_signaling_connection(connection, packet, size, context);
514747ec646SMilanka Ringwald                 break;
515747ec646SMilanka Ringwald             }
516747ec646SMilanka Ringwald 
5178c0f3635SMilanka Ringwald             if (stream_endpoint->connection){
5189413b167SMilanka Ringwald                 if (channel == stream_endpoint->connection->l2cap_signaling_cid){
519a466d508SMilanka Ringwald                     int offset = avdtp_read_signaling_header(&stream_endpoint->connection->signaling_packet, packet, size);
520a466d508SMilanka Ringwald                     if (stream_endpoint->connection->signaling_packet.message_type == AVDTP_CMD_MSG){
521a466d508SMilanka Ringwald                         avdtp_acceptor_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context);
522a466d508SMilanka Ringwald                     } else {
523a466d508SMilanka Ringwald                         avdtp_initiator_stream_config_subsm(stream_endpoint->connection, packet, size, offset, context);
524a466d508SMilanka Ringwald                     }
525747ec646SMilanka Ringwald                     break;
526747ec646SMilanka Ringwald                 }
5278c0f3635SMilanka Ringwald             }
528747ec646SMilanka Ringwald 
529747ec646SMilanka Ringwald             if (channel == stream_endpoint->l2cap_media_cid){
5308b097e29SMilanka Ringwald                 if (handle_media_data){
531fd58c900SMilanka Ringwald                     (*handle_media_data)(avdtp_local_seid(stream_endpoint), packet, size);
5328b097e29SMilanka Ringwald                 }
533747ec646SMilanka Ringwald                 break;
534747ec646SMilanka Ringwald             }
535747ec646SMilanka Ringwald 
536747ec646SMilanka Ringwald             if (channel == stream_endpoint->l2cap_reporting_cid){
537747ec646SMilanka Ringwald                 // TODO
5388587e32cSMilanka Ringwald                 log_info("L2CAP_DATA_PACKET for reporting: NOT IMPLEMENTED");
539747ec646SMilanka Ringwald             } else if (channel == stream_endpoint->l2cap_recovery_cid){
540747ec646SMilanka Ringwald                 // TODO
5418587e32cSMilanka Ringwald                 log_info("L2CAP_DATA_PACKET for recovery: NOT IMPLEMENTED");
542747ec646SMilanka Ringwald             } else {
543747ec646SMilanka Ringwald                 log_error("avdtp packet handler L2CAP_DATA_PACKET: local cid 0x%02x not found", channel);
544747ec646SMilanka Ringwald             }
545747ec646SMilanka Ringwald             break;
546747ec646SMilanka Ringwald 
547747ec646SMilanka Ringwald         case HCI_EVENT_PACKET:
548747ec646SMilanka Ringwald             switch (hci_event_packet_get_type(packet)) {
549747ec646SMilanka Ringwald                 case L2CAP_EVENT_INCOMING_CONNECTION:
550747ec646SMilanka Ringwald                     l2cap_event_incoming_connection_get_address(packet, event_addr);
551747ec646SMilanka Ringwald                     local_cid = l2cap_event_incoming_connection_get_local_cid(packet);
552747ec646SMilanka Ringwald                     connection = avdtp_connection_for_bd_addr(event_addr, context);
553355ac553SMilanka Ringwald                     log_info("L2CAP_EVENT_INCOMING_CONNECTION, local cid 0x%02x ", local_cid);
5540d4a198eSMatthias Ringwald                     if (!connection){
5550d4a198eSMatthias Ringwald                         // create connection struct for regular inconming connection request
556747ec646SMilanka Ringwald                         connection = avdtp_create_connection(event_addr, context);
5570d4a198eSMatthias Ringwald                         if (!connection){
5580d4a198eSMatthias Ringwald                             log_error("Could not create connection, reject");
5590d4a198eSMatthias Ringwald                             l2cap_decline_connection(local_cid);
5600d4a198eSMatthias Ringwald                             break;
5610d4a198eSMatthias Ringwald                         }
5620d4a198eSMatthias Ringwald                         connection->state = AVDTP_SIGNALING_CONNECTION_IDLE;
5630d4a198eSMatthias Ringwald                     }
5640d4a198eSMatthias Ringwald 
5650d4a198eSMatthias Ringwald                     // connection exists, check states
5660d4a198eSMatthias Ringwald                     log_info("Connection state %d", connection->state);
567*cf837c28SMatthias Ringwald                     accept_signaling_connection = 0;
5680d4a198eSMatthias Ringwald                     switch (connection->state){
5690d4a198eSMatthias Ringwald                         case AVDTP_SIGNALING_CONNECTION_IDLE:
5700d4a198eSMatthias Ringwald                             // regular incoming connection
5710d4a198eSMatthias Ringwald                             accept_signaling_connection = 1;
5720d4a198eSMatthias Ringwald                             break;
5730d4a198eSMatthias Ringwald                         case AVDTP_SIGNALING_W4_SDP_QUERY_COMPLETE:
5740d4a198eSMatthias Ringwald                             // incoming connection during sdp query, just accept it
5750d4a198eSMatthias Ringwald                             accept_signaling_connection = 1;
5760d4a198eSMatthias Ringwald                             break;
5770d4a198eSMatthias Ringwald                         case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED:
5780d4a198eSMatthias Ringwald                             log_info("Reject incoming connection after creating outgoing");
5790d4a198eSMatthias Ringwald                             l2cap_decline_connection(local_cid);
5800d4a198eSMatthias Ringwald                             return;
5810d4a198eSMatthias Ringwald                         case AVDTP_SIGNALING_CONNECTION_OPENED:
5820d4a198eSMatthias Ringwald                             // handled below
5830d4a198eSMatthias Ringwald                             break;
5840d4a198eSMatthias Ringwald                         case AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED:
5850d4a198eSMatthias Ringwald                             log_info("Reject incoming connection during disconnect");
5860d4a198eSMatthias Ringwald                             l2cap_decline_connection(local_cid);
5870d4a198eSMatthias Ringwald                             return;
5880d4a198eSMatthias Ringwald                     }
5890d4a198eSMatthias Ringwald 
5900d4a198eSMatthias Ringwald                     if (accept_signaling_connection){
59163331bf4SMilanka Ringwald                         connection->is_initiator = 0;
592747ec646SMilanka Ringwald                         connection->state = AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED;
5930d4a198eSMatthias Ringwald                         log_info("L2CAP_EVENT_INCOMING_CONNECTION: role is_initiator %d", connection->is_initiator);
594355ac553SMilanka Ringwald                         log_info("L2CAP_EVENT_INCOMING_CONNECTION, connection %p, state connection %d, avdtp cid 0x%02x", connection, connection->state, connection->avdtp_cid);
595747ec646SMilanka Ringwald                         l2cap_accept_connection(local_cid);
596747ec646SMilanka Ringwald                         break;
597747ec646SMilanka Ringwald                     }
598747ec646SMilanka Ringwald 
5990d4a198eSMatthias Ringwald                     // now, we're only dealing with media connections
6006b0ee1d0SMilanka Ringwald                     stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context);
601747ec646SMilanka Ringwald                     if (!stream_endpoint) {
602355ac553SMilanka Ringwald                         log_info("L2CAP_EVENT_INCOMING_CONNECTION no streamendpoint found for seid %d", connection->local_seid);
6030d4a198eSMatthias Ringwald                         l2cap_decline_connection(local_cid);
604747ec646SMilanka Ringwald                         break;
605747ec646SMilanka Ringwald                     }
606747ec646SMilanka Ringwald 
607355ac553SMilanka Ringwald                     log_info("Checking l2cap_media_cid %d, for local seid %d, state of stream endpoint %d, role is_initiator %d", stream_endpoint->l2cap_media_cid, connection->local_seid, stream_endpoint->state, connection->is_initiator);
608747ec646SMilanka Ringwald                     if (stream_endpoint->l2cap_media_cid == 0){
6099900b7faSMilanka Ringwald                         if (connection->is_initiator){
6109900b7faSMilanka Ringwald                             l2cap_decline_connection(local_cid);
6119900b7faSMilanka Ringwald                             break;
6129900b7faSMilanka Ringwald                         }
613747ec646SMilanka Ringwald                         l2cap_accept_connection(local_cid);
614747ec646SMilanka Ringwald                         break;
615747ec646SMilanka Ringwald                     }
6169900b7faSMilanka Ringwald                     l2cap_decline_connection(local_cid);
617747ec646SMilanka Ringwald                     break;
618747ec646SMilanka Ringwald 
619747ec646SMilanka Ringwald                 case L2CAP_EVENT_CHANNEL_OPENED:
620a0b8a58cSMilanka Ringwald                     psm = l2cap_event_channel_opened_get_psm(packet);
621a0b8a58cSMilanka Ringwald                     if (psm != BLUETOOTH_PROTOCOL_AVDTP){
622355ac553SMilanka Ringwald                         log_info("Unexpected PSM - Not implemented yet, avdtp sink: L2CAP_EVENT_CHANNEL_OPENED ");
623a0b8a58cSMilanka Ringwald                         return;
624a0b8a58cSMilanka Ringwald                     }
625a0b8a58cSMilanka Ringwald 
6261e1ae2bcSMilanka Ringwald                     status = l2cap_event_channel_opened_get_status(packet);
627747ec646SMilanka Ringwald                     // inform about new l2cap connection
628747ec646SMilanka Ringwald                     l2cap_event_channel_opened_get_address(packet, event_addr);
6297050d2caSMilanka Ringwald                     local_cid = l2cap_event_channel_opened_get_local_cid(packet);
630a0b8a58cSMilanka Ringwald                     connection = avdtp_connection_for_bd_addr(event_addr, context);
631a0b8a58cSMilanka Ringwald 
632355ac553SMilanka Ringwald                     log_info("L2CAP_EVENT_CHANNEL_OPENED: status %d, cid 0x%02x , signaling connection %p ", status, local_cid, connection);
633a0b8a58cSMilanka Ringwald                     connection = avdtp_connection_for_bd_addr(event_addr, context);
634a0b8a58cSMilanka Ringwald                     if (!connection){
635355ac553SMilanka Ringwald                         log_info("L2CAP_EVENT_CHANNEL_OPENED 2: status %d, signaling connection %p ", status, connection);
636a0b8a58cSMilanka Ringwald                         break;
637a0b8a58cSMilanka Ringwald                     }
638a0b8a58cSMilanka Ringwald 
639a0b8a58cSMilanka Ringwald                     if (connection->l2cap_signaling_cid == 0) {
6401e1ae2bcSMilanka Ringwald                         if (status){
641355ac553SMilanka Ringwald                             log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status);
64255ddebccSMilanka Ringwald                             connection->state = AVDTP_SIGNALING_CONNECTION_IDLE;
643a0b8a58cSMilanka Ringwald                             connection->l2cap_signaling_cid = 0;
6444ccacc40SMilanka Ringwald                             avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, l2cap_event_channel_opened_get_status(packet));
645747ec646SMilanka Ringwald                             break;
646747ec646SMilanka Ringwald                         }
647a0b8a58cSMilanka Ringwald                         if (connection->state != AVDTP_SIGNALING_CONNECTION_W4_L2CAP_CONNECTED) {
648355ac553SMilanka Ringwald                             log_info("L2CAP connection to %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status);
649a0b8a58cSMilanka Ringwald                             avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, AVDTP_CONNECTION_IN_WRONG_STATE);
650a0b8a58cSMilanka Ringwald                             break;
65155ddebccSMilanka Ringwald                         }
6529413b167SMilanka Ringwald                         connection->l2cap_signaling_cid = local_cid;
6536b0ee1d0SMilanka Ringwald                         connection->local_seid = 0;
654747ec646SMilanka Ringwald                         connection->state = AVDTP_SIGNALING_CONNECTION_OPENED;
655355ac553SMilanka Ringwald                         log_info("AVDTP_SIGNALING_CONNECTION_OPENED, connection %p, l2cap_signaling_cid 0x%02x, avdtp_cid 0x%02x", connection, local_cid, connection->avdtp_cid);
6564ccacc40SMilanka Ringwald                         avdtp_signaling_emit_connection_established(context->avdtp_callback, connection->avdtp_cid, event_addr, 0);
657747ec646SMilanka Ringwald                         break;
658747ec646SMilanka Ringwald                     }
659747ec646SMilanka Ringwald 
6606b0ee1d0SMilanka Ringwald                     stream_endpoint = avdtp_stream_endpoint_for_seid(connection->local_seid, context);
661747ec646SMilanka Ringwald                     if (!stream_endpoint){
662355ac553SMilanka Ringwald                         log_info("L2CAP_EVENT_CHANNEL_OPENED: stream_endpoint not found");
663747ec646SMilanka Ringwald                         return;
664747ec646SMilanka Ringwald                     }
665a466d508SMilanka Ringwald 
666a466d508SMilanka Ringwald                     if (stream_endpoint->l2cap_media_cid == 0){
667a466d508SMilanka Ringwald                         status = l2cap_event_channel_opened_get_status(packet);
668a466d508SMilanka Ringwald                         if (status){
669355ac553SMilanka 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));
670a466d508SMilanka Ringwald                             stream_endpoint->state = AVDTP_STREAM_ENDPOINT_IDLE;
671596b7fdcSMilanka 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);
672a466d508SMilanka Ringwald                             break;
673a466d508SMilanka Ringwald                         }
674a466d508SMilanka Ringwald                         if (stream_endpoint->state != AVDTP_STREAM_ENDPOINT_W4_L2CAP_FOR_MEDIA_CONNECTED){
675355ac553SMilanka 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));
676596b7fdcSMilanka 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);
677a466d508SMilanka Ringwald                             break;
678a466d508SMilanka Ringwald                         }
679a466d508SMilanka Ringwald 
680a466d508SMilanka Ringwald                         stream_endpoint->state = AVDTP_STREAM_ENDPOINT_OPENED;
681a466d508SMilanka Ringwald                         stream_endpoint->connection = connection;
682a466d508SMilanka Ringwald                         stream_endpoint->l2cap_media_cid = l2cap_event_channel_opened_get_local_cid(packet);
683a466d508SMilanka Ringwald                         stream_endpoint->media_con_handle = l2cap_event_channel_opened_get_handle(packet);
684d1207cd8SMilanka Ringwald 
685355ac553SMilanka 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));
686596b7fdcSMilanka 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);
687a466d508SMilanka Ringwald                         return;
688a466d508SMilanka Ringwald                     }
689747ec646SMilanka Ringwald                     break;
690747ec646SMilanka Ringwald 
691747ec646SMilanka Ringwald                 case L2CAP_EVENT_CHANNEL_CLOSED:
692747ec646SMilanka Ringwald                     local_cid = l2cap_event_channel_closed_get_local_cid(packet);
6939413b167SMilanka Ringwald                     connection = avdtp_connection_for_l2cap_signaling_cid(local_cid, context);
6940bd7cb1fSMilanka Ringwald                     stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(local_cid, context);
695485c0a4cSMilanka Ringwald                     log_info("Received L2CAP_EVENT_CHANNEL_CLOSED, cid 0x%2x, connection %p, stream_endpoint %p", local_cid, connection, stream_endpoint);
6960bd7cb1fSMilanka Ringwald 
697a466d508SMilanka Ringwald                     if (stream_endpoint){
698a466d508SMilanka Ringwald                         if (stream_endpoint->l2cap_media_cid == local_cid){
699a466d508SMilanka Ringwald                             connection = stream_endpoint->connection;
700596b7fdcSMilanka Ringwald                             if (connection) {
701596b7fdcSMilanka Ringwald                                 avdtp_streaming_emit_connection_released(context->avdtp_callback, connection->avdtp_cid, avdtp_local_seid(stream_endpoint));
702596b7fdcSMilanka Ringwald                             }
703485c0a4cSMilanka Ringwald                             avdtp_reset_stream_endpoint(stream_endpoint);
704a466d508SMilanka Ringwald                             if (connection && connection->disconnect){
705a466d508SMilanka Ringwald                                 avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid);
706a466d508SMilanka Ringwald                             }
707a466d508SMilanka Ringwald                             break;
708a466d508SMilanka Ringwald                         }
709a466d508SMilanka Ringwald                         if (stream_endpoint->l2cap_recovery_cid == local_cid){
710355ac553SMilanka Ringwald                             log_info("L2CAP_EVENT_CHANNEL_CLOSED recovery cid 0x%0x", local_cid);
711a466d508SMilanka Ringwald                             stream_endpoint->l2cap_recovery_cid = 0;
712a466d508SMilanka Ringwald                             break;
713a466d508SMilanka Ringwald                         }
714a466d508SMilanka Ringwald 
715a466d508SMilanka Ringwald                         if (stream_endpoint->l2cap_reporting_cid == local_cid){
716355ac553SMilanka Ringwald                             log_info("L2CAP_EVENT_CHANNEL_CLOSED reporting cid 0x%0x", local_cid);
717a466d508SMilanka Ringwald                             stream_endpoint->l2cap_reporting_cid = 0;
718a466d508SMilanka Ringwald                             break;
719a466d508SMilanka Ringwald                         }
720a466d508SMilanka Ringwald                     }
721596b7fdcSMilanka Ringwald 
722596b7fdcSMilanka Ringwald                     if (connection){
723596b7fdcSMilanka Ringwald                         btstack_linked_list_iterator_t it;
724596b7fdcSMilanka Ringwald                         btstack_linked_list_iterator_init(&it, stream_endpoints);
725596b7fdcSMilanka Ringwald                         while (btstack_linked_list_iterator_has_next(&it)){
726596b7fdcSMilanka Ringwald                             avdtp_stream_endpoint_t * _stream_endpoint = (avdtp_stream_endpoint_t *)btstack_linked_list_iterator_next(&it);
727596b7fdcSMilanka Ringwald                             if (_stream_endpoint->connection == connection){
728485c0a4cSMilanka Ringwald                                 avdtp_reset_stream_endpoint(_stream_endpoint);
729596b7fdcSMilanka Ringwald                             }
730596b7fdcSMilanka Ringwald                         }
731485c0a4cSMilanka Ringwald                         btstack_run_loop_remove_timer(&connection->configuration_timer);
732485c0a4cSMilanka Ringwald                         avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid);
733485c0a4cSMilanka Ringwald                         btstack_linked_list_remove(avdtp_connections, (btstack_linked_item_t*) connection);
734596b7fdcSMilanka Ringwald                         btstack_memory_avdtp_connection_free(connection);
735596b7fdcSMilanka Ringwald                         break;
736596b7fdcSMilanka Ringwald                     }
737596b7fdcSMilanka Ringwald 
738747ec646SMilanka Ringwald                     break;
739747ec646SMilanka Ringwald 
740747ec646SMilanka Ringwald                 case HCI_EVENT_DISCONNECTION_COMPLETE:
741747ec646SMilanka Ringwald                     break;
742747ec646SMilanka Ringwald 
743747ec646SMilanka Ringwald                 case L2CAP_EVENT_CAN_SEND_NOW:
7449413b167SMilanka Ringwald                     connection = avdtp_connection_for_l2cap_signaling_cid(channel, context);
745747ec646SMilanka Ringwald                     if (!connection) {
746747ec646SMilanka Ringwald                         stream_endpoint = avdtp_stream_endpoint_for_l2cap_cid(channel, context);
747747ec646SMilanka Ringwald                         if (!stream_endpoint->connection) break;
748747ec646SMilanka Ringwald                         connection = stream_endpoint->connection;
749747ec646SMilanka Ringwald                     }
750747ec646SMilanka Ringwald                     avdtp_handle_can_send_now(connection, channel, context);
751747ec646SMilanka Ringwald                     break;
752747ec646SMilanka Ringwald                 default:
753355ac553SMilanka Ringwald                     log_info("Unknown HCI event type %02x", hci_event_packet_get_type(packet));
754747ec646SMilanka Ringwald                     break;
755747ec646SMilanka Ringwald             }
756747ec646SMilanka Ringwald             break;
757747ec646SMilanka Ringwald 
758747ec646SMilanka Ringwald         default:
759747ec646SMilanka Ringwald             // other packet type
760747ec646SMilanka Ringwald             break;
761747ec646SMilanka Ringwald     }
762747ec646SMilanka Ringwald }
763747ec646SMilanka Ringwald 
7644ccacc40SMilanka Ringwald uint8_t avdtp_disconnect(uint16_t avdtp_cid, avdtp_context_t * context){
7654ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
7664ccacc40SMilanka Ringwald     if (!connection) return BTSTACK_MEMORY_ALLOC_FAILED;
767a466d508SMilanka Ringwald     if (connection->state == AVDTP_SIGNALING_CONNECTION_IDLE){
768a466d508SMilanka Ringwald         avdtp_signaling_emit_connection_released(context->avdtp_callback, connection->avdtp_cid);
769a466d508SMilanka Ringwald         return ERROR_CODE_SUCCESS;
770a466d508SMilanka Ringwald     }
771a466d508SMilanka Ringwald     if (connection->state == AVDTP_SIGNALING_CONNECTION_W4_L2CAP_DISCONNECTED) return ERROR_CODE_SUCCESS;
772747ec646SMilanka Ringwald 
773747ec646SMilanka Ringwald     connection->disconnect = 1;
7749413b167SMilanka Ringwald     avdtp_request_can_send_now_self(connection, connection->l2cap_signaling_cid);
7754ccacc40SMilanka Ringwald     return ERROR_CODE_SUCCESS;
776747ec646SMilanka Ringwald }
777747ec646SMilanka Ringwald 
7786b0ee1d0SMilanka Ringwald uint8_t avdtp_open_stream(uint16_t avdtp_cid, uint8_t local_seid, uint8_t remote_seid, avdtp_context_t * context){
7794ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
780747ec646SMilanka Ringwald     if (!connection){
7818587e32cSMilanka Ringwald         log_error("avdtp_media_connect: no connection for signaling cid 0x%02x found", avdtp_cid);
7824ccacc40SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
783747ec646SMilanka Ringwald     }
784747ec646SMilanka Ringwald 
785747ec646SMilanka Ringwald     if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED) {
7868587e32cSMilanka Ringwald         log_error("avdtp_media_connect: wrong connection state %d", connection->state);
7874ccacc40SMilanka Ringwald         return AVDTP_CONNECTION_IN_WRONG_STATE;
788747ec646SMilanka Ringwald     }
789747ec646SMilanka Ringwald 
7906b0ee1d0SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context);
791747ec646SMilanka Ringwald     if (!stream_endpoint) {
7926b0ee1d0SMilanka Ringwald         log_error("avdtp_media_connect: no stream_endpoint with seid %d found", local_seid);
7934ccacc40SMilanka Ringwald         return AVDTP_SEID_DOES_NOT_EXIST;
794747ec646SMilanka Ringwald     }
795747ec646SMilanka Ringwald 
796485c0a4cSMilanka Ringwald     if (stream_endpoint->remote_sep.seid != remote_seid){
797485c0a4cSMilanka Ringwald         log_error("avdtp_media_connect: no remote sep with seid %d registered with the stream endpoint", remote_seid);
798485c0a4cSMilanka Ringwald         return AVDTP_SEID_DOES_NOT_EXIST;
799485c0a4cSMilanka Ringwald     }
800485c0a4cSMilanka Ringwald 
8014ccacc40SMilanka Ringwald     if (stream_endpoint->state < AVDTP_STREAM_ENDPOINT_CONFIGURED) return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE;
802747ec646SMilanka Ringwald 
803747ec646SMilanka Ringwald     connection->initiator_transaction_label++;
8049413b167SMilanka Ringwald     connection->remote_seid = remote_seid;
8056b0ee1d0SMilanka Ringwald     connection->local_seid = local_seid;
806747ec646SMilanka Ringwald     stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_OPEN_STREAM;
807747ec646SMilanka Ringwald     stream_endpoint->state = AVDTP_STREAM_ENDPOINT_W2_REQUEST_OPEN_STREAM;
8089413b167SMilanka Ringwald     avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
8094ccacc40SMilanka Ringwald     return ERROR_CODE_SUCCESS;
810747ec646SMilanka Ringwald }
811747ec646SMilanka Ringwald 
8124ccacc40SMilanka Ringwald uint8_t avdtp_start_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){
8134ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
81446e6b063SMilanka Ringwald     if (!connection){
8154ccacc40SMilanka Ringwald         log_error("avdtp_start_stream: no connection for signaling cid 0x%02x found", avdtp_cid);
8164ccacc40SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
81746e6b063SMilanka Ringwald     }
8185cfe7f4cSMilanka Ringwald 
8194ccacc40SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context);
8204ccacc40SMilanka Ringwald     if (!stream_endpoint) {
8214ccacc40SMilanka Ringwald         log_error("avdtp_start_stream: no stream_endpoint with seid %d found", local_seid);
8224ccacc40SMilanka Ringwald         return AVDTP_SEID_DOES_NOT_EXIST;
8234ccacc40SMilanka Ringwald     }
8244ccacc40SMilanka Ringwald 
8254ccacc40SMilanka Ringwald     if (stream_endpoint->l2cap_media_cid == 0){
8264ccacc40SMilanka Ringwald         log_error("avdtp_start_stream: no media connection for stream_endpoint with seid %d found", local_seid);
8274ccacc40SMilanka Ringwald         return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST;
8284ccacc40SMilanka Ringwald     }
8294ccacc40SMilanka Ringwald 
830485c0a4cSMilanka Ringwald     if (!is_avdtp_remote_seid_registered(stream_endpoint)){
831485c0a4cSMilanka Ringwald         log_error("avdtp_media_connect: no remote sep registered with the stream endpoint");
832485c0a4cSMilanka Ringwald         return AVDTP_SEID_DOES_NOT_EXIST;
8334ccacc40SMilanka Ringwald     }
8344ccacc40SMilanka Ringwald 
83560ec20d0SMilanka Ringwald     stream_endpoint->start_stream = 1;
8364ccacc40SMilanka Ringwald     connection->local_seid = local_seid;
837c580323eSMilanka Ringwald     connection->remote_seid = stream_endpoint->remote_sep.seid;
8389413b167SMilanka Ringwald     avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
8394ccacc40SMilanka Ringwald     return ERROR_CODE_SUCCESS;
840747ec646SMilanka Ringwald }
841747ec646SMilanka Ringwald 
8424ccacc40SMilanka Ringwald uint8_t avdtp_stop_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){
8434ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
844747ec646SMilanka Ringwald     if (!connection){
8454ccacc40SMilanka Ringwald         log_error("avdtp_stop_stream: no connection for signaling cid 0x%02x found", avdtp_cid);
8464ccacc40SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
847747ec646SMilanka Ringwald     }
8484ccacc40SMilanka Ringwald 
8494ccacc40SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context);
8504ccacc40SMilanka Ringwald     if (!stream_endpoint) {
8514ccacc40SMilanka Ringwald         log_error("avdtp_stop_stream: no stream_endpoint with seid %d found", local_seid);
8524ccacc40SMilanka Ringwald         return AVDTP_SEID_DOES_NOT_EXIST;
8534ccacc40SMilanka Ringwald     }
8544ccacc40SMilanka Ringwald 
8554ccacc40SMilanka Ringwald     if (stream_endpoint->l2cap_media_cid == 0){
8564ccacc40SMilanka Ringwald         log_error("avdtp_stop_stream: no media connection for stream_endpoint with seid %d found", local_seid);
8574ccacc40SMilanka Ringwald         return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST;
8584ccacc40SMilanka Ringwald     }
859485c0a4cSMilanka Ringwald 
860485c0a4cSMilanka Ringwald     if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->stop_stream){
861485c0a4cSMilanka Ringwald         return ERROR_CODE_SUCCESS;
862485c0a4cSMilanka Ringwald     }
8634ccacc40SMilanka Ringwald 
86460ec20d0SMilanka Ringwald     stream_endpoint->stop_stream = 1;
8654ccacc40SMilanka Ringwald     connection->local_seid = local_seid;
866c580323eSMilanka Ringwald     connection->remote_seid = stream_endpoint->remote_sep.seid;
8679413b167SMilanka Ringwald     avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
8684ccacc40SMilanka Ringwald     return ERROR_CODE_SUCCESS;
869747ec646SMilanka Ringwald }
870747ec646SMilanka Ringwald 
8714ccacc40SMilanka Ringwald uint8_t avdtp_abort_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){
8724ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
87360ec20d0SMilanka Ringwald     if (!connection){
8744ccacc40SMilanka Ringwald         log_error("avdtp_abort_stream: no connection for signaling cid 0x%02x found", avdtp_cid);
8754ccacc40SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
876747ec646SMilanka Ringwald     }
8774ccacc40SMilanka Ringwald 
8784ccacc40SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context);
8794ccacc40SMilanka Ringwald     if (!stream_endpoint) {
8804ccacc40SMilanka Ringwald         log_error("avdtp_abort_stream: no stream_endpoint with seid %d found", local_seid);
8814ccacc40SMilanka Ringwald         return AVDTP_SEID_DOES_NOT_EXIST;
8824ccacc40SMilanka Ringwald     }
8834ccacc40SMilanka Ringwald 
8844ccacc40SMilanka Ringwald     if (stream_endpoint->l2cap_media_cid == 0){
8854ccacc40SMilanka Ringwald         log_error("avdtp_abort_stream: no media connection for stream_endpoint with seid %d found", local_seid);
8864ccacc40SMilanka Ringwald         return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST;
8874ccacc40SMilanka Ringwald     }
888485c0a4cSMilanka Ringwald 
889485c0a4cSMilanka Ringwald     if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->abort_stream){
890485c0a4cSMilanka Ringwald         return ERROR_CODE_SUCCESS;
891485c0a4cSMilanka Ringwald     }
8924ccacc40SMilanka Ringwald 
89360ec20d0SMilanka Ringwald     stream_endpoint->abort_stream = 1;
8946b0ee1d0SMilanka Ringwald     connection->local_seid = local_seid;
895c580323eSMilanka Ringwald     connection->remote_seid = stream_endpoint->remote_sep.seid;
8969413b167SMilanka Ringwald     avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
8974ccacc40SMilanka Ringwald     return ERROR_CODE_SUCCESS;
898747ec646SMilanka Ringwald }
899747ec646SMilanka Ringwald 
9004ccacc40SMilanka Ringwald uint8_t avdtp_suspend_stream(uint16_t avdtp_cid, uint8_t local_seid, avdtp_context_t * context){
9014ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
902747ec646SMilanka Ringwald     if (!connection){
9034ccacc40SMilanka Ringwald         log_error("avdtp_suspend_stream: no connection for signaling cid 0x%02x found", avdtp_cid);
9044ccacc40SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
90560ec20d0SMilanka Ringwald     }
9064ccacc40SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_with_seid(local_seid, context);
9074ccacc40SMilanka Ringwald     if (!stream_endpoint) {
9084ccacc40SMilanka Ringwald         log_error("avdtp_suspend_stream: no stream_endpoint with seid %d found", local_seid);
9094ccacc40SMilanka Ringwald         return AVDTP_SEID_DOES_NOT_EXIST;
9104ccacc40SMilanka Ringwald     }
9114ccacc40SMilanka Ringwald 
9124ccacc40SMilanka Ringwald     if (stream_endpoint->l2cap_media_cid == 0){
9134ccacc40SMilanka Ringwald         log_error("avdtp_suspend_stream: no media connection for stream_endpoint with seid %d found", local_seid);
9144ccacc40SMilanka Ringwald         return AVDTP_MEDIA_CONNECTION_DOES_NOT_EXIST;
9154ccacc40SMilanka Ringwald     }
916485c0a4cSMilanka Ringwald 
917485c0a4cSMilanka Ringwald     if (!is_avdtp_remote_seid_registered(stream_endpoint) || stream_endpoint->suspend_stream){
918485c0a4cSMilanka Ringwald         return ERROR_CODE_SUCCESS;
919485c0a4cSMilanka Ringwald     }
9204ccacc40SMilanka Ringwald 
92160ec20d0SMilanka Ringwald     stream_endpoint->suspend_stream = 1;
9224ccacc40SMilanka Ringwald     connection->local_seid = local_seid;
923c580323eSMilanka Ringwald     connection->remote_seid = stream_endpoint->remote_sep.seid;
9249413b167SMilanka Ringwald     avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
9254ccacc40SMilanka Ringwald     return ERROR_CODE_SUCCESS;
926747ec646SMilanka Ringwald }
927747ec646SMilanka Ringwald 
9289974aee0SMilanka Ringwald uint8_t avdtp_discover_stream_endpoints(uint16_t avdtp_cid, avdtp_context_t * context){
9294ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
930747ec646SMilanka Ringwald     if (!connection){
9318587e32cSMilanka Ringwald         log_error("avdtp_discover_stream_endpoints: no connection for signaling cid 0x%02x found", avdtp_cid);
9329974aee0SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
9339974aee0SMilanka Ringwald     }
9349974aee0SMilanka Ringwald     if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED ||
9359974aee0SMilanka Ringwald         connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) {
9369974aee0SMilanka Ringwald         return AVDTP_CONNECTION_IN_WRONG_STATE;
937747ec646SMilanka Ringwald     }
938ec3d71e3SMilanka Ringwald 
939747ec646SMilanka Ringwald     connection->initiator_transaction_label++;
940747ec646SMilanka Ringwald     connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_DISCOVER_SEPS;
9419974aee0SMilanka Ringwald     return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
942747ec646SMilanka Ringwald }
943747ec646SMilanka Ringwald 
944747ec646SMilanka Ringwald 
9459974aee0SMilanka Ringwald uint8_t avdtp_get_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){
9464ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
947747ec646SMilanka Ringwald     if (!connection){
9489900b7faSMilanka Ringwald         log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid);
9499974aee0SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
950747ec646SMilanka Ringwald     }
9519974aee0SMilanka Ringwald     if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED ||
9529974aee0SMilanka Ringwald         connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) {
9539974aee0SMilanka Ringwald         return AVDTP_CONNECTION_IN_WRONG_STATE;
9549974aee0SMilanka Ringwald     }
9559974aee0SMilanka Ringwald 
956747ec646SMilanka Ringwald     connection->initiator_transaction_label++;
957747ec646SMilanka Ringwald     connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CAPABILITIES;
9589413b167SMilanka Ringwald     connection->remote_seid = remote_seid;
9599974aee0SMilanka Ringwald     return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
960747ec646SMilanka Ringwald }
961747ec646SMilanka Ringwald 
962747ec646SMilanka Ringwald 
9639974aee0SMilanka Ringwald uint8_t avdtp_get_all_capabilities(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){
9644ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
965747ec646SMilanka Ringwald     if (!connection){
9669900b7faSMilanka Ringwald         log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid);
9679974aee0SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
968747ec646SMilanka Ringwald     }
9699974aee0SMilanka Ringwald     if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED ||
9709974aee0SMilanka Ringwald         connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) {
9719974aee0SMilanka Ringwald         return AVDTP_CONNECTION_IN_WRONG_STATE;
9729974aee0SMilanka Ringwald     }
9739974aee0SMilanka Ringwald 
974747ec646SMilanka Ringwald     connection->initiator_transaction_label++;
975747ec646SMilanka Ringwald     connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_ALL_CAPABILITIES;
9769413b167SMilanka Ringwald     connection->remote_seid = remote_seid;
9779974aee0SMilanka Ringwald     return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
978747ec646SMilanka Ringwald }
979747ec646SMilanka Ringwald 
9809974aee0SMilanka Ringwald uint8_t avdtp_get_configuration(uint16_t avdtp_cid, uint8_t remote_seid, avdtp_context_t * context){
9814ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
982747ec646SMilanka Ringwald     if (!connection){
9839900b7faSMilanka Ringwald         log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid);
9849974aee0SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
985747ec646SMilanka Ringwald     }
9869974aee0SMilanka Ringwald     if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED ||
9879974aee0SMilanka Ringwald         connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) {
9889974aee0SMilanka Ringwald         return AVDTP_CONNECTION_IN_WRONG_STATE;
9899974aee0SMilanka Ringwald     }
9909974aee0SMilanka Ringwald 
991747ec646SMilanka Ringwald     connection->initiator_transaction_label++;
992747ec646SMilanka Ringwald     connection->initiator_connection_state = AVDTP_SIGNALING_CONNECTION_INITIATOR_W2_GET_CONFIGURATION;
9939413b167SMilanka Ringwald     connection->remote_seid = remote_seid;
9949974aee0SMilanka Ringwald     return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
995747ec646SMilanka Ringwald }
996747ec646SMilanka Ringwald 
9979974aee0SMilanka 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){
9984ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
999747ec646SMilanka Ringwald     if (!connection){
10009900b7faSMilanka Ringwald         log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid);
10019974aee0SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
1002747ec646SMilanka Ringwald     }
10039974aee0SMilanka Ringwald     if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED ||
10049974aee0SMilanka Ringwald         connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) {
1005485c0a4cSMilanka Ringwald         log_error("connection in wrong state, %d, initiator state %d", connection->state, connection->initiator_connection_state);
10069974aee0SMilanka Ringwald         return AVDTP_CONNECTION_IN_WRONG_STATE;
10079974aee0SMilanka Ringwald     }
1008747ec646SMilanka Ringwald 
10094ccacc40SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context);
1010747ec646SMilanka Ringwald     if (!stream_endpoint) {
10119900b7faSMilanka Ringwald         log_error("No initiator stream endpoint for seid %d", local_seid);
10129974aee0SMilanka Ringwald         return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST;
1013747ec646SMilanka Ringwald     }
1014417b4996SMilanka Ringwald     if (stream_endpoint->state >= AVDTP_STREAM_ENDPOINT_CONFIGURED){
1015485c0a4cSMilanka Ringwald         log_error("Stream endpoint seid %d in wrong state %d", local_seid, stream_endpoint->state);
1016417b4996SMilanka Ringwald         return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE;
1017417b4996SMilanka Ringwald     }
1018bdbc3ef6SMilanka Ringwald     connection->active_stream_endpoint = (void*) stream_endpoint;
1019d8b859a2SMilanka Ringwald     connection->is_configuration_initiated_locally = 1;
102063331bf4SMilanka Ringwald     connection->is_initiator = 1;
1021747ec646SMilanka Ringwald 
1022747ec646SMilanka Ringwald     connection->initiator_transaction_label++;
10239413b167SMilanka Ringwald     connection->remote_seid = remote_seid;
10244ccacc40SMilanka Ringwald     connection->local_seid = local_seid;
1025f53d6fa7SMilanka Ringwald     stream_endpoint->remote_configuration_bitmap = configured_services_bitmap;
1026f53d6fa7SMilanka Ringwald     stream_endpoint->remote_configuration = configuration;
1027747ec646SMilanka Ringwald     stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_SET_CONFIGURATION;
1028ffa6c160SMilanka Ringwald 
1029ffa6c160SMilanka Ringwald     // cache media codec information for SBC
1030ffa6c160SMilanka Ringwald     stream_endpoint->media_codec_type = configuration.media_codec.media_codec_type;
1031ffa6c160SMilanka Ringwald     if (configuration.media_codec.media_codec_type == AVDTP_CODEC_SBC){
1032ffa6c160SMilanka Ringwald         stream_endpoint->media_type = configuration.media_codec.media_type;
1033ffa6c160SMilanka Ringwald         memcpy(stream_endpoint->media_codec_sbc_info, configuration.media_codec.media_codec_information, 4);
1034ffa6c160SMilanka Ringwald     }
10359974aee0SMilanka Ringwald     return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
1036747ec646SMilanka Ringwald }
1037747ec646SMilanka Ringwald 
10389974aee0SMilanka 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){
10394ccacc40SMilanka Ringwald     avdtp_connection_t * connection = avdtp_connection_for_avdtp_cid(avdtp_cid, context);
1040747ec646SMilanka Ringwald     if (!connection){
10419900b7faSMilanka Ringwald         log_error("No connection for AVDTP cid 0x%02x found", avdtp_cid);
10429974aee0SMilanka Ringwald         return AVDTP_CONNECTION_DOES_NOT_EXIST;
1043747ec646SMilanka Ringwald     }
1044747ec646SMilanka Ringwald     //TODO: if opened only app capabilities, enable reconfigure for not opened
10459974aee0SMilanka Ringwald     if (connection->state != AVDTP_SIGNALING_CONNECTION_OPENED ||
10469974aee0SMilanka Ringwald         connection->initiator_connection_state != AVDTP_SIGNALING_CONNECTION_INITIATOR_IDLE) {
10479974aee0SMilanka Ringwald         return AVDTP_CONNECTION_IN_WRONG_STATE;
10489974aee0SMilanka Ringwald     }
10499e42cfccSMilanka Ringwald 
10504ccacc40SMilanka Ringwald     avdtp_stream_endpoint_t * stream_endpoint = avdtp_stream_endpoint_for_seid(local_seid, context);
105178d08d09SMilanka Ringwald     if (!stream_endpoint) {
10524ccacc40SMilanka Ringwald         log_error("avdtp_reconfigure: no initiator stream endpoint for seid %d", local_seid);
10539974aee0SMilanka Ringwald         return AVDTP_STREAM_ENDPOINT_DOES_NOT_EXIST;
105478d08d09SMilanka Ringwald     }
105578d08d09SMilanka Ringwald 
1056485c0a4cSMilanka Ringwald     if (!is_avdtp_remote_seid_registered(stream_endpoint)){
10578587e32cSMilanka Ringwald         log_error("avdtp_reconfigure: no associated remote sep");
10589974aee0SMilanka Ringwald         return AVDTP_STREAM_ENDPOINT_IN_WRONG_STATE;
105978d08d09SMilanka Ringwald     }
1060485c0a4cSMilanka Ringwald 
1061747ec646SMilanka Ringwald     connection->initiator_transaction_label++;
10629413b167SMilanka Ringwald     connection->remote_seid = remote_seid;
10636b0ee1d0SMilanka Ringwald     connection->local_seid = local_seid;
1064f53d6fa7SMilanka Ringwald     stream_endpoint->remote_configuration_bitmap = configured_services_bitmap;
1065f53d6fa7SMilanka Ringwald     stream_endpoint->remote_configuration = configuration;
1066747ec646SMilanka Ringwald     stream_endpoint->initiator_config_state = AVDTP_INITIATOR_W2_RECONFIGURE_STREAM_WITH_SEID;
10679974aee0SMilanka Ringwald     return avdtp_request_can_send_now_initiator(connection, connection->l2cap_signaling_cid);
1068747ec646SMilanka Ringwald }
1069747ec646SMilanka Ringwald 
10708e7044f9SMatthias Ringwald void    avdtp_set_preferred_sampling_frequeny(avdtp_stream_endpoint_t * stream_endpoint, uint32_t sampling_frequency){
10718e7044f9SMatthias Ringwald     stream_endpoint->preferred_sampling_frequency = sampling_frequency;
10728e7044f9SMatthias Ringwald }
10738e7044f9SMatthias Ringwald 
107478d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_channel_mode(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_channel_mode_bitmap){
107578d08d09SMilanka Ringwald     uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information;
107678d08d09SMilanka Ringwald     uint8_t channel_mode_bitmap = (media_codec[0] & 0x0F) & remote_channel_mode_bitmap;
107778d08d09SMilanka Ringwald 
107878d08d09SMilanka Ringwald     uint8_t channel_mode = AVDTP_SBC_STEREO;
107978d08d09SMilanka Ringwald     if (channel_mode_bitmap & AVDTP_SBC_JOINT_STEREO){
108078d08d09SMilanka Ringwald         channel_mode = AVDTP_SBC_JOINT_STEREO;
108178d08d09SMilanka Ringwald     } else if (channel_mode_bitmap & AVDTP_SBC_STEREO){
108278d08d09SMilanka Ringwald         channel_mode = AVDTP_SBC_STEREO;
108378d08d09SMilanka Ringwald     } else if (channel_mode_bitmap & AVDTP_SBC_DUAL_CHANNEL){
108478d08d09SMilanka Ringwald         channel_mode = AVDTP_SBC_DUAL_CHANNEL;
108578d08d09SMilanka Ringwald     } else if (channel_mode_bitmap & AVDTP_SBC_MONO){
108678d08d09SMilanka Ringwald         channel_mode = AVDTP_SBC_MONO;
108778d08d09SMilanka Ringwald     }
108878d08d09SMilanka Ringwald     return channel_mode;
108978d08d09SMilanka Ringwald }
109078d08d09SMilanka Ringwald 
109178d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_allocation_method(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_allocation_method_bitmap){
109278d08d09SMilanka Ringwald     uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information;
109378d08d09SMilanka Ringwald     uint8_t allocation_method_bitmap = (media_codec[1] & 0x03) & remote_allocation_method_bitmap;
109478d08d09SMilanka Ringwald 
109578d08d09SMilanka Ringwald     uint8_t allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS;
109678d08d09SMilanka Ringwald     if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS){
109778d08d09SMilanka Ringwald         allocation_method = AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS;
109878d08d09SMilanka Ringwald     } else if (allocation_method_bitmap & AVDTP_SBC_ALLOCATION_METHOD_SNR){
109978d08d09SMilanka Ringwald         allocation_method = AVDTP_SBC_ALLOCATION_METHOD_SNR;
110078d08d09SMilanka Ringwald     }
110178d08d09SMilanka Ringwald     return allocation_method;
110278d08d09SMilanka Ringwald }
110378d08d09SMilanka Ringwald 
1104bd1ecb8aSMilanka Ringwald uint8_t avdtp_stream_endpoint_seid(avdtp_stream_endpoint_t * stream_endpoint){
1105bd1ecb8aSMilanka Ringwald     if (!stream_endpoint) return 0;
1106bd1ecb8aSMilanka Ringwald     return stream_endpoint->sep.seid;
1107bd1ecb8aSMilanka Ringwald }
110878d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_subbands(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_subbands_bitmap){
110967ae582dSMilanka Ringwald     if (!stream_endpoint) return 0;
111078d08d09SMilanka Ringwald     uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information;
111178d08d09SMilanka Ringwald     uint8_t subbands_bitmap = ((media_codec[1] >> 2) & 0x03) & remote_subbands_bitmap;
111278d08d09SMilanka Ringwald 
111378d08d09SMilanka Ringwald     uint8_t subbands = AVDTP_SBC_SUBBANDS_8;
111478d08d09SMilanka Ringwald     if (subbands_bitmap & AVDTP_SBC_SUBBANDS_8){
111578d08d09SMilanka Ringwald         subbands = AVDTP_SBC_SUBBANDS_8;
111678d08d09SMilanka Ringwald     } else if (subbands_bitmap & AVDTP_SBC_SUBBANDS_4){
111778d08d09SMilanka Ringwald         subbands = AVDTP_SBC_SUBBANDS_4;
111878d08d09SMilanka Ringwald     }
111978d08d09SMilanka Ringwald     return subbands;
112078d08d09SMilanka Ringwald }
112178d08d09SMilanka Ringwald 
112278d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_block_length(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_block_length_bitmap){
112367ae582dSMilanka Ringwald     if (!stream_endpoint) return 0;
112478d08d09SMilanka Ringwald     uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information;
112578d08d09SMilanka Ringwald     uint8_t block_length_bitmap = (media_codec[1] >> 4) & remote_block_length_bitmap;
112678d08d09SMilanka Ringwald 
112778d08d09SMilanka Ringwald     uint8_t block_length = AVDTP_SBC_BLOCK_LENGTH_16;
112878d08d09SMilanka Ringwald     if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_16){
112978d08d09SMilanka Ringwald         block_length = AVDTP_SBC_BLOCK_LENGTH_16;
113078d08d09SMilanka Ringwald     } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_12){
113178d08d09SMilanka Ringwald         block_length = AVDTP_SBC_BLOCK_LENGTH_12;
113278d08d09SMilanka Ringwald     } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_8){
113378d08d09SMilanka Ringwald         block_length = AVDTP_SBC_BLOCK_LENGTH_8;
113478d08d09SMilanka Ringwald     } else if (block_length_bitmap & AVDTP_SBC_BLOCK_LENGTH_4){
113578d08d09SMilanka Ringwald         block_length = AVDTP_SBC_BLOCK_LENGTH_4;
113678d08d09SMilanka Ringwald     }
113778d08d09SMilanka Ringwald     return block_length;
113878d08d09SMilanka Ringwald }
113978d08d09SMilanka Ringwald 
114078d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_sampling_frequency(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_sampling_frequency_bitmap){
114167ae582dSMilanka Ringwald     if (!stream_endpoint) return 0;
114278d08d09SMilanka Ringwald     uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information;
11438e7044f9SMatthias Ringwald     uint8_t supported_sampling_frequency_bitmap = (media_codec[0] >> 4) & remote_sampling_frequency_bitmap;
11448e7044f9SMatthias Ringwald     uint8_t sampling_frequency = AVDTP_SBC_44100;   // some default
114578d08d09SMilanka Ringwald 
11468e7044f9SMatthias Ringwald     // use preferred sampling frequency if possible
11478e7044f9SMatthias Ringwald     if        ((stream_endpoint->preferred_sampling_frequency == 48000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_48000)){
114878d08d09SMilanka Ringwald         sampling_frequency = AVDTP_SBC_48000;
11498e7044f9SMatthias Ringwald     } else if ((stream_endpoint->preferred_sampling_frequency == 44100) && (supported_sampling_frequency_bitmap & AVDTP_SBC_44100)){
115078d08d09SMilanka Ringwald         sampling_frequency = AVDTP_SBC_44100;
11518e7044f9SMatthias Ringwald     } else if ((stream_endpoint->preferred_sampling_frequency == 32000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_32000)){
115278d08d09SMilanka Ringwald         sampling_frequency = AVDTP_SBC_32000;
11538e7044f9SMatthias Ringwald     } else if ((stream_endpoint->preferred_sampling_frequency == 16000) && (supported_sampling_frequency_bitmap & AVDTP_SBC_16000)){
11548e7044f9SMatthias Ringwald         sampling_frequency = AVDTP_SBC_16000;
11558e7044f9SMatthias Ringwald     }
11568e7044f9SMatthias Ringwald     // otherwise, use highest available
11578e7044f9SMatthias Ringwald     else if (supported_sampling_frequency_bitmap & AVDTP_SBC_48000){
11588e7044f9SMatthias Ringwald         sampling_frequency = AVDTP_SBC_48000;
11598e7044f9SMatthias Ringwald     } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_44100){
11608e7044f9SMatthias Ringwald         sampling_frequency = AVDTP_SBC_44100;
11618e7044f9SMatthias Ringwald     } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_32000){
11628e7044f9SMatthias Ringwald         sampling_frequency = AVDTP_SBC_32000;
11638e7044f9SMatthias Ringwald     } else if (supported_sampling_frequency_bitmap & AVDTP_SBC_16000){
116478d08d09SMilanka Ringwald         sampling_frequency = AVDTP_SBC_16000;
116578d08d09SMilanka Ringwald     }
116678d08d09SMilanka Ringwald     return sampling_frequency;
116778d08d09SMilanka Ringwald }
116878d08d09SMilanka Ringwald 
116978d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_max_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_max_bitpool_value){
117067ae582dSMilanka Ringwald     if (!stream_endpoint) return 0;
117178d08d09SMilanka Ringwald     uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information;
117278d08d09SMilanka Ringwald     return btstack_min(media_codec[3], remote_max_bitpool_value);
117378d08d09SMilanka Ringwald }
117478d08d09SMilanka Ringwald 
117578d08d09SMilanka Ringwald uint8_t avdtp_choose_sbc_min_bitpool_value(avdtp_stream_endpoint_t * stream_endpoint, uint8_t remote_min_bitpool_value){
117667ae582dSMilanka Ringwald     if (!stream_endpoint) return 0;
117778d08d09SMilanka Ringwald     uint8_t * media_codec = stream_endpoint->sep.capabilities.media_codec.media_codec_information;
117878d08d09SMilanka Ringwald     return btstack_max(media_codec[2], remote_min_bitpool_value);
1179747ec646SMilanka Ringwald }
1180485c0a4cSMilanka Ringwald 
1181485c0a4cSMilanka Ringwald uint8_t is_avdtp_remote_seid_registered(avdtp_stream_endpoint_t * stream_endpoint){
1182485c0a4cSMilanka Ringwald     if (!stream_endpoint) return 0;
1183485c0a4cSMilanka Ringwald     if (stream_endpoint->remote_sep.seid == 0) return 0;
1184485c0a4cSMilanka Ringwald     if (stream_endpoint->remote_sep.seid > 0x3E) return 0;
1185485c0a4cSMilanka Ringwald     return 1;
1186485c0a4cSMilanka Ringwald }
1187