xref: /btstack/src/classic/avrcp_browsing.c (revision 64a27ec54bba6691cf14c86688f97ba7ca3f1f14)
1665a00cbSMilanka Ringwald /*
2665a00cbSMilanka Ringwald  * Copyright (C) 2016 BlueKitchen GmbH
3665a00cbSMilanka Ringwald  *
4665a00cbSMilanka Ringwald  * Redistribution and use in source and binary forms, with or without
5665a00cbSMilanka Ringwald  * modification, are permitted provided that the following conditions
6665a00cbSMilanka Ringwald  * are met:
7665a00cbSMilanka Ringwald  *
8665a00cbSMilanka Ringwald  * 1. Redistributions of source code must retain the above copyright
9665a00cbSMilanka Ringwald  *    notice, this list of conditions and the following disclaimer.
10665a00cbSMilanka Ringwald  * 2. Redistributions in binary form must reproduce the above copyright
11665a00cbSMilanka Ringwald  *    notice, this list of conditions and the following disclaimer in the
12665a00cbSMilanka Ringwald  *    documentation and/or other materials provided with the distribution.
13665a00cbSMilanka Ringwald  * 3. Neither the name of the copyright holders nor the names of
14665a00cbSMilanka Ringwald  *    contributors may be used to endorse or promote products derived
15665a00cbSMilanka Ringwald  *    from this software without specific prior written permission.
16665a00cbSMilanka Ringwald  * 4. Any redistribution, use, or modification is done solely for
17665a00cbSMilanka Ringwald  *    personal benefit and not for any commercial purpose or for
18665a00cbSMilanka Ringwald  *    monetary gain.
19665a00cbSMilanka Ringwald  *
20665a00cbSMilanka Ringwald  * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
21665a00cbSMilanka Ringwald  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22665a00cbSMilanka Ringwald  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23665a00cbSMilanka Ringwald  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
24665a00cbSMilanka Ringwald  * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25665a00cbSMilanka Ringwald  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26665a00cbSMilanka Ringwald  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27665a00cbSMilanka Ringwald  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28665a00cbSMilanka Ringwald  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29665a00cbSMilanka Ringwald  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30665a00cbSMilanka Ringwald  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31665a00cbSMilanka Ringwald  * SUCH DAMAGE.
32665a00cbSMilanka Ringwald  *
33665a00cbSMilanka Ringwald  * Please inquire about commercial licensing options at
34665a00cbSMilanka Ringwald  * [email protected]
35665a00cbSMilanka Ringwald  *
36665a00cbSMilanka Ringwald  */
37665a00cbSMilanka Ringwald 
38665a00cbSMilanka Ringwald #define BTSTACK_FILE__ "avrcp_browsing.c"
39665a00cbSMilanka Ringwald 
40665a00cbSMilanka Ringwald #include <stdint.h>
41665a00cbSMilanka Ringwald #include <string.h>
42665a00cbSMilanka Ringwald 
43665a00cbSMilanka Ringwald #include "bluetooth_psm.h"
44665a00cbSMilanka Ringwald #include "bluetooth_sdp.h"
45665a00cbSMilanka Ringwald #include "btstack_debug.h"
46665a00cbSMilanka Ringwald #include "btstack_event.h"
47665a00cbSMilanka Ringwald #include "btstack_memory.h"
48665a00cbSMilanka Ringwald #include "classic/sdp_client.h"
49665a00cbSMilanka Ringwald #include "classic/sdp_util.h"
50665a00cbSMilanka Ringwald #include "classic/avrcp_browsing.h"
51665a00cbSMilanka Ringwald 
52*64a27ec5SMilanka Ringwald typedef struct {
53*64a27ec5SMilanka Ringwald     uint16_t browsing_cid;
54*64a27ec5SMilanka Ringwald     uint16_t browsing_l2cap_psm;
55*64a27ec5SMilanka Ringwald     uint16_t browsing_version;
56*64a27ec5SMilanka Ringwald } avrcp_browsing_sdp_query_context_t;
57665a00cbSMilanka Ringwald 
58665a00cbSMilanka Ringwald static void avrcp_browsing_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
59665a00cbSMilanka Ringwald 
60665a00cbSMilanka Ringwald static btstack_packet_handler_t avrcp_browsing_callback;
61*64a27ec5SMilanka Ringwald static avrcp_browsing_sdp_query_context_t sdp_query_context;
62665a00cbSMilanka Ringwald 
63665a00cbSMilanka Ringwald static bool l2cap_browsing_service_registered = false;
64665a00cbSMilanka Ringwald static btstack_packet_handler_t avrcp_browsing_controller_packet_handler;
65665a00cbSMilanka Ringwald static btstack_packet_handler_t avrcp_browsing_target_packet_handler;
66*64a27ec5SMilanka Ringwald static btstack_context_callback_registration_t avrcp_browsing_handle_sdp_client_query_request;
67665a00cbSMilanka Ringwald 
68e71114b0SMatthias Ringwald static bd_addr_t avrcp_browsing_sdp_addr;
69665a00cbSMilanka Ringwald 
70665a00cbSMilanka Ringwald void avrcp_browsing_request_can_send_now(avrcp_browsing_connection_t * connection, uint16_t l2cap_cid){
71665a00cbSMilanka Ringwald     connection->wait_to_send = true;
72665a00cbSMilanka Ringwald     l2cap_request_can_send_now_event(l2cap_cid);
73665a00cbSMilanka Ringwald }
74665a00cbSMilanka Ringwald 
7537fae987SMilanka Ringwald static void avrcp_retry_timer_timeout_handler(btstack_timer_source_t * timer){
76665a00cbSMilanka Ringwald     uint16_t avrcp_cid = (uint16_t)(uintptr_t) btstack_run_loop_get_timer_context(timer);
771945fe3eSMilanka Ringwald     avrcp_connection_t * connection_controller = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_CONTROLLER, avrcp_cid);
78665a00cbSMilanka Ringwald     if (connection_controller == NULL) return;
791945fe3eSMilanka Ringwald     avrcp_connection_t * connection_target = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid);
80665a00cbSMilanka Ringwald     if (connection_target == NULL) return;
81665a00cbSMilanka Ringwald 
82665a00cbSMilanka Ringwald     if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)) return;
83665a00cbSMilanka Ringwald 
8437fae987SMilanka Ringwald     if (connection_controller->browsing_connection->state == AVCTP_CONNECTION_W2_L2CAP_RETRY){
85665a00cbSMilanka Ringwald         connection_controller->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
86665a00cbSMilanka Ringwald         connection_target->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
87665a00cbSMilanka Ringwald 
88665a00cbSMilanka Ringwald         l2cap_create_ertm_channel(avrcp_browsing_packet_handler, connection_controller->remote_addr, connection_controller->browsing_l2cap_psm,
89665a00cbSMilanka Ringwald                 &connection_controller->browsing_connection->ertm_config,
90665a00cbSMilanka Ringwald                 connection_controller->browsing_connection->ertm_buffer,
91665a00cbSMilanka Ringwald                 connection_controller->browsing_connection->ertm_buffer_size, NULL);
92665a00cbSMilanka Ringwald     }
93665a00cbSMilanka Ringwald }
94665a00cbSMilanka Ringwald 
9537fae987SMilanka Ringwald static void avrcp_retry_timer_start(avrcp_connection_t * connection){
9637fae987SMilanka Ringwald     btstack_run_loop_set_timer_handler(&connection->retry_timer, avrcp_retry_timer_timeout_handler);
9737fae987SMilanka Ringwald     btstack_run_loop_set_timer_context(&connection->retry_timer, (void *)(uintptr_t)connection->avrcp_cid);
98665a00cbSMilanka Ringwald 
99665a00cbSMilanka Ringwald     // add some jitter/randomness to reconnect delay
100665a00cbSMilanka Ringwald     uint32_t timeout = 100 + (btstack_run_loop_get_time_ms() & 0x7F);
10137fae987SMilanka Ringwald     btstack_run_loop_set_timer(&connection->retry_timer, timeout);
102665a00cbSMilanka Ringwald 
10337fae987SMilanka Ringwald     btstack_run_loop_add_timer(&connection->retry_timer);
104665a00cbSMilanka Ringwald }
105665a00cbSMilanka Ringwald 
106665a00cbSMilanka Ringwald // AVRCP Browsing Service functions
107665a00cbSMilanka Ringwald static void avrcp_browsing_finalize_connection(avrcp_connection_t * connection){
10837fae987SMilanka Ringwald     btstack_run_loop_remove_timer(&connection->retry_timer);
109665a00cbSMilanka Ringwald     btstack_memory_avrcp_browsing_connection_free(connection->browsing_connection);
110665a00cbSMilanka Ringwald     connection->browsing_connection = NULL;
111665a00cbSMilanka Ringwald }
112665a00cbSMilanka Ringwald 
113665a00cbSMilanka Ringwald static void avrcp_browsing_emit_connection_established(uint16_t browsing_cid, bd_addr_t addr, uint8_t status){
114665a00cbSMilanka Ringwald     btstack_assert(avrcp_browsing_callback != NULL);
115665a00cbSMilanka Ringwald 
116665a00cbSMilanka Ringwald     uint8_t event[12];
117665a00cbSMilanka Ringwald     int pos = 0;
118665a00cbSMilanka Ringwald     event[pos++] = HCI_EVENT_AVRCP_META;
119665a00cbSMilanka Ringwald     event[pos++] = sizeof(event) - 2;
120665a00cbSMilanka Ringwald     event[pos++] = AVRCP_SUBEVENT_BROWSING_CONNECTION_ESTABLISHED;
121665a00cbSMilanka Ringwald     event[pos++] = status;
122665a00cbSMilanka Ringwald     reverse_bd_addr(addr,&event[pos]);
123665a00cbSMilanka Ringwald     pos += 6;
124665a00cbSMilanka Ringwald     little_endian_store_16(event, pos, browsing_cid);
125665a00cbSMilanka Ringwald     pos += 2;
126665a00cbSMilanka Ringwald     (*avrcp_browsing_callback)(HCI_EVENT_PACKET, 0, event, sizeof(event));
127665a00cbSMilanka Ringwald }
128665a00cbSMilanka Ringwald 
129665a00cbSMilanka Ringwald static void avrcp_browsing_emit_incoming_connection(uint16_t browsing_cid, bd_addr_t addr){
130665a00cbSMilanka Ringwald     btstack_assert(avrcp_browsing_callback != NULL);
131665a00cbSMilanka Ringwald 
132665a00cbSMilanka Ringwald     uint8_t event[11];
133665a00cbSMilanka Ringwald     int pos = 0;
134665a00cbSMilanka Ringwald     event[pos++] = HCI_EVENT_AVRCP_META;
135665a00cbSMilanka Ringwald     event[pos++] = sizeof(event) - 2;
136665a00cbSMilanka Ringwald     event[pos++] = AVRCP_SUBEVENT_INCOMING_BROWSING_CONNECTION;
137665a00cbSMilanka Ringwald     reverse_bd_addr(addr,&event[pos]);
138665a00cbSMilanka Ringwald     pos += 6;
139665a00cbSMilanka Ringwald     little_endian_store_16(event, pos, browsing_cid);
140665a00cbSMilanka Ringwald     pos += 2;
141665a00cbSMilanka Ringwald     (*avrcp_browsing_callback)(HCI_EVENT_PACKET, 0, event, sizeof(event));
142665a00cbSMilanka Ringwald }
143665a00cbSMilanka Ringwald 
144665a00cbSMilanka Ringwald static void avrcp_browsing_emit_connection_closed(uint16_t browsing_cid){
145665a00cbSMilanka Ringwald     btstack_assert(avrcp_browsing_callback != NULL);
146665a00cbSMilanka Ringwald 
147665a00cbSMilanka Ringwald     uint8_t event[5];
148665a00cbSMilanka Ringwald     int pos = 0;
149665a00cbSMilanka Ringwald     event[pos++] = HCI_EVENT_AVRCP_META;
150665a00cbSMilanka Ringwald     event[pos++] = sizeof(event) - 2;
151665a00cbSMilanka Ringwald     event[pos++] = AVRCP_SUBEVENT_BROWSING_CONNECTION_RELEASED;
152665a00cbSMilanka Ringwald     little_endian_store_16(event, pos, browsing_cid);
153665a00cbSMilanka Ringwald     pos += 2;
154665a00cbSMilanka Ringwald     (*avrcp_browsing_callback)(HCI_EVENT_PACKET, 0, event, sizeof(event));
155665a00cbSMilanka Ringwald }
156665a00cbSMilanka Ringwald 
157665a00cbSMilanka Ringwald 
158665a00cbSMilanka Ringwald static avrcp_browsing_connection_t * avrcp_browsing_create_connection(avrcp_connection_t * avrcp_connection, uint16_t avrcp_browsing_cid){
159665a00cbSMilanka Ringwald     avrcp_browsing_connection_t * browsing_connection = btstack_memory_avrcp_browsing_connection_get();
160665a00cbSMilanka Ringwald     if (!browsing_connection){
161665a00cbSMilanka Ringwald         log_error("Not enough memory to create browsing connection");
162665a00cbSMilanka Ringwald         return NULL;
163665a00cbSMilanka Ringwald     }
164665a00cbSMilanka Ringwald     browsing_connection->state = AVCTP_CONNECTION_IDLE;
165665a00cbSMilanka Ringwald     browsing_connection->transaction_label = 0xFF;
166665a00cbSMilanka Ringwald 
167665a00cbSMilanka Ringwald     avrcp_connection->avrcp_browsing_cid = avrcp_browsing_cid;
168665a00cbSMilanka Ringwald     avrcp_connection->browsing_connection = browsing_connection;
169665a00cbSMilanka Ringwald 
170665a00cbSMilanka Ringwald     log_info("avrcp_browsing_create_connection, avrcp cid 0x%02x", avrcp_connection->avrcp_browsing_cid);
171665a00cbSMilanka Ringwald     return browsing_connection;
172665a00cbSMilanka Ringwald }
173665a00cbSMilanka Ringwald 
174665a00cbSMilanka Ringwald static void avrcp_browsing_configure_ertm(avrcp_browsing_connection_t * browsing_connection, uint8_t * ertm_buffer, uint32_t ertm_buffer_size, l2cap_ertm_config_t * ertm_config){
175665a00cbSMilanka Ringwald     browsing_connection->ertm_buffer = ertm_buffer;
176665a00cbSMilanka Ringwald     browsing_connection->ertm_buffer_size = ertm_buffer_size;
177665a00cbSMilanka Ringwald 
178665a00cbSMilanka Ringwald     if (ertm_buffer_size > 0) {
179665a00cbSMilanka Ringwald         (void)memcpy(&browsing_connection->ertm_config, ertm_config,
180665a00cbSMilanka Ringwald                  sizeof(l2cap_ertm_config_t));
181665a00cbSMilanka Ringwald         log_info("avrcp_browsing_configure_ertm");
182665a00cbSMilanka Ringwald     }
183665a00cbSMilanka Ringwald }
184665a00cbSMilanka Ringwald 
185665a00cbSMilanka Ringwald static avrcp_browsing_connection_t * avrcp_browsing_handle_incoming_connection(avrcp_connection_t * connection, uint16_t local_cid, uint16_t avrcp_browsing_cid){
186665a00cbSMilanka Ringwald     if (connection->browsing_connection == NULL){
187665a00cbSMilanka Ringwald         avrcp_browsing_create_connection(connection, avrcp_browsing_cid);
188665a00cbSMilanka Ringwald     }
189665a00cbSMilanka Ringwald     if (connection->browsing_connection) {
190665a00cbSMilanka Ringwald         connection->browsing_connection->l2cap_browsing_cid = local_cid;
191665a00cbSMilanka Ringwald         connection->browsing_connection->state = AVCTP_CONNECTION_W4_ERTM_CONFIGURATION;
19237fae987SMilanka Ringwald         btstack_run_loop_remove_timer(&connection->retry_timer);
193665a00cbSMilanka Ringwald     }
194665a00cbSMilanka Ringwald     return connection->browsing_connection;
195665a00cbSMilanka Ringwald }
196665a00cbSMilanka Ringwald 
197665a00cbSMilanka Ringwald static void avrcp_browsing_handle_open_connection_for_role(avrcp_connection_t * connection, uint16_t local_cid){
198665a00cbSMilanka Ringwald     connection->browsing_connection->l2cap_browsing_cid = local_cid;
199665a00cbSMilanka Ringwald     connection->browsing_connection->incoming_declined = false;
200665a00cbSMilanka Ringwald     connection->browsing_connection->state = AVCTP_CONNECTION_OPENED;
201665a00cbSMilanka Ringwald     log_info("L2CAP_EVENT_CHANNEL_OPENED browsing_avrcp_cid 0x%02x, l2cap_signaling_cid 0x%02x, role %d", connection->avrcp_cid, connection->l2cap_signaling_cid, connection->role);
202665a00cbSMilanka Ringwald }
203665a00cbSMilanka Ringwald 
204665a00cbSMilanka Ringwald static avrcp_frame_type_t avrcp_get_frame_type(uint8_t header){
205665a00cbSMilanka Ringwald     return (avrcp_frame_type_t)((header & 0x02) >> 1);
206665a00cbSMilanka Ringwald }
207665a00cbSMilanka Ringwald 
208665a00cbSMilanka Ringwald static void avrcp_browsing_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
209665a00cbSMilanka Ringwald     UNUSED(channel);
210665a00cbSMilanka Ringwald     UNUSED(size);
211665a00cbSMilanka Ringwald     bd_addr_t event_addr;
212665a00cbSMilanka Ringwald     uint16_t local_cid;
213665a00cbSMilanka Ringwald     uint8_t  status;
214665a00cbSMilanka Ringwald     bool decline_connection;
215665a00cbSMilanka Ringwald     bool outoing_active;
216665a00cbSMilanka Ringwald 
217665a00cbSMilanka Ringwald     avrcp_connection_t * connection_controller;
218665a00cbSMilanka Ringwald     avrcp_connection_t * connection_target;
219665a00cbSMilanka Ringwald 
220665a00cbSMilanka Ringwald     switch (packet_type){
221665a00cbSMilanka Ringwald         case L2CAP_DATA_PACKET:
222665a00cbSMilanka Ringwald             switch (avrcp_get_frame_type(packet[0])){
223665a00cbSMilanka Ringwald                 case AVRCP_RESPONSE_FRAME:
224665a00cbSMilanka Ringwald                     (*avrcp_browsing_controller_packet_handler)(packet_type, channel, packet, size);
225665a00cbSMilanka Ringwald                     break;
226665a00cbSMilanka Ringwald                 case AVRCP_COMMAND_FRAME:
227665a00cbSMilanka Ringwald                 default:    // make compiler happy
228665a00cbSMilanka Ringwald                     (*avrcp_browsing_target_packet_handler)(packet_type, channel, packet, size);
229665a00cbSMilanka Ringwald                     break;
230665a00cbSMilanka Ringwald             }
231665a00cbSMilanka Ringwald             break;
232665a00cbSMilanka Ringwald         case HCI_EVENT_PACKET:
233665a00cbSMilanka Ringwald             btstack_assert(avrcp_browsing_controller_packet_handler != NULL);
234665a00cbSMilanka Ringwald             btstack_assert(avrcp_browsing_target_packet_handler != NULL);
235665a00cbSMilanka Ringwald 
236665a00cbSMilanka Ringwald             switch (hci_event_packet_get_type(packet)) {
237665a00cbSMilanka Ringwald 
238665a00cbSMilanka Ringwald                 case L2CAP_EVENT_INCOMING_CONNECTION:
239665a00cbSMilanka Ringwald                     btstack_assert(avrcp_browsing_controller_packet_handler != NULL);
240665a00cbSMilanka Ringwald                     btstack_assert(avrcp_browsing_target_packet_handler != NULL);
241665a00cbSMilanka Ringwald 
242665a00cbSMilanka Ringwald                     l2cap_event_incoming_connection_get_address(packet, event_addr);
243665a00cbSMilanka Ringwald                     local_cid = l2cap_event_incoming_connection_get_local_cid(packet);
244665a00cbSMilanka Ringwald                     outoing_active = false;
245665a00cbSMilanka Ringwald 
2461945fe3eSMilanka Ringwald                     connection_target = avrcp_get_connection_for_bd_addr_for_role(AVRCP_TARGET, event_addr);
2471945fe3eSMilanka Ringwald                     connection_controller = avrcp_get_connection_for_bd_addr_for_role(AVRCP_CONTROLLER, event_addr);
248665a00cbSMilanka Ringwald 
249665a00cbSMilanka Ringwald                     if (connection_target == NULL || connection_controller == NULL) {
250665a00cbSMilanka Ringwald                         l2cap_decline_connection(local_cid);
251665a00cbSMilanka Ringwald                         return;
252665a00cbSMilanka Ringwald                     }
253665a00cbSMilanka Ringwald 
254665a00cbSMilanka Ringwald                     if (connection_target->browsing_connection != NULL){
255665a00cbSMilanka Ringwald                         if (connection_target->browsing_connection->state == AVCTP_CONNECTION_W4_L2CAP_CONNECTED){
256665a00cbSMilanka Ringwald                             outoing_active = true;
257665a00cbSMilanka Ringwald                             connection_target->browsing_connection->incoming_declined = true;
258665a00cbSMilanka Ringwald                         }
259665a00cbSMilanka Ringwald                     }
260665a00cbSMilanka Ringwald 
261665a00cbSMilanka Ringwald                     if (connection_controller->browsing_connection != NULL){
262665a00cbSMilanka Ringwald                         if (connection_controller->browsing_connection->state == AVCTP_CONNECTION_W4_L2CAP_CONNECTED) {
263665a00cbSMilanka Ringwald                             outoing_active = true;
264665a00cbSMilanka Ringwald                             connection_controller->browsing_connection->incoming_declined = true;
265665a00cbSMilanka Ringwald                         }
266665a00cbSMilanka Ringwald                     }
267665a00cbSMilanka Ringwald 
268665a00cbSMilanka Ringwald                     decline_connection = outoing_active;
269665a00cbSMilanka Ringwald                     if (decline_connection == false){
270665a00cbSMilanka Ringwald                         uint16_t avrcp_browsing_cid;
271665a00cbSMilanka Ringwald                         if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)){
272665a00cbSMilanka Ringwald                             avrcp_browsing_cid = avrcp_get_next_cid(AVRCP_CONTROLLER);
273665a00cbSMilanka Ringwald                         } else {
274665a00cbSMilanka Ringwald                             avrcp_browsing_cid = connection_controller->avrcp_browsing_cid;
275665a00cbSMilanka Ringwald                         }
276665a00cbSMilanka Ringwald 
277665a00cbSMilanka Ringwald                         // create two connection objects (both)
278665a00cbSMilanka Ringwald                         connection_target->browsing_connection     = avrcp_browsing_handle_incoming_connection(connection_target, local_cid, avrcp_browsing_cid);
279665a00cbSMilanka Ringwald                         connection_controller->browsing_connection = avrcp_browsing_handle_incoming_connection(connection_controller, local_cid, avrcp_browsing_cid);
280665a00cbSMilanka Ringwald 
281665a00cbSMilanka Ringwald                         if ((connection_target->browsing_connection  == NULL) || (connection_controller->browsing_connection == NULL)){
282665a00cbSMilanka Ringwald                             decline_connection = true;
283665a00cbSMilanka Ringwald                             if (connection_target->browsing_connection) {
284665a00cbSMilanka Ringwald                                 avrcp_browsing_finalize_connection(connection_target);
285665a00cbSMilanka Ringwald                             }
286665a00cbSMilanka Ringwald                             if (connection_controller->browsing_connection) {
287665a00cbSMilanka Ringwald                                 avrcp_browsing_finalize_connection(connection_controller);
288665a00cbSMilanka Ringwald                             }
289665a00cbSMilanka Ringwald                         }
290665a00cbSMilanka Ringwald                     }
291665a00cbSMilanka Ringwald                     if (decline_connection){
292665a00cbSMilanka Ringwald                         l2cap_decline_connection(local_cid);
293665a00cbSMilanka Ringwald                     } else {
294665a00cbSMilanka Ringwald                         log_info("AVRCP: L2CAP_EVENT_INCOMING_CONNECTION browsing_avrcp_cid 0x%02x", connection_controller->avrcp_browsing_cid);
295665a00cbSMilanka Ringwald                         avrcp_browsing_emit_incoming_connection(connection_controller->avrcp_browsing_cid, event_addr);
296665a00cbSMilanka Ringwald                     }
297665a00cbSMilanka Ringwald                     break;
298665a00cbSMilanka Ringwald 
299665a00cbSMilanka Ringwald                 case L2CAP_EVENT_CHANNEL_OPENED:
300665a00cbSMilanka Ringwald                     l2cap_event_channel_opened_get_address(packet, event_addr);
301665a00cbSMilanka Ringwald                     status = l2cap_event_channel_opened_get_status(packet);
302665a00cbSMilanka Ringwald                     local_cid = l2cap_event_channel_opened_get_local_cid(packet);
303665a00cbSMilanka Ringwald 
3041945fe3eSMilanka Ringwald                     connection_controller = avrcp_get_connection_for_bd_addr_for_role(AVRCP_CONTROLLER, event_addr);
3051945fe3eSMilanka Ringwald                     connection_target = avrcp_get_connection_for_bd_addr_for_role(AVRCP_TARGET, event_addr);
306665a00cbSMilanka Ringwald 
307665a00cbSMilanka Ringwald                     // incoming: structs are already created in L2CAP_EVENT_INCOMING_CONNECTION
308665a00cbSMilanka Ringwald                     // outgoing: structs are cteated in avrcp_connect() and avrcp_browsing_connect()
309665a00cbSMilanka Ringwald                     if ((connection_controller == NULL) || (connection_target == NULL)) {
310665a00cbSMilanka Ringwald                         break;
311665a00cbSMilanka Ringwald                     }
312665a00cbSMilanka Ringwald                     if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)) {
313665a00cbSMilanka Ringwald                         break;
314665a00cbSMilanka Ringwald                     }
315665a00cbSMilanka Ringwald 
316665a00cbSMilanka Ringwald                     switch (status){
317665a00cbSMilanka Ringwald                         case ERROR_CODE_SUCCESS:
318665a00cbSMilanka Ringwald                             avrcp_browsing_handle_open_connection_for_role(connection_target, local_cid);
319665a00cbSMilanka Ringwald                             avrcp_browsing_handle_open_connection_for_role(connection_controller, local_cid);
320665a00cbSMilanka Ringwald                             avrcp_browsing_emit_connection_established(connection_controller->avrcp_browsing_cid, event_addr, status);
321665a00cbSMilanka Ringwald                             return;
322665a00cbSMilanka Ringwald                         case L2CAP_CONNECTION_RESPONSE_RESULT_REFUSED_RESOURCES:
323665a00cbSMilanka Ringwald                             if (connection_controller->browsing_connection->incoming_declined == true){
324665a00cbSMilanka Ringwald                                 log_info("Incoming browsing connection was declined, and the outgoing failed");
32537fae987SMilanka Ringwald                                 connection_controller->browsing_connection->state = AVCTP_CONNECTION_W2_L2CAP_RETRY;
326665a00cbSMilanka Ringwald                                 connection_controller->browsing_connection->incoming_declined = false;
32737fae987SMilanka Ringwald                                 connection_target->browsing_connection->state = AVCTP_CONNECTION_W2_L2CAP_RETRY;
328665a00cbSMilanka Ringwald                                 connection_target->browsing_connection->incoming_declined = false;
32937fae987SMilanka Ringwald                                 avrcp_retry_timer_start(connection_controller);
330665a00cbSMilanka Ringwald                                 return;
331665a00cbSMilanka Ringwald                             }
332665a00cbSMilanka Ringwald                             break;
333665a00cbSMilanka Ringwald                         default:
334665a00cbSMilanka Ringwald                             break;
335665a00cbSMilanka Ringwald                     }
336665a00cbSMilanka Ringwald                     log_info("L2CAP connection to connection %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status);
337665a00cbSMilanka Ringwald                     avrcp_browsing_emit_connection_established(connection_controller->avrcp_browsing_cid, event_addr, status);
338665a00cbSMilanka Ringwald                     avrcp_browsing_finalize_connection(connection_controller);
339665a00cbSMilanka Ringwald                     avrcp_browsing_finalize_connection(connection_target);
340665a00cbSMilanka Ringwald                     break;
341665a00cbSMilanka Ringwald 
342665a00cbSMilanka Ringwald                 case L2CAP_EVENT_CHANNEL_CLOSED:
343665a00cbSMilanka Ringwald                     local_cid = l2cap_event_channel_closed_get_local_cid(packet);
344665a00cbSMilanka Ringwald 
3451945fe3eSMilanka Ringwald                     connection_controller = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_CONTROLLER, local_cid);
3461945fe3eSMilanka Ringwald                     connection_target = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_TARGET, local_cid);
347665a00cbSMilanka Ringwald                     if ((connection_controller == NULL) || (connection_target == NULL)) {
348665a00cbSMilanka Ringwald                         break;
349665a00cbSMilanka Ringwald                     }
350665a00cbSMilanka Ringwald                     if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)) {
351665a00cbSMilanka Ringwald                         break;
352665a00cbSMilanka Ringwald                     }
353665a00cbSMilanka Ringwald                     avrcp_browsing_emit_connection_closed(connection_controller->avrcp_browsing_cid);
354665a00cbSMilanka Ringwald                     avrcp_browsing_finalize_connection(connection_controller);
355665a00cbSMilanka Ringwald                     avrcp_browsing_finalize_connection(connection_target);
356665a00cbSMilanka Ringwald                     break;
357665a00cbSMilanka Ringwald 
358665a00cbSMilanka Ringwald                 case L2CAP_EVENT_CAN_SEND_NOW:
359665a00cbSMilanka Ringwald                     local_cid = l2cap_event_can_send_now_get_local_cid(packet);
3601945fe3eSMilanka Ringwald                     connection_target = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_TARGET, local_cid);
361665a00cbSMilanka Ringwald                     if ((connection_target != NULL) && (connection_target->browsing_connection != NULL) && connection_target->browsing_connection->wait_to_send) {
362665a00cbSMilanka Ringwald                         connection_target->browsing_connection->wait_to_send = false;
363665a00cbSMilanka Ringwald                         (*avrcp_browsing_target_packet_handler)(HCI_EVENT_PACKET, channel, packet, size);
364665a00cbSMilanka Ringwald                         break;
365665a00cbSMilanka Ringwald                     }
3661945fe3eSMilanka Ringwald                     connection_controller = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_CONTROLLER, local_cid);
367665a00cbSMilanka Ringwald                     if ((connection_controller != NULL) && (connection_controller->browsing_connection != NULL) && connection_controller->browsing_connection->wait_to_send) {
368665a00cbSMilanka Ringwald                         connection_controller->browsing_connection->wait_to_send = false;
369665a00cbSMilanka Ringwald                         (*avrcp_browsing_controller_packet_handler)(HCI_EVENT_PACKET, channel, packet, size);
370665a00cbSMilanka Ringwald                         break;
371665a00cbSMilanka Ringwald                     }
372665a00cbSMilanka Ringwald                     break;
373665a00cbSMilanka Ringwald 
374665a00cbSMilanka Ringwald                 default:
375665a00cbSMilanka Ringwald                     break;
376665a00cbSMilanka Ringwald             }
377665a00cbSMilanka Ringwald             break;
378665a00cbSMilanka Ringwald         default:
379665a00cbSMilanka Ringwald             break;
380665a00cbSMilanka Ringwald     }
381665a00cbSMilanka Ringwald 
382665a00cbSMilanka Ringwald }
383665a00cbSMilanka Ringwald 
384665a00cbSMilanka Ringwald void avrcp_browsing_init(void){
385665a00cbSMilanka Ringwald     if (l2cap_browsing_service_registered) return;
386665a00cbSMilanka Ringwald     int status = l2cap_register_service(&avrcp_browsing_packet_handler, PSM_AVCTP_BROWSING, 0xffff, LEVEL_2);
387665a00cbSMilanka Ringwald 
388665a00cbSMilanka Ringwald     if (status != ERROR_CODE_SUCCESS) return;
389665a00cbSMilanka Ringwald     l2cap_browsing_service_registered = true;
390665a00cbSMilanka Ringwald }
391665a00cbSMilanka Ringwald 
392e71114b0SMatthias Ringwald static void avrcp_browsing_handle_sdp_client_query_result(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
393e71114b0SMatthias Ringwald     UNUSED(packet_type);
394e71114b0SMatthias Ringwald     UNUSED(channel);
395e71114b0SMatthias Ringwald     UNUSED(size);
396e71114b0SMatthias Ringwald 
397*64a27ec5SMilanka Ringwald     avrcp_connection_t * avrcp_target_connection = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_TARGET, sdp_query_context.browsing_cid);
398*64a27ec5SMilanka Ringwald     avrcp_connection_t * avrcp_controller_connection = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, sdp_query_context.browsing_cid);
399*64a27ec5SMilanka Ringwald 
400*64a27ec5SMilanka Ringwald     if ((avrcp_target_connection == NULL) || (avrcp_target_connection->browsing_connection == NULL)) return;
401*64a27ec5SMilanka Ringwald     if (avrcp_target_connection->browsing_connection->state != AVCTP_CONNECTION_W4_SDP_QUERY_COMPLETE) return;
402*64a27ec5SMilanka Ringwald 
403*64a27ec5SMilanka Ringwald     if ((avrcp_controller_connection == NULL) || (avrcp_controller_connection->browsing_connection == NULL)) return;
404*64a27ec5SMilanka Ringwald     if (avrcp_controller_connection->browsing_connection->state != AVCTP_CONNECTION_W4_SDP_QUERY_COMPLETE) return;
405*64a27ec5SMilanka Ringwald 
406e71114b0SMatthias Ringwald 
407e71114b0SMatthias Ringwald     uint8_t status;
408e71114b0SMatthias Ringwald     uint16_t browsing_l2cap_psm;
409e71114b0SMatthias Ringwald 
410e71114b0SMatthias Ringwald     switch (hci_event_packet_get_type(packet)){
411e71114b0SMatthias Ringwald         case SDP_EVENT_QUERY_ATTRIBUTE_VALUE:
412e71114b0SMatthias Ringwald             avrcp_handle_sdp_client_query_attribute_value(packet);
413e71114b0SMatthias Ringwald             break;
414e71114b0SMatthias Ringwald 
415e71114b0SMatthias Ringwald         case SDP_EVENT_QUERY_COMPLETE:
416e71114b0SMatthias Ringwald             status = sdp_event_query_complete_get_status(packet);
417e71114b0SMatthias Ringwald 
418e71114b0SMatthias Ringwald             if (status != ERROR_CODE_SUCCESS){
41994518574SMatthias Ringwald                 avrcp_browsing_emit_connection_established(avrcp_target_connection->avrcp_browsing_cid, avrcp_browsing_sdp_addr, status);
42094518574SMatthias Ringwald                 avrcp_browsing_finalize_connection(avrcp_target_connection);
42194518574SMatthias Ringwald                 avrcp_browsing_finalize_connection(avrcp_controller_connection);
422e71114b0SMatthias Ringwald                 break;
423e71114b0SMatthias Ringwald             }
424e71114b0SMatthias Ringwald 
425e71114b0SMatthias Ringwald             browsing_l2cap_psm = avrcp_sdp_sdp_query_browsing_l2cap_psm();
426e71114b0SMatthias Ringwald             if (!browsing_l2cap_psm){
42794518574SMatthias Ringwald                 avrcp_browsing_emit_connection_established(avrcp_target_connection->avrcp_browsing_cid, avrcp_browsing_sdp_addr, SDP_SERVICE_NOT_FOUND);
42894518574SMatthias Ringwald                 avrcp_browsing_finalize_connection(avrcp_target_connection);
42994518574SMatthias Ringwald                 avrcp_browsing_finalize_connection(avrcp_controller_connection);
430e71114b0SMatthias Ringwald                 break;
431e71114b0SMatthias Ringwald             }
432e71114b0SMatthias Ringwald 
433e71114b0SMatthias Ringwald             l2cap_create_ertm_channel(avrcp_browsing_packet_handler, avrcp_browsing_sdp_addr, browsing_l2cap_psm,
434e71114b0SMatthias Ringwald                                             &avrcp_controller_connection->browsing_connection->ertm_config,
435e71114b0SMatthias Ringwald                                             avrcp_controller_connection->browsing_connection->ertm_buffer,
436e71114b0SMatthias Ringwald                                             avrcp_controller_connection->browsing_connection->ertm_buffer_size, NULL);
437e71114b0SMatthias Ringwald             break;
438e71114b0SMatthias Ringwald 
439e71114b0SMatthias Ringwald         default:
440e71114b0SMatthias Ringwald             break;
441*64a27ec5SMilanka Ringwald     }
442*64a27ec5SMilanka Ringwald     // register the SDP Query request to check if there is another connection waiting for the query
443*64a27ec5SMilanka Ringwald     // ignore ERROR_CODE_COMMAND_DISALLOWED because in that case, we already have requested an SDP callback
444*64a27ec5SMilanka Ringwald     (void) sdp_client_register_query_callback(&avrcp_browsing_handle_sdp_client_query_request);
445*64a27ec5SMilanka Ringwald }
446e71114b0SMatthias Ringwald 
447*64a27ec5SMilanka Ringwald static void avrcp_browsing_handle_start_sdp_client_query(void * context){
448*64a27ec5SMilanka Ringwald     UNUSED(context);
449*64a27ec5SMilanka Ringwald     // TODO
450*64a27ec5SMilanka Ringwald 
451*64a27ec5SMilanka Ringwald     btstack_linked_list_t connections = avrcp_get_connections();
452*64a27ec5SMilanka Ringwald     btstack_linked_list_iterator_t it;
453*64a27ec5SMilanka Ringwald     btstack_linked_list_iterator_init(&it, &connections);
454*64a27ec5SMilanka Ringwald     while (btstack_linked_list_iterator_has_next(&it)){
455*64a27ec5SMilanka Ringwald         avrcp_connection_t * connection = (avrcp_connection_t *)btstack_linked_list_iterator_next(&it);
456*64a27ec5SMilanka Ringwald         if (connection->browsing_connection == NULL) continue;
457*64a27ec5SMilanka Ringwald         if (connection->browsing_connection->state != AVCTP_CONNECTION_W2_SEND_SDP_QUERY) continue;
458*64a27ec5SMilanka Ringwald         connection->browsing_connection->state = AVCTP_CONNECTION_W4_SDP_QUERY_COMPLETE;
459*64a27ec5SMilanka Ringwald 
460*64a27ec5SMilanka Ringwald         // prevent triggering SDP query twice (for each role once)
461*64a27ec5SMilanka Ringwald         avrcp_connection_t * connection_with_opposite_role;
462*64a27ec5SMilanka Ringwald         switch (connection->role){
463*64a27ec5SMilanka Ringwald             case AVRCP_CONTROLLER:
464*64a27ec5SMilanka Ringwald                 connection_with_opposite_role = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, connection->avrcp_cid);
465*64a27ec5SMilanka Ringwald                 break;
466*64a27ec5SMilanka Ringwald             case AVRCP_TARGET:
467*64a27ec5SMilanka Ringwald                 connection_with_opposite_role = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_CONTROLLER, connection->avrcp_cid);
468*64a27ec5SMilanka Ringwald                 break;
469*64a27ec5SMilanka Ringwald             default:
470*64a27ec5SMilanka Ringwald                 btstack_assert(false);
471*64a27ec5SMilanka Ringwald                 return;
472*64a27ec5SMilanka Ringwald         }
473*64a27ec5SMilanka Ringwald         if (connection->browsing_connection != NULL){
474*64a27ec5SMilanka Ringwald             connection_with_opposite_role->browsing_connection->state = AVCTP_CONNECTION_W4_SDP_QUERY_COMPLETE;
475*64a27ec5SMilanka Ringwald         }
476*64a27ec5SMilanka Ringwald 
477*64a27ec5SMilanka Ringwald         sdp_query_context.browsing_l2cap_psm = 0;
478*64a27ec5SMilanka Ringwald         sdp_query_context.browsing_version = 0;
479*64a27ec5SMilanka Ringwald         sdp_query_context.browsing_cid = connection->avrcp_browsing_cid;
480*64a27ec5SMilanka Ringwald 
481*64a27ec5SMilanka Ringwald         sdp_client_query_uuid16(&avrcp_browsing_handle_sdp_client_query_result, (uint8_t *) connection->remote_addr, BLUETOOTH_PROTOCOL_AVCTP);
482*64a27ec5SMilanka Ringwald         return;
483e71114b0SMatthias Ringwald     }
484e71114b0SMatthias Ringwald }
485665a00cbSMilanka Ringwald 
486*64a27ec5SMilanka Ringwald 
487665a00cbSMilanka Ringwald uint8_t avrcp_browsing_connect(bd_addr_t remote_addr, uint8_t * ertm_buffer, uint32_t ertm_buffer_size, l2cap_ertm_config_t * ertm_config, uint16_t * avrcp_browsing_cid){
488665a00cbSMilanka Ringwald     btstack_assert(avrcp_browsing_controller_packet_handler != NULL);
489665a00cbSMilanka Ringwald     btstack_assert(avrcp_browsing_target_packet_handler != NULL);
490665a00cbSMilanka Ringwald 
4911945fe3eSMilanka Ringwald     avrcp_connection_t * connection_controller = avrcp_get_connection_for_bd_addr_for_role(AVRCP_CONTROLLER, remote_addr);
492665a00cbSMilanka Ringwald     if (!connection_controller){
493665a00cbSMilanka Ringwald         return ERROR_CODE_COMMAND_DISALLOWED;
494665a00cbSMilanka Ringwald     }
4951945fe3eSMilanka Ringwald     avrcp_connection_t * connection_target = avrcp_get_connection_for_bd_addr_for_role(AVRCP_TARGET, remote_addr);
496665a00cbSMilanka Ringwald     if (!connection_target){
497665a00cbSMilanka Ringwald         return ERROR_CODE_COMMAND_DISALLOWED;
498665a00cbSMilanka Ringwald     }
499665a00cbSMilanka Ringwald 
500665a00cbSMilanka Ringwald     if (connection_controller->browsing_connection){
501665a00cbSMilanka Ringwald         return ERROR_CODE_COMMAND_DISALLOWED;
502665a00cbSMilanka Ringwald     }
503665a00cbSMilanka Ringwald     if (connection_target->browsing_connection){
504665a00cbSMilanka Ringwald         return ERROR_CODE_COMMAND_DISALLOWED;
505665a00cbSMilanka Ringwald     }
506665a00cbSMilanka Ringwald 
507665a00cbSMilanka Ringwald     uint16_t cid = avrcp_get_next_cid(AVRCP_CONTROLLER);
508665a00cbSMilanka Ringwald 
509665a00cbSMilanka Ringwald     connection_controller->browsing_connection = avrcp_browsing_create_connection(connection_controller, cid);
510665a00cbSMilanka Ringwald     if (!connection_controller->browsing_connection) return BTSTACK_MEMORY_ALLOC_FAILED;
511665a00cbSMilanka Ringwald 
512665a00cbSMilanka Ringwald     connection_target->browsing_connection = avrcp_browsing_create_connection(connection_target, cid);
513665a00cbSMilanka Ringwald     if (!connection_target->browsing_connection){
514665a00cbSMilanka Ringwald         avrcp_browsing_finalize_connection(connection_controller);
515665a00cbSMilanka Ringwald         return BTSTACK_MEMORY_ALLOC_FAILED;
516665a00cbSMilanka Ringwald     }
517665a00cbSMilanka Ringwald     avrcp_browsing_configure_ertm(connection_controller->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
518665a00cbSMilanka Ringwald     avrcp_browsing_configure_ertm(connection_target->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
519665a00cbSMilanka Ringwald 
520665a00cbSMilanka Ringwald     if (avrcp_browsing_cid != NULL){
521665a00cbSMilanka Ringwald         *avrcp_browsing_cid = cid;
522665a00cbSMilanka Ringwald     }
523665a00cbSMilanka Ringwald 
524e71114b0SMatthias Ringwald     if (connection_controller->browsing_l2cap_psm == 0){
525e71114b0SMatthias Ringwald         memcpy(avrcp_browsing_sdp_addr, remote_addr, 6);
526*64a27ec5SMilanka Ringwald         connection_controller->browsing_connection->state = AVCTP_CONNECTION_W2_SEND_SDP_QUERY;
527*64a27ec5SMilanka Ringwald         connection_target->browsing_connection->state     = AVCTP_CONNECTION_W2_SEND_SDP_QUERY;
528*64a27ec5SMilanka Ringwald         avrcp_browsing_handle_sdp_client_query_request.callback = &avrcp_browsing_handle_start_sdp_client_query;
529*64a27ec5SMilanka Ringwald 
530*64a27ec5SMilanka Ringwald         // ignore ERROR_CODE_COMMAND_DISALLOWED because in that case, we already have requested an SDP callback
531*64a27ec5SMilanka Ringwald         (void) sdp_client_register_query_callback(&avrcp_browsing_handle_sdp_client_query_request);
532*64a27ec5SMilanka Ringwald         return ERROR_CODE_SUCCESS;
533e71114b0SMatthias Ringwald     } else {
534665a00cbSMilanka Ringwald         connection_controller->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
535665a00cbSMilanka Ringwald         connection_target->browsing_connection->state     = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
536665a00cbSMilanka Ringwald 
537665a00cbSMilanka Ringwald         return l2cap_create_ertm_channel(avrcp_browsing_packet_handler, remote_addr, connection_controller->browsing_l2cap_psm,
538665a00cbSMilanka Ringwald                                          &connection_controller->browsing_connection->ertm_config,
539665a00cbSMilanka Ringwald                                          connection_controller->browsing_connection->ertm_buffer,
540665a00cbSMilanka Ringwald                                          connection_controller->browsing_connection->ertm_buffer_size, NULL);
541e71114b0SMatthias Ringwald     }
542665a00cbSMilanka Ringwald }
543665a00cbSMilanka Ringwald 
544665a00cbSMilanka Ringwald uint8_t avrcp_browsing_configure_incoming_connection(uint16_t avrcp_browsing_cid, uint8_t * ertm_buffer, uint32_t ertm_buffer_size, l2cap_ertm_config_t * ertm_config){
5451945fe3eSMilanka Ringwald     avrcp_connection_t * connection_controller = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
546665a00cbSMilanka Ringwald     if (!connection_controller){
547665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
548665a00cbSMilanka Ringwald     }
5491945fe3eSMilanka Ringwald     avrcp_connection_t * connection_target = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_TARGET, avrcp_browsing_cid);
550665a00cbSMilanka Ringwald     if (!connection_target){
551665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
552665a00cbSMilanka Ringwald     }
553665a00cbSMilanka Ringwald 
554665a00cbSMilanka Ringwald     if (!connection_controller->browsing_connection){
555665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
556665a00cbSMilanka Ringwald     }
557665a00cbSMilanka Ringwald     if (!connection_target->browsing_connection){
558665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
559665a00cbSMilanka Ringwald     }
560665a00cbSMilanka Ringwald 
561665a00cbSMilanka Ringwald     if (connection_controller->browsing_connection->state != AVCTP_CONNECTION_W4_ERTM_CONFIGURATION){
562665a00cbSMilanka Ringwald         return ERROR_CODE_COMMAND_DISALLOWED;
563665a00cbSMilanka Ringwald     }
564665a00cbSMilanka Ringwald 
565665a00cbSMilanka Ringwald     avrcp_browsing_configure_ertm(connection_controller->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
566665a00cbSMilanka Ringwald     avrcp_browsing_configure_ertm(connection_target->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
567665a00cbSMilanka Ringwald 
568665a00cbSMilanka Ringwald     connection_controller->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
569665a00cbSMilanka Ringwald     connection_target->browsing_connection->state     = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
570665a00cbSMilanka Ringwald 
571665a00cbSMilanka Ringwald     l2cap_accept_ertm_connection(connection_controller->browsing_connection->l2cap_browsing_cid,
572665a00cbSMilanka Ringwald         &connection_controller->browsing_connection->ertm_config,
573665a00cbSMilanka Ringwald         connection_controller->browsing_connection->ertm_buffer,
574665a00cbSMilanka Ringwald         connection_controller->browsing_connection->ertm_buffer_size);
575665a00cbSMilanka Ringwald     return ERROR_CODE_SUCCESS;
576665a00cbSMilanka Ringwald }
577665a00cbSMilanka Ringwald 
578665a00cbSMilanka Ringwald 
579665a00cbSMilanka Ringwald uint8_t avrcp_browsing_decline_incoming_connection(uint16_t avrcp_browsing_cid){
5801945fe3eSMilanka Ringwald     avrcp_connection_t * connection_controller = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
581665a00cbSMilanka Ringwald     if (!connection_controller){
582665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
583665a00cbSMilanka Ringwald     }
5841945fe3eSMilanka Ringwald     avrcp_connection_t * connection_target = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_TARGET, avrcp_browsing_cid);
585665a00cbSMilanka Ringwald     if (!connection_target){
586665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
587665a00cbSMilanka Ringwald     }
588665a00cbSMilanka Ringwald 
589665a00cbSMilanka Ringwald     if (!connection_controller->browsing_connection){
590665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
591665a00cbSMilanka Ringwald     }
592665a00cbSMilanka Ringwald     if (!connection_target->browsing_connection){
593665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
594665a00cbSMilanka Ringwald     }
595665a00cbSMilanka Ringwald 
596665a00cbSMilanka Ringwald     if (connection_controller->browsing_connection->state != AVCTP_CONNECTION_W4_ERTM_CONFIGURATION){
597665a00cbSMilanka Ringwald         return ERROR_CODE_COMMAND_DISALLOWED;
598665a00cbSMilanka Ringwald     }
599665a00cbSMilanka Ringwald 
600665a00cbSMilanka Ringwald     l2cap_decline_connection(connection_controller->browsing_connection->l2cap_browsing_cid);
601665a00cbSMilanka Ringwald 
602665a00cbSMilanka Ringwald     avrcp_browsing_finalize_connection(connection_controller);
603665a00cbSMilanka Ringwald     avrcp_browsing_finalize_connection(connection_target);
604665a00cbSMilanka Ringwald     return ERROR_CODE_SUCCESS;
605665a00cbSMilanka Ringwald }
606665a00cbSMilanka Ringwald 
607665a00cbSMilanka Ringwald uint8_t avrcp_browsing_disconnect(uint16_t avrcp_browsing_cid){
6081945fe3eSMilanka Ringwald     avrcp_connection_t * connection_controller = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
609665a00cbSMilanka Ringwald     if (!connection_controller){
610665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
611665a00cbSMilanka Ringwald     }
6121945fe3eSMilanka Ringwald     avrcp_connection_t * connection_target = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_TARGET, avrcp_browsing_cid);
613665a00cbSMilanka Ringwald     if (!connection_target){
614665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
615665a00cbSMilanka Ringwald     }
616665a00cbSMilanka Ringwald 
617665a00cbSMilanka Ringwald     if (!connection_controller->browsing_connection){
618665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
619665a00cbSMilanka Ringwald     }
620665a00cbSMilanka Ringwald     if (!connection_target->browsing_connection){
621665a00cbSMilanka Ringwald         return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
622665a00cbSMilanka Ringwald     }
623665a00cbSMilanka Ringwald 
624665a00cbSMilanka Ringwald     l2cap_disconnect(connection_controller->browsing_connection->l2cap_browsing_cid, 0);
625665a00cbSMilanka Ringwald     return ERROR_CODE_SUCCESS;
626665a00cbSMilanka Ringwald }
627665a00cbSMilanka Ringwald 
628665a00cbSMilanka Ringwald void avrcp_browsing_register_controller_packet_handler(btstack_packet_handler_t callback){
629665a00cbSMilanka Ringwald     avrcp_browsing_controller_packet_handler = callback;
630665a00cbSMilanka Ringwald }
631665a00cbSMilanka Ringwald 
632665a00cbSMilanka Ringwald void avrcp_browsing_register_target_packet_handler(btstack_packet_handler_t callback){
633665a00cbSMilanka Ringwald     avrcp_browsing_target_packet_handler = callback;
634665a00cbSMilanka Ringwald }
635665a00cbSMilanka Ringwald 
636665a00cbSMilanka Ringwald void avrcp_browsing_register_packet_handler(btstack_packet_handler_t callback){
637665a00cbSMilanka Ringwald     btstack_assert(callback != NULL);
638665a00cbSMilanka Ringwald     avrcp_browsing_callback = callback;
639665a00cbSMilanka Ringwald }
640