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
232fca4dadSMilanka Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BLUEKITCHEN
242fca4dadSMilanka Ringwald * GMBH 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
52665a00cbSMilanka Ringwald static void avrcp_browsing_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
53665a00cbSMilanka Ringwald
5465bd7af5SMatthias Ringwald // higher layer callbacks
55665a00cbSMilanka Ringwald static btstack_packet_handler_t avrcp_browsing_callback;
56665a00cbSMilanka Ringwald static btstack_packet_handler_t avrcp_browsing_controller_packet_handler;
57665a00cbSMilanka Ringwald static btstack_packet_handler_t avrcp_browsing_target_packet_handler;
58665a00cbSMilanka Ringwald
5965bd7af5SMatthias Ringwald // sdp query
60e71114b0SMatthias Ringwald static bd_addr_t avrcp_browsing_sdp_addr;
6165bd7af5SMatthias Ringwald static btstack_context_callback_registration_t avrcp_browsing_handle_sdp_client_query_request;
6265bd7af5SMatthias Ringwald
6365bd7af5SMatthias Ringwald static bool avrcp_browsing_l2cap_service_registered;
6465bd7af5SMatthias Ringwald
65665a00cbSMilanka Ringwald
avrcp_browsing_request_can_send_now(avrcp_browsing_connection_t * connection,uint16_t l2cap_cid)66665a00cbSMilanka Ringwald void avrcp_browsing_request_can_send_now(avrcp_browsing_connection_t * connection, uint16_t l2cap_cid){
67665a00cbSMilanka Ringwald connection->wait_to_send = true;
68665a00cbSMilanka Ringwald l2cap_request_can_send_now_event(l2cap_cid);
69665a00cbSMilanka Ringwald }
70665a00cbSMilanka Ringwald
avrcp_retry_timer_timeout_handler(btstack_timer_source_t * timer)7137fae987SMilanka Ringwald static void avrcp_retry_timer_timeout_handler(btstack_timer_source_t * timer){
72665a00cbSMilanka Ringwald uint16_t avrcp_cid = (uint16_t)(uintptr_t) btstack_run_loop_get_timer_context(timer);
731945fe3eSMilanka Ringwald avrcp_connection_t * connection_controller = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_CONTROLLER, avrcp_cid);
74665a00cbSMilanka Ringwald if (connection_controller == NULL) return;
751945fe3eSMilanka Ringwald avrcp_connection_t * connection_target = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid);
76665a00cbSMilanka Ringwald if (connection_target == NULL) return;
77665a00cbSMilanka Ringwald
78665a00cbSMilanka Ringwald if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)) return;
79665a00cbSMilanka Ringwald
8037fae987SMilanka Ringwald if (connection_controller->browsing_connection->state == AVCTP_CONNECTION_W2_L2CAP_RETRY){
81665a00cbSMilanka Ringwald connection_controller->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
82665a00cbSMilanka Ringwald connection_target->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
83665a00cbSMilanka Ringwald
84f25e60deSMatthias Ringwald l2cap_ertm_create_channel(avrcp_browsing_packet_handler, connection_controller->remote_addr, connection_controller->browsing_l2cap_psm,
85665a00cbSMilanka Ringwald &connection_controller->browsing_connection->ertm_config,
86665a00cbSMilanka Ringwald connection_controller->browsing_connection->ertm_buffer,
87665a00cbSMilanka Ringwald connection_controller->browsing_connection->ertm_buffer_size, NULL);
88665a00cbSMilanka Ringwald }
89665a00cbSMilanka Ringwald }
90665a00cbSMilanka Ringwald
avrcp_retry_timer_start(avrcp_connection_t * connection)9137fae987SMilanka Ringwald static void avrcp_retry_timer_start(avrcp_connection_t * connection){
9237fae987SMilanka Ringwald btstack_run_loop_set_timer_handler(&connection->retry_timer, avrcp_retry_timer_timeout_handler);
9337fae987SMilanka Ringwald btstack_run_loop_set_timer_context(&connection->retry_timer, (void *)(uintptr_t)connection->avrcp_cid);
94665a00cbSMilanka Ringwald
95665a00cbSMilanka Ringwald // add some jitter/randomness to reconnect delay
96665a00cbSMilanka Ringwald uint32_t timeout = 100 + (btstack_run_loop_get_time_ms() & 0x7F);
9737fae987SMilanka Ringwald btstack_run_loop_set_timer(&connection->retry_timer, timeout);
98665a00cbSMilanka Ringwald
9937fae987SMilanka Ringwald btstack_run_loop_add_timer(&connection->retry_timer);
100665a00cbSMilanka Ringwald }
101665a00cbSMilanka Ringwald
102665a00cbSMilanka Ringwald // AVRCP Browsing Service functions
avrcp_browsing_finalize_connection(avrcp_connection_t * connection)103665a00cbSMilanka Ringwald static void avrcp_browsing_finalize_connection(avrcp_connection_t * connection){
10437fae987SMilanka Ringwald btstack_run_loop_remove_timer(&connection->retry_timer);
105665a00cbSMilanka Ringwald btstack_memory_avrcp_browsing_connection_free(connection->browsing_connection);
106665a00cbSMilanka Ringwald connection->browsing_connection = NULL;
107665a00cbSMilanka Ringwald }
108665a00cbSMilanka Ringwald
avrcp_browsing_emit_connection_established(uint16_t browsing_cid,bd_addr_t addr,uint8_t status)109665a00cbSMilanka Ringwald static void avrcp_browsing_emit_connection_established(uint16_t browsing_cid, bd_addr_t addr, uint8_t status){
110665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_callback != NULL);
111665a00cbSMilanka Ringwald
112665a00cbSMilanka Ringwald uint8_t event[12];
113665a00cbSMilanka Ringwald int pos = 0;
114665a00cbSMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META;
115665a00cbSMilanka Ringwald event[pos++] = sizeof(event) - 2;
116665a00cbSMilanka Ringwald event[pos++] = AVRCP_SUBEVENT_BROWSING_CONNECTION_ESTABLISHED;
117665a00cbSMilanka Ringwald event[pos++] = status;
118665a00cbSMilanka Ringwald reverse_bd_addr(addr,&event[pos]);
119665a00cbSMilanka Ringwald pos += 6;
120665a00cbSMilanka Ringwald little_endian_store_16(event, pos, browsing_cid);
121665a00cbSMilanka Ringwald pos += 2;
122665a00cbSMilanka Ringwald (*avrcp_browsing_callback)(HCI_EVENT_PACKET, 0, event, sizeof(event));
123665a00cbSMilanka Ringwald }
124665a00cbSMilanka Ringwald
avrcp_browsing_emit_incoming_connection(uint16_t browsing_cid,bd_addr_t addr)125665a00cbSMilanka Ringwald static void avrcp_browsing_emit_incoming_connection(uint16_t browsing_cid, bd_addr_t addr){
126665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_callback != NULL);
127665a00cbSMilanka Ringwald
128665a00cbSMilanka Ringwald uint8_t event[11];
129665a00cbSMilanka Ringwald int pos = 0;
130665a00cbSMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META;
131665a00cbSMilanka Ringwald event[pos++] = sizeof(event) - 2;
132665a00cbSMilanka Ringwald event[pos++] = AVRCP_SUBEVENT_INCOMING_BROWSING_CONNECTION;
133665a00cbSMilanka Ringwald reverse_bd_addr(addr,&event[pos]);
134665a00cbSMilanka Ringwald pos += 6;
135665a00cbSMilanka Ringwald little_endian_store_16(event, pos, browsing_cid);
136665a00cbSMilanka Ringwald pos += 2;
137665a00cbSMilanka Ringwald (*avrcp_browsing_callback)(HCI_EVENT_PACKET, 0, event, sizeof(event));
138665a00cbSMilanka Ringwald }
139665a00cbSMilanka Ringwald
avrcp_browsing_emit_connection_closed(uint16_t browsing_cid)140665a00cbSMilanka Ringwald static void avrcp_browsing_emit_connection_closed(uint16_t browsing_cid){
141665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_callback != NULL);
142665a00cbSMilanka Ringwald
143665a00cbSMilanka Ringwald uint8_t event[5];
144665a00cbSMilanka Ringwald int pos = 0;
145665a00cbSMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META;
146665a00cbSMilanka Ringwald event[pos++] = sizeof(event) - 2;
147665a00cbSMilanka Ringwald event[pos++] = AVRCP_SUBEVENT_BROWSING_CONNECTION_RELEASED;
148665a00cbSMilanka Ringwald little_endian_store_16(event, pos, browsing_cid);
149665a00cbSMilanka Ringwald pos += 2;
150665a00cbSMilanka Ringwald (*avrcp_browsing_callback)(HCI_EVENT_PACKET, 0, event, sizeof(event));
151665a00cbSMilanka Ringwald }
152665a00cbSMilanka Ringwald
153665a00cbSMilanka Ringwald
avrcp_browsing_create_connection(avrcp_connection_t * avrcp_connection,uint16_t avrcp_browsing_cid)154665a00cbSMilanka Ringwald static avrcp_browsing_connection_t * avrcp_browsing_create_connection(avrcp_connection_t * avrcp_connection, uint16_t avrcp_browsing_cid){
155665a00cbSMilanka Ringwald avrcp_browsing_connection_t * browsing_connection = btstack_memory_avrcp_browsing_connection_get();
156665a00cbSMilanka Ringwald if (!browsing_connection){
157665a00cbSMilanka Ringwald log_error("Not enough memory to create browsing connection");
158665a00cbSMilanka Ringwald return NULL;
159665a00cbSMilanka Ringwald }
160665a00cbSMilanka Ringwald browsing_connection->state = AVCTP_CONNECTION_IDLE;
161665a00cbSMilanka Ringwald browsing_connection->transaction_label = 0xFF;
162665a00cbSMilanka Ringwald
163665a00cbSMilanka Ringwald avrcp_connection->avrcp_browsing_cid = avrcp_browsing_cid;
164665a00cbSMilanka Ringwald avrcp_connection->browsing_connection = browsing_connection;
165665a00cbSMilanka Ringwald
166665a00cbSMilanka Ringwald log_info("avrcp_browsing_create_connection, avrcp cid 0x%02x", avrcp_connection->avrcp_browsing_cid);
167665a00cbSMilanka Ringwald return browsing_connection;
168665a00cbSMilanka Ringwald }
169665a00cbSMilanka Ringwald
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)170665a00cbSMilanka 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){
171665a00cbSMilanka Ringwald browsing_connection->ertm_buffer = ertm_buffer;
172665a00cbSMilanka Ringwald browsing_connection->ertm_buffer_size = ertm_buffer_size;
173665a00cbSMilanka Ringwald
174665a00cbSMilanka Ringwald if (ertm_buffer_size > 0) {
175665a00cbSMilanka Ringwald (void)memcpy(&browsing_connection->ertm_config, ertm_config,
176665a00cbSMilanka Ringwald sizeof(l2cap_ertm_config_t));
177665a00cbSMilanka Ringwald log_info("avrcp_browsing_configure_ertm");
178665a00cbSMilanka Ringwald }
179665a00cbSMilanka Ringwald }
180665a00cbSMilanka Ringwald
avrcp_browsing_handle_incoming_connection(avrcp_connection_t * connection,uint16_t local_cid,uint16_t avrcp_browsing_cid)181665a00cbSMilanka Ringwald static avrcp_browsing_connection_t * avrcp_browsing_handle_incoming_connection(avrcp_connection_t * connection, uint16_t local_cid, uint16_t avrcp_browsing_cid){
182665a00cbSMilanka Ringwald if (connection->browsing_connection == NULL){
183665a00cbSMilanka Ringwald avrcp_browsing_create_connection(connection, avrcp_browsing_cid);
184665a00cbSMilanka Ringwald }
185665a00cbSMilanka Ringwald if (connection->browsing_connection) {
186665a00cbSMilanka Ringwald connection->browsing_connection->l2cap_browsing_cid = local_cid;
187665a00cbSMilanka Ringwald connection->browsing_connection->state = AVCTP_CONNECTION_W4_ERTM_CONFIGURATION;
18837fae987SMilanka Ringwald btstack_run_loop_remove_timer(&connection->retry_timer);
189665a00cbSMilanka Ringwald }
190665a00cbSMilanka Ringwald return connection->browsing_connection;
191665a00cbSMilanka Ringwald }
192665a00cbSMilanka Ringwald
avrcp_browsing_handle_open_connection_for_role(avrcp_connection_t * connection,uint16_t local_cid)193665a00cbSMilanka Ringwald static void avrcp_browsing_handle_open_connection_for_role(avrcp_connection_t * connection, uint16_t local_cid){
194665a00cbSMilanka Ringwald connection->browsing_connection->l2cap_browsing_cid = local_cid;
195665a00cbSMilanka Ringwald connection->browsing_connection->incoming_declined = false;
196665a00cbSMilanka Ringwald connection->browsing_connection->state = AVCTP_CONNECTION_OPENED;
197665a00cbSMilanka 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);
198665a00cbSMilanka Ringwald }
199665a00cbSMilanka Ringwald
avrcp_get_frame_type(uint8_t header)200665a00cbSMilanka Ringwald static avrcp_frame_type_t avrcp_get_frame_type(uint8_t header){
201665a00cbSMilanka Ringwald return (avrcp_frame_type_t)((header & 0x02) >> 1);
202665a00cbSMilanka Ringwald }
203665a00cbSMilanka Ringwald
avrcp_browsing_packet_handler(uint8_t packet_type,uint16_t channel,uint8_t * packet,uint16_t size)204665a00cbSMilanka Ringwald static void avrcp_browsing_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
205665a00cbSMilanka Ringwald UNUSED(channel);
206665a00cbSMilanka Ringwald UNUSED(size);
207665a00cbSMilanka Ringwald bd_addr_t event_addr;
208665a00cbSMilanka Ringwald uint16_t local_cid;
209665a00cbSMilanka Ringwald uint8_t status;
210665a00cbSMilanka Ringwald bool decline_connection;
211665a00cbSMilanka Ringwald bool outoing_active;
212665a00cbSMilanka Ringwald
213665a00cbSMilanka Ringwald avrcp_connection_t * connection_controller;
214665a00cbSMilanka Ringwald avrcp_connection_t * connection_target;
215665a00cbSMilanka Ringwald
216665a00cbSMilanka Ringwald switch (packet_type){
217665a00cbSMilanka Ringwald case L2CAP_DATA_PACKET:
218665a00cbSMilanka Ringwald switch (avrcp_get_frame_type(packet[0])){
219665a00cbSMilanka Ringwald case AVRCP_RESPONSE_FRAME:
220665a00cbSMilanka Ringwald (*avrcp_browsing_controller_packet_handler)(packet_type, channel, packet, size);
221665a00cbSMilanka Ringwald break;
222665a00cbSMilanka Ringwald case AVRCP_COMMAND_FRAME:
223665a00cbSMilanka Ringwald default: // make compiler happy
224665a00cbSMilanka Ringwald (*avrcp_browsing_target_packet_handler)(packet_type, channel, packet, size);
225665a00cbSMilanka Ringwald break;
226665a00cbSMilanka Ringwald }
227665a00cbSMilanka Ringwald break;
228665a00cbSMilanka Ringwald case HCI_EVENT_PACKET:
229665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_controller_packet_handler != NULL);
230665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_target_packet_handler != NULL);
231665a00cbSMilanka Ringwald
232665a00cbSMilanka Ringwald switch (hci_event_packet_get_type(packet)) {
233665a00cbSMilanka Ringwald
234665a00cbSMilanka Ringwald case L2CAP_EVENT_INCOMING_CONNECTION:
235665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_controller_packet_handler != NULL);
236665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_target_packet_handler != NULL);
237665a00cbSMilanka Ringwald
238665a00cbSMilanka Ringwald l2cap_event_incoming_connection_get_address(packet, event_addr);
239665a00cbSMilanka Ringwald local_cid = l2cap_event_incoming_connection_get_local_cid(packet);
240665a00cbSMilanka Ringwald outoing_active = false;
241665a00cbSMilanka Ringwald
2421945fe3eSMilanka Ringwald connection_target = avrcp_get_connection_for_bd_addr_for_role(AVRCP_TARGET, event_addr);
2431945fe3eSMilanka Ringwald connection_controller = avrcp_get_connection_for_bd_addr_for_role(AVRCP_CONTROLLER, event_addr);
244665a00cbSMilanka Ringwald
245665a00cbSMilanka Ringwald if (connection_target == NULL || connection_controller == NULL) {
246665a00cbSMilanka Ringwald l2cap_decline_connection(local_cid);
247665a00cbSMilanka Ringwald return;
248665a00cbSMilanka Ringwald }
249665a00cbSMilanka Ringwald
250665a00cbSMilanka Ringwald if (connection_target->browsing_connection != NULL){
251665a00cbSMilanka Ringwald if (connection_target->browsing_connection->state == AVCTP_CONNECTION_W4_L2CAP_CONNECTED){
252665a00cbSMilanka Ringwald outoing_active = true;
253665a00cbSMilanka Ringwald connection_target->browsing_connection->incoming_declined = true;
254665a00cbSMilanka Ringwald }
255665a00cbSMilanka Ringwald }
256665a00cbSMilanka Ringwald
257665a00cbSMilanka Ringwald if (connection_controller->browsing_connection != NULL){
258665a00cbSMilanka Ringwald if (connection_controller->browsing_connection->state == AVCTP_CONNECTION_W4_L2CAP_CONNECTED) {
259665a00cbSMilanka Ringwald outoing_active = true;
260665a00cbSMilanka Ringwald connection_controller->browsing_connection->incoming_declined = true;
261665a00cbSMilanka Ringwald }
262665a00cbSMilanka Ringwald }
263665a00cbSMilanka Ringwald
264665a00cbSMilanka Ringwald decline_connection = outoing_active;
265665a00cbSMilanka Ringwald if (decline_connection == false){
266665a00cbSMilanka Ringwald uint16_t avrcp_browsing_cid;
267665a00cbSMilanka Ringwald if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)){
268665a00cbSMilanka Ringwald avrcp_browsing_cid = avrcp_get_next_cid(AVRCP_CONTROLLER);
269665a00cbSMilanka Ringwald } else {
270665a00cbSMilanka Ringwald avrcp_browsing_cid = connection_controller->avrcp_browsing_cid;
271665a00cbSMilanka Ringwald }
272665a00cbSMilanka Ringwald
273665a00cbSMilanka Ringwald // create two connection objects (both)
274665a00cbSMilanka Ringwald connection_target->browsing_connection = avrcp_browsing_handle_incoming_connection(connection_target, local_cid, avrcp_browsing_cid);
275665a00cbSMilanka Ringwald connection_controller->browsing_connection = avrcp_browsing_handle_incoming_connection(connection_controller, local_cid, avrcp_browsing_cid);
276665a00cbSMilanka Ringwald
277665a00cbSMilanka Ringwald if ((connection_target->browsing_connection == NULL) || (connection_controller->browsing_connection == NULL)){
278665a00cbSMilanka Ringwald decline_connection = true;
279665a00cbSMilanka Ringwald if (connection_target->browsing_connection) {
280665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_target);
281665a00cbSMilanka Ringwald }
282665a00cbSMilanka Ringwald if (connection_controller->browsing_connection) {
283665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_controller);
284665a00cbSMilanka Ringwald }
285665a00cbSMilanka Ringwald }
286665a00cbSMilanka Ringwald }
287665a00cbSMilanka Ringwald if (decline_connection){
288665a00cbSMilanka Ringwald l2cap_decline_connection(local_cid);
289665a00cbSMilanka Ringwald } else {
290665a00cbSMilanka Ringwald log_info("AVRCP: L2CAP_EVENT_INCOMING_CONNECTION browsing_avrcp_cid 0x%02x", connection_controller->avrcp_browsing_cid);
291665a00cbSMilanka Ringwald avrcp_browsing_emit_incoming_connection(connection_controller->avrcp_browsing_cid, event_addr);
292665a00cbSMilanka Ringwald }
293665a00cbSMilanka Ringwald break;
294665a00cbSMilanka Ringwald
295665a00cbSMilanka Ringwald case L2CAP_EVENT_CHANNEL_OPENED:
296665a00cbSMilanka Ringwald l2cap_event_channel_opened_get_address(packet, event_addr);
297665a00cbSMilanka Ringwald status = l2cap_event_channel_opened_get_status(packet);
298665a00cbSMilanka Ringwald local_cid = l2cap_event_channel_opened_get_local_cid(packet);
299665a00cbSMilanka Ringwald
3001945fe3eSMilanka Ringwald connection_controller = avrcp_get_connection_for_bd_addr_for_role(AVRCP_CONTROLLER, event_addr);
3011945fe3eSMilanka Ringwald connection_target = avrcp_get_connection_for_bd_addr_for_role(AVRCP_TARGET, event_addr);
302665a00cbSMilanka Ringwald
303665a00cbSMilanka Ringwald // incoming: structs are already created in L2CAP_EVENT_INCOMING_CONNECTION
304665a00cbSMilanka Ringwald // outgoing: structs are cteated in avrcp_connect() and avrcp_browsing_connect()
305665a00cbSMilanka Ringwald if ((connection_controller == NULL) || (connection_target == NULL)) {
306665a00cbSMilanka Ringwald break;
307665a00cbSMilanka Ringwald }
308665a00cbSMilanka Ringwald if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)) {
309665a00cbSMilanka Ringwald break;
310665a00cbSMilanka Ringwald }
311665a00cbSMilanka Ringwald
312665a00cbSMilanka Ringwald switch (status){
313665a00cbSMilanka Ringwald case ERROR_CODE_SUCCESS:
314665a00cbSMilanka Ringwald avrcp_browsing_handle_open_connection_for_role(connection_target, local_cid);
315665a00cbSMilanka Ringwald avrcp_browsing_handle_open_connection_for_role(connection_controller, local_cid);
316665a00cbSMilanka Ringwald avrcp_browsing_emit_connection_established(connection_controller->avrcp_browsing_cid, event_addr, status);
317665a00cbSMilanka Ringwald return;
318665a00cbSMilanka Ringwald case L2CAP_CONNECTION_RESPONSE_RESULT_REFUSED_RESOURCES:
319665a00cbSMilanka Ringwald if (connection_controller->browsing_connection->incoming_declined == true){
320665a00cbSMilanka Ringwald log_info("Incoming browsing connection was declined, and the outgoing failed");
32137fae987SMilanka Ringwald connection_controller->browsing_connection->state = AVCTP_CONNECTION_W2_L2CAP_RETRY;
322665a00cbSMilanka Ringwald connection_controller->browsing_connection->incoming_declined = false;
32337fae987SMilanka Ringwald connection_target->browsing_connection->state = AVCTP_CONNECTION_W2_L2CAP_RETRY;
324665a00cbSMilanka Ringwald connection_target->browsing_connection->incoming_declined = false;
32537fae987SMilanka Ringwald avrcp_retry_timer_start(connection_controller);
326665a00cbSMilanka Ringwald return;
327665a00cbSMilanka Ringwald }
328665a00cbSMilanka Ringwald break;
329665a00cbSMilanka Ringwald default:
330665a00cbSMilanka Ringwald break;
331665a00cbSMilanka Ringwald }
332665a00cbSMilanka Ringwald log_info("L2CAP connection to connection %s failed. status code 0x%02x", bd_addr_to_str(event_addr), status);
333665a00cbSMilanka Ringwald avrcp_browsing_emit_connection_established(connection_controller->avrcp_browsing_cid, event_addr, status);
334665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_controller);
335665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_target);
336665a00cbSMilanka Ringwald break;
337665a00cbSMilanka Ringwald
338665a00cbSMilanka Ringwald case L2CAP_EVENT_CHANNEL_CLOSED:
339665a00cbSMilanka Ringwald local_cid = l2cap_event_channel_closed_get_local_cid(packet);
340665a00cbSMilanka Ringwald
3411945fe3eSMilanka Ringwald connection_controller = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_CONTROLLER, local_cid);
3421945fe3eSMilanka Ringwald connection_target = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_TARGET, local_cid);
343665a00cbSMilanka Ringwald if ((connection_controller == NULL) || (connection_target == NULL)) {
344665a00cbSMilanka Ringwald break;
345665a00cbSMilanka Ringwald }
346665a00cbSMilanka Ringwald if ((connection_controller->browsing_connection == NULL) || (connection_target->browsing_connection == NULL)) {
347665a00cbSMilanka Ringwald break;
348665a00cbSMilanka Ringwald }
349665a00cbSMilanka Ringwald avrcp_browsing_emit_connection_closed(connection_controller->avrcp_browsing_cid);
350665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_controller);
351665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_target);
352665a00cbSMilanka Ringwald break;
353665a00cbSMilanka Ringwald
354665a00cbSMilanka Ringwald case L2CAP_EVENT_CAN_SEND_NOW:
355665a00cbSMilanka Ringwald local_cid = l2cap_event_can_send_now_get_local_cid(packet);
3561945fe3eSMilanka Ringwald connection_target = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_TARGET, local_cid);
357665a00cbSMilanka Ringwald if ((connection_target != NULL) && (connection_target->browsing_connection != NULL) && connection_target->browsing_connection->wait_to_send) {
358665a00cbSMilanka Ringwald connection_target->browsing_connection->wait_to_send = false;
359665a00cbSMilanka Ringwald (*avrcp_browsing_target_packet_handler)(HCI_EVENT_PACKET, channel, packet, size);
360665a00cbSMilanka Ringwald break;
361665a00cbSMilanka Ringwald }
3621945fe3eSMilanka Ringwald connection_controller = avrcp_get_connection_for_browsing_l2cap_cid_for_role(AVRCP_CONTROLLER, local_cid);
363665a00cbSMilanka Ringwald if ((connection_controller != NULL) && (connection_controller->browsing_connection != NULL) && connection_controller->browsing_connection->wait_to_send) {
364665a00cbSMilanka Ringwald connection_controller->browsing_connection->wait_to_send = false;
365665a00cbSMilanka Ringwald (*avrcp_browsing_controller_packet_handler)(HCI_EVENT_PACKET, channel, packet, size);
366665a00cbSMilanka Ringwald break;
367665a00cbSMilanka Ringwald }
368665a00cbSMilanka Ringwald break;
369665a00cbSMilanka Ringwald
370665a00cbSMilanka Ringwald default:
371665a00cbSMilanka Ringwald break;
372665a00cbSMilanka Ringwald }
373665a00cbSMilanka Ringwald break;
374665a00cbSMilanka Ringwald default:
375665a00cbSMilanka Ringwald break;
376665a00cbSMilanka Ringwald }
377665a00cbSMilanka Ringwald
378665a00cbSMilanka Ringwald }
379665a00cbSMilanka Ringwald
avrcp_browsing_handle_sdp_query_complete(avrcp_connection_t * connection,uint8_t status)3802f0acaf7SMatthias Ringwald static void avrcp_browsing_handle_sdp_query_complete(avrcp_connection_t * connection, uint8_t status){
381665a00cbSMilanka Ringwald
3822f0acaf7SMatthias Ringwald if (connection->browsing_connection == NULL) {
3832f0acaf7SMatthias Ringwald return;
3842f0acaf7SMatthias Ringwald }
3852f0acaf7SMatthias Ringwald if (connection->browsing_connection->state != AVCTP_CONNECTION_W4_SDP_QUERY_COMPLETE){
3862f0acaf7SMatthias Ringwald return;
3872f0acaf7SMatthias Ringwald }
3882f0acaf7SMatthias Ringwald
3892f0acaf7SMatthias Ringwald // l2cap available?
3902f0acaf7SMatthias Ringwald if (status == ERROR_CODE_SUCCESS){
3912f0acaf7SMatthias Ringwald if (connection->browsing_l2cap_psm == 0){
3922f0acaf7SMatthias Ringwald status = SDP_SERVICE_NOT_FOUND;
3932f0acaf7SMatthias Ringwald }
3942f0acaf7SMatthias Ringwald }
3952f0acaf7SMatthias Ringwald
3962f0acaf7SMatthias Ringwald if (status == ERROR_CODE_SUCCESS) {
3972f0acaf7SMatthias Ringwald // ready to connect
3982f0acaf7SMatthias Ringwald connection->browsing_connection->state = AVCTP_CONNECTION_W2_L2CAP_CONNECT;
3992f0acaf7SMatthias Ringwald
4002f0acaf7SMatthias Ringwald // check if both events have been handled
4012f0acaf7SMatthias Ringwald avrcp_connection_t *connection_with_opposite_role;
4022f0acaf7SMatthias Ringwald switch (connection->role) {
4032f0acaf7SMatthias Ringwald case AVRCP_CONTROLLER:
4042f0acaf7SMatthias Ringwald connection_with_opposite_role = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET,
4052f0acaf7SMatthias Ringwald connection->avrcp_cid);
4062f0acaf7SMatthias Ringwald break;
4072f0acaf7SMatthias Ringwald case AVRCP_TARGET:
4082f0acaf7SMatthias Ringwald connection_with_opposite_role = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_CONTROLLER,
4092f0acaf7SMatthias Ringwald connection->avrcp_cid);
4102f0acaf7SMatthias Ringwald break;
4112f0acaf7SMatthias Ringwald default:
4122f0acaf7SMatthias Ringwald btstack_assert(false);
4132f0acaf7SMatthias Ringwald return;
4142f0acaf7SMatthias Ringwald }
415*2e97cf5eSMatthias Ringwald if (connection_with_opposite_role->browsing_connection->state == AVCTP_CONNECTION_W2_L2CAP_CONNECT) {
4162f0acaf7SMatthias Ringwald
4172f0acaf7SMatthias Ringwald connection->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
4182f0acaf7SMatthias Ringwald connection_with_opposite_role->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
4192f0acaf7SMatthias Ringwald
4202f0acaf7SMatthias Ringwald l2cap_ertm_create_channel(avrcp_browsing_packet_handler,
4212f0acaf7SMatthias Ringwald connection->remote_addr,
4222f0acaf7SMatthias Ringwald connection->browsing_l2cap_psm,
4232f0acaf7SMatthias Ringwald &connection->browsing_connection->ertm_config,
4242f0acaf7SMatthias Ringwald connection->browsing_connection->ertm_buffer,
4252f0acaf7SMatthias Ringwald connection->browsing_connection->ertm_buffer_size,
4262f0acaf7SMatthias Ringwald NULL);
4272f0acaf7SMatthias Ringwald }
4282f0acaf7SMatthias Ringwald } else {
4292f0acaf7SMatthias Ringwald avrcp_browsing_finalize_connection(connection);
4302f0acaf7SMatthias Ringwald avrcp_browsing_emit_connection_established(connection->avrcp_browsing_cid, connection->remote_addr, status);
4312f0acaf7SMatthias Ringwald }
4322f0acaf7SMatthias Ringwald }
4332f0acaf7SMatthias Ringwald
avrcp_browsing_init(void)4342f0acaf7SMatthias Ringwald void avrcp_browsing_init(void){
4352f0acaf7SMatthias Ringwald avrcp_register_browsing_sdp_query_complete_handler(&avrcp_browsing_handle_sdp_query_complete);
4362f0acaf7SMatthias Ringwald if (avrcp_browsing_l2cap_service_registered) return;
4372f0acaf7SMatthias Ringwald uint8_t status = l2cap_register_service(&avrcp_browsing_packet_handler, PSM_AVCTP_BROWSING, 0xffff, LEVEL_2);
4382f0acaf7SMatthias Ringwald avrcp_browsing_l2cap_service_registered = status == ERROR_CODE_SUCCESS;
439665a00cbSMilanka Ringwald }
440665a00cbSMilanka Ringwald
avrcp_browsing_deinit(void)441680af5dcSMatthias Ringwald void avrcp_browsing_deinit(void){
442680af5dcSMatthias Ringwald avrcp_browsing_callback = NULL;
443680af5dcSMatthias Ringwald avrcp_browsing_controller_packet_handler = NULL;
444680af5dcSMatthias Ringwald avrcp_browsing_target_packet_handler = NULL;
44565bd7af5SMatthias Ringwald
446680af5dcSMatthias Ringwald (void) memset(avrcp_browsing_sdp_addr, 0, 6);
44765bd7af5SMatthias Ringwald (void) memset(&avrcp_browsing_handle_sdp_client_query_request, 0, sizeof(avrcp_browsing_handle_sdp_client_query_request));
44865bd7af5SMatthias Ringwald
44965bd7af5SMatthias Ringwald avrcp_browsing_l2cap_service_registered = false;
450680af5dcSMatthias Ringwald }
451680af5dcSMatthias Ringwald
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)452665a00cbSMilanka 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){
453665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_controller_packet_handler != NULL);
454665a00cbSMilanka Ringwald btstack_assert(avrcp_browsing_target_packet_handler != NULL);
455665a00cbSMilanka Ringwald
4561945fe3eSMilanka Ringwald avrcp_connection_t * connection_controller = avrcp_get_connection_for_bd_addr_for_role(AVRCP_CONTROLLER, remote_addr);
457665a00cbSMilanka Ringwald if (!connection_controller){
458665a00cbSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED;
459665a00cbSMilanka Ringwald }
4601945fe3eSMilanka Ringwald avrcp_connection_t * connection_target = avrcp_get_connection_for_bd_addr_for_role(AVRCP_TARGET, remote_addr);
461665a00cbSMilanka Ringwald if (!connection_target){
462665a00cbSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED;
463665a00cbSMilanka Ringwald }
464665a00cbSMilanka Ringwald
465665a00cbSMilanka Ringwald if (connection_controller->browsing_connection){
466665a00cbSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED;
467665a00cbSMilanka Ringwald }
468665a00cbSMilanka Ringwald if (connection_target->browsing_connection){
469665a00cbSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED;
470665a00cbSMilanka Ringwald }
471665a00cbSMilanka Ringwald
472665a00cbSMilanka Ringwald uint16_t cid = avrcp_get_next_cid(AVRCP_CONTROLLER);
473665a00cbSMilanka Ringwald
474665a00cbSMilanka Ringwald connection_controller->browsing_connection = avrcp_browsing_create_connection(connection_controller, cid);
475665a00cbSMilanka Ringwald if (!connection_controller->browsing_connection) return BTSTACK_MEMORY_ALLOC_FAILED;
476665a00cbSMilanka Ringwald
477665a00cbSMilanka Ringwald connection_target->browsing_connection = avrcp_browsing_create_connection(connection_target, cid);
478665a00cbSMilanka Ringwald if (!connection_target->browsing_connection){
479665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_controller);
480665a00cbSMilanka Ringwald return BTSTACK_MEMORY_ALLOC_FAILED;
481665a00cbSMilanka Ringwald }
482665a00cbSMilanka Ringwald avrcp_browsing_configure_ertm(connection_controller->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
483665a00cbSMilanka Ringwald avrcp_browsing_configure_ertm(connection_target->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
484665a00cbSMilanka Ringwald
485665a00cbSMilanka Ringwald if (avrcp_browsing_cid != NULL){
486665a00cbSMilanka Ringwald *avrcp_browsing_cid = cid;
487665a00cbSMilanka Ringwald }
488665a00cbSMilanka Ringwald
489e71114b0SMatthias Ringwald if (connection_controller->browsing_l2cap_psm == 0){
490e71114b0SMatthias Ringwald memcpy(avrcp_browsing_sdp_addr, remote_addr, 6);
4912f0acaf7SMatthias Ringwald connection_controller->browsing_connection->state = AVCTP_CONNECTION_W4_SDP_QUERY_COMPLETE;
4922f0acaf7SMatthias Ringwald connection_target->browsing_connection->state = AVCTP_CONNECTION_W4_SDP_QUERY_COMPLETE;
49364a27ec5SMilanka Ringwald
4942f0acaf7SMatthias Ringwald avrcp_trigger_sdp_query(connection_controller, connection_target);
49564a27ec5SMilanka Ringwald return ERROR_CODE_SUCCESS;
496e71114b0SMatthias Ringwald } else {
497665a00cbSMilanka Ringwald connection_controller->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
498665a00cbSMilanka Ringwald connection_target->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
499665a00cbSMilanka Ringwald
500f25e60deSMatthias Ringwald return l2cap_ertm_create_channel(avrcp_browsing_packet_handler, remote_addr, connection_controller->browsing_l2cap_psm,
501665a00cbSMilanka Ringwald &connection_controller->browsing_connection->ertm_config,
502665a00cbSMilanka Ringwald connection_controller->browsing_connection->ertm_buffer,
503665a00cbSMilanka Ringwald connection_controller->browsing_connection->ertm_buffer_size, NULL);
504e71114b0SMatthias Ringwald }
505665a00cbSMilanka Ringwald }
506665a00cbSMilanka Ringwald
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)507665a00cbSMilanka 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){
5081945fe3eSMilanka Ringwald avrcp_connection_t * connection_controller = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
509665a00cbSMilanka Ringwald if (!connection_controller){
510665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
511665a00cbSMilanka Ringwald }
5121945fe3eSMilanka Ringwald avrcp_connection_t * connection_target = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_TARGET, avrcp_browsing_cid);
513665a00cbSMilanka Ringwald if (!connection_target){
514665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
515665a00cbSMilanka Ringwald }
516665a00cbSMilanka Ringwald
517665a00cbSMilanka Ringwald if (!connection_controller->browsing_connection){
518665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
519665a00cbSMilanka Ringwald }
520665a00cbSMilanka Ringwald if (!connection_target->browsing_connection){
521665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
522665a00cbSMilanka Ringwald }
523665a00cbSMilanka Ringwald
524665a00cbSMilanka Ringwald if (connection_controller->browsing_connection->state != AVCTP_CONNECTION_W4_ERTM_CONFIGURATION){
525665a00cbSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED;
526665a00cbSMilanka Ringwald }
527665a00cbSMilanka Ringwald
528665a00cbSMilanka Ringwald avrcp_browsing_configure_ertm(connection_controller->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
529665a00cbSMilanka Ringwald avrcp_browsing_configure_ertm(connection_target->browsing_connection, ertm_buffer, ertm_buffer_size, ertm_config);
530665a00cbSMilanka Ringwald
531665a00cbSMilanka Ringwald connection_controller->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
532665a00cbSMilanka Ringwald connection_target->browsing_connection->state = AVCTP_CONNECTION_W4_L2CAP_CONNECTED;
533665a00cbSMilanka Ringwald
534f25e60deSMatthias Ringwald l2cap_ertm_accept_connection(connection_controller->browsing_connection->l2cap_browsing_cid,
535665a00cbSMilanka Ringwald &connection_controller->browsing_connection->ertm_config,
536665a00cbSMilanka Ringwald connection_controller->browsing_connection->ertm_buffer,
537665a00cbSMilanka Ringwald connection_controller->browsing_connection->ertm_buffer_size);
538665a00cbSMilanka Ringwald return ERROR_CODE_SUCCESS;
539665a00cbSMilanka Ringwald }
540665a00cbSMilanka Ringwald
541665a00cbSMilanka Ringwald
avrcp_browsing_decline_incoming_connection(uint16_t avrcp_browsing_cid)542665a00cbSMilanka Ringwald uint8_t avrcp_browsing_decline_incoming_connection(uint16_t avrcp_browsing_cid){
5431945fe3eSMilanka Ringwald avrcp_connection_t * connection_controller = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
544665a00cbSMilanka Ringwald if (!connection_controller){
545665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
546665a00cbSMilanka Ringwald }
5471945fe3eSMilanka Ringwald avrcp_connection_t * connection_target = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_TARGET, avrcp_browsing_cid);
548665a00cbSMilanka Ringwald if (!connection_target){
549665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
550665a00cbSMilanka Ringwald }
551665a00cbSMilanka Ringwald
552665a00cbSMilanka Ringwald if (!connection_controller->browsing_connection){
553665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
554665a00cbSMilanka Ringwald }
555665a00cbSMilanka Ringwald if (!connection_target->browsing_connection){
556665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
557665a00cbSMilanka Ringwald }
558665a00cbSMilanka Ringwald
559665a00cbSMilanka Ringwald if (connection_controller->browsing_connection->state != AVCTP_CONNECTION_W4_ERTM_CONFIGURATION){
560665a00cbSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED;
561665a00cbSMilanka Ringwald }
562665a00cbSMilanka Ringwald
563665a00cbSMilanka Ringwald l2cap_decline_connection(connection_controller->browsing_connection->l2cap_browsing_cid);
564665a00cbSMilanka Ringwald
565665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_controller);
566665a00cbSMilanka Ringwald avrcp_browsing_finalize_connection(connection_target);
567665a00cbSMilanka Ringwald return ERROR_CODE_SUCCESS;
568665a00cbSMilanka Ringwald }
569665a00cbSMilanka Ringwald
avrcp_browsing_disconnect(uint16_t avrcp_browsing_cid)570665a00cbSMilanka Ringwald uint8_t avrcp_browsing_disconnect(uint16_t avrcp_browsing_cid){
5711945fe3eSMilanka Ringwald avrcp_connection_t * connection_controller = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
572665a00cbSMilanka Ringwald if (!connection_controller){
573665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
574665a00cbSMilanka Ringwald }
5751945fe3eSMilanka Ringwald avrcp_connection_t * connection_target = avrcp_get_connection_for_browsing_cid_for_role(AVRCP_TARGET, avrcp_browsing_cid);
576665a00cbSMilanka Ringwald if (!connection_target){
577665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
578665a00cbSMilanka Ringwald }
579665a00cbSMilanka Ringwald
580665a00cbSMilanka Ringwald if (!connection_controller->browsing_connection){
581665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
582665a00cbSMilanka Ringwald }
583665a00cbSMilanka Ringwald if (!connection_target->browsing_connection){
584665a00cbSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
585665a00cbSMilanka Ringwald }
586665a00cbSMilanka Ringwald
587b93f8966SMatthias Ringwald l2cap_disconnect(connection_controller->browsing_connection->l2cap_browsing_cid);
588665a00cbSMilanka Ringwald return ERROR_CODE_SUCCESS;
589665a00cbSMilanka Ringwald }
590665a00cbSMilanka Ringwald
avrcp_browsing_register_controller_packet_handler(btstack_packet_handler_t callback)591665a00cbSMilanka Ringwald void avrcp_browsing_register_controller_packet_handler(btstack_packet_handler_t callback){
592665a00cbSMilanka Ringwald avrcp_browsing_controller_packet_handler = callback;
593665a00cbSMilanka Ringwald }
594665a00cbSMilanka Ringwald
avrcp_browsing_register_target_packet_handler(btstack_packet_handler_t callback)595665a00cbSMilanka Ringwald void avrcp_browsing_register_target_packet_handler(btstack_packet_handler_t callback){
596665a00cbSMilanka Ringwald avrcp_browsing_target_packet_handler = callback;
597665a00cbSMilanka Ringwald }
598665a00cbSMilanka Ringwald
avrcp_browsing_register_packet_handler(btstack_packet_handler_t callback)599665a00cbSMilanka Ringwald void avrcp_browsing_register_packet_handler(btstack_packet_handler_t callback){
600665a00cbSMilanka Ringwald btstack_assert(callback != NULL);
601665a00cbSMilanka Ringwald avrcp_browsing_callback = callback;
602665a00cbSMilanka Ringwald }
603