xref: /btstack/example/avrcp_browsing_client.c (revision fcd55a0b5a8c0d39e1cebac18e790e1dbf38bd64)
1462aa085SMilanka Ringwald /*
2462aa085SMilanka Ringwald  * Copyright (C) 2016 BlueKitchen GmbH
3462aa085SMilanka Ringwald  *
4462aa085SMilanka Ringwald  * Redistribution and use in source and binary forms, with or without
5462aa085SMilanka Ringwald  * modification, are permitted provided that the following conditions
6462aa085SMilanka Ringwald  * are met:
7462aa085SMilanka Ringwald  *
8462aa085SMilanka Ringwald  * 1. Redistributions of source code must retain the above copyright
9462aa085SMilanka Ringwald  *    notice, this list of conditions and the following disclaimer.
10462aa085SMilanka Ringwald  * 2. Redistributions in binary form must reproduce the above copyright
11462aa085SMilanka Ringwald  *    notice, this list of conditions and the following disclaimer in the
12462aa085SMilanka Ringwald  *    documentation and/or other materials provided with the distribution.
13462aa085SMilanka Ringwald  * 3. Neither the name of the copyright holders nor the names of
14462aa085SMilanka Ringwald  *    contributors may be used to endorse or promote products derived
15462aa085SMilanka Ringwald  *    from this software without specific prior written permission.
16462aa085SMilanka Ringwald  * 4. Any redistribution, use, or modification is done solely for
17462aa085SMilanka Ringwald  *    personal benefit and not for any commercial purpose or for
18462aa085SMilanka Ringwald  *    monetary gain.
19462aa085SMilanka Ringwald  *
20462aa085SMilanka Ringwald  * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
21462aa085SMilanka Ringwald  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22462aa085SMilanka 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,
25462aa085SMilanka Ringwald  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26462aa085SMilanka Ringwald  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
27462aa085SMilanka Ringwald  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
28462aa085SMilanka Ringwald  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
29462aa085SMilanka Ringwald  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
30462aa085SMilanka Ringwald  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31462aa085SMilanka Ringwald  * SUCH DAMAGE.
32462aa085SMilanka Ringwald  *
33462aa085SMilanka Ringwald  * Please inquire about commercial licensing options at
34462aa085SMilanka Ringwald  * [email protected]
35462aa085SMilanka Ringwald  *
36462aa085SMilanka Ringwald  */
37462aa085SMilanka Ringwald 
38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "avrcp_browsing_client.c"
39462aa085SMilanka Ringwald 
40462aa085SMilanka Ringwald /*
41462aa085SMilanka Ringwald  * avrcp_browsing_client.c
42462aa085SMilanka Ringwald  */
43462aa085SMilanka Ringwald 
44462aa085SMilanka Ringwald // *****************************************************************************
45ec8ae085SMilanka Ringwald /* EXAMPLE_START(avrcp_browsing_client): AVRCP Browsing - Browse Media Players and Media Information
46462aa085SMilanka Ringwald  *
47462aa085SMilanka Ringwald  * @text This example demonstrates how to use the AVRCP Controller Browsing service to
48462aa085SMilanka Ringwald  * browse madia players and media information on a remote AVRCP Source device.
49462aa085SMilanka Ringwald  *
5069f3ffd7SMilanka Ringwald  * @text To test with a remote device, e.g. a mobile phone,
5169f3ffd7SMilanka Ringwald  * pair from the remote device with the demo, then use the UI for browsing. If HAVE_BTSTACK_STDIN is set,
5269f3ffd7SMilanka Ringwald  * press SPACE on the console to show the available AVDTP and AVRCP commands.
53462aa085SMilanka Ringwald  *
54462aa085SMilanka Ringwald  */
55462aa085SMilanka Ringwald // *****************************************************************************
56462aa085SMilanka Ringwald 
57462aa085SMilanka Ringwald #include <stdint.h>
58954cc391SMilanka Ringwald #include <inttypes.h>
59462aa085SMilanka Ringwald #include <stdio.h>
60462aa085SMilanka Ringwald #include <stdlib.h>
61462aa085SMilanka Ringwald #include <string.h>
62462aa085SMilanka Ringwald 
63462aa085SMilanka Ringwald #include "btstack.h"
64462aa085SMilanka Ringwald 
65462aa085SMilanka Ringwald #ifdef HAVE_BTSTACK_STDIN
66462aa085SMilanka Ringwald #include "btstack_stdin.h"
67462aa085SMilanka Ringwald #endif
68462aa085SMilanka Ringwald 
6916fc96d5SMatthias Ringwald #ifndef ENABLE_L2CAP_ENHANCED_RETRANSMISSION_MODE
7016fc96d5SMatthias Ringwald #error "AVRCP Browsing requires L2CAP ERTM, please add ENABLE_L2CAP_ENHANCED_RETRANSMISSION_MODE to your btstack_config.h"
7116fc96d5SMatthias Ringwald #endif
7216fc96d5SMatthias Ringwald 
734f0111ebSMilanka Ringwald #define AVRCP_BROWSING_ENABLED
744f0111ebSMilanka Ringwald 
75db79553aSMilanka Ringwald #define AVRCP_BROWSING_MAX_PLAYERS                  10
76db79553aSMilanka Ringwald #define AVRCP_BROWSING_MAX_FOLDERS                  10
775666fff4SMilanka Ringwald #define AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN  30
785666fff4SMilanka Ringwald #define AVRCP_BROWSING_MAX_MEDIA_ITEMS              10
79db79553aSMilanka Ringwald 
80462aa085SMilanka Ringwald #ifdef HAVE_BTSTACK_STDIN
81462aa085SMilanka Ringwald // mac 2011: static bd_addr_t remote = {0x04, 0x0C, 0xCE, 0xE4, 0x85, 0xD3};
82462aa085SMilanka Ringwald // pts: static bd_addr_t remote = {0x00, 0x1B, 0xDC, 0x08, 0x0A, 0xA5};
83462aa085SMilanka Ringwald // mac 2013:
84462aa085SMilanka Ringwald // static const char * device_addr_string = "84:38:35:65:d1:15";
85462aa085SMilanka Ringwald // iPhone 5S: static const char * device_addr_string = "54:E4:3A:26:A2:39";
86462aa085SMilanka Ringwald // phone 2013:
87143efbe8SMilanka Ringwald // static const char * device_addr_string = "B0:34:95:CB:97:C4";
8802d2f9caSMilanka Ringwald // iPod
890eebc132SMilanka Ringwald // static const char * device_addr_string = "B0:34:95:CB:97:C4";
900eebc132SMilanka Ringwald // iPhone
910eebc132SMilanka Ringwald static const char * device_addr_string = "6C:72:E7:10:22:EE";
92143efbe8SMilanka Ringwald 
93462aa085SMilanka Ringwald static bd_addr_t device_addr;
94462aa085SMilanka Ringwald #endif
95462aa085SMilanka Ringwald 
965666fff4SMilanka Ringwald typedef enum {
975666fff4SMilanka Ringwald     AVRCP_BROWSING_STATE_IDLE,
985666fff4SMilanka Ringwald     AVRCP_BROWSING_STATE_W4_GET_PLAYERS,
995666fff4SMilanka Ringwald     AVRCP_BROWSING_STATE_W4_SET_PLAYER,
1005666fff4SMilanka Ringwald     AVRCP_BROWSING_STATE_READY
1015666fff4SMilanka Ringwald } avrcp_browsing_state_t;
102954cc391SMilanka Ringwald 
103db79553aSMilanka Ringwald typedef struct {
104db79553aSMilanka Ringwald     uint16_t  charset;
105db79553aSMilanka Ringwald     uint8_t   depth;
106db79553aSMilanka Ringwald     uint16_t  name_len;
1075666fff4SMilanka Ringwald     char      name[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
108db79553aSMilanka Ringwald } avrcp_browsing_root_folder_t;
109db79553aSMilanka Ringwald 
110db79553aSMilanka Ringwald typedef struct {
111db79553aSMilanka Ringwald     uint8_t  uid[8];
112db79553aSMilanka Ringwald     uint16_t name_len;
1135666fff4SMilanka Ringwald     char     name[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
1145666fff4SMilanka Ringwald } avrcp_browsable_item_t;
1155666fff4SMilanka Ringwald 
1165666fff4SMilanka Ringwald typedef struct {
1175666fff4SMilanka Ringwald     uint16_t player_id;
1185666fff4SMilanka Ringwald     uint8_t  major_player_type;
1195666fff4SMilanka Ringwald     uint32_t player_sub_type;
1205666fff4SMilanka Ringwald     uint8_t  play_status;
1215666fff4SMilanka Ringwald     uint8_t  feature_bitmask[16];
1225666fff4SMilanka Ringwald     uint16_t charset;
1235666fff4SMilanka Ringwald     uint16_t name_len;
1245666fff4SMilanka Ringwald     char     name[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
1255666fff4SMilanka Ringwald } avrcp_browsable_media_player_item_t;
1265666fff4SMilanka Ringwald 
1275666fff4SMilanka Ringwald typedef struct {
1285666fff4SMilanka Ringwald     uint32_t id;
1295666fff4SMilanka Ringwald     uint16_t charset;
1305666fff4SMilanka Ringwald     uint16_t value_len;
1315666fff4SMilanka Ringwald     char     value[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
1325666fff4SMilanka Ringwald } avrcp_browsable_media_element_item_attribute_t;
1335666fff4SMilanka Ringwald 
1345666fff4SMilanka Ringwald typedef struct {
1355666fff4SMilanka Ringwald     uint8_t  uid[8];
1365666fff4SMilanka Ringwald     uint8_t  media_type;
1375666fff4SMilanka Ringwald     uint16_t charset;
1385666fff4SMilanka Ringwald     uint16_t name_len;
1395666fff4SMilanka Ringwald     char     name[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
1405666fff4SMilanka Ringwald     uint8_t  num_attributes;
1415666fff4SMilanka Ringwald } avrcp_browsable_media_element_item_t;
1425666fff4SMilanka Ringwald 
1435666fff4SMilanka Ringwald static avrcp_browsing_state_t browsing_state = AVRCP_BROWSING_STATE_IDLE;
1445666fff4SMilanka Ringwald static uint16_t avrcp_cid = 0;
1455666fff4SMilanka Ringwald static bool     avrcp_connected = false;
1465666fff4SMilanka Ringwald 
1475666fff4SMilanka Ringwald static uint16_t browsing_cid = 0;
1485666fff4SMilanka Ringwald static bool     browsing_connected = false;
1495666fff4SMilanka Ringwald 
1505666fff4SMilanka Ringwald static uint8_t  sdp_avrcp_browsing_controller_service_buffer[200];
1515666fff4SMilanka Ringwald static uint8_t  sdp_avdtp_sink_service_buffer[150];
1525666fff4SMilanka Ringwald 
1535666fff4SMilanka Ringwald static uint16_t a2dp_cid = 0;
1545666fff4SMilanka Ringwald static uint8_t  a2dp_local_seid = 0;
1555666fff4SMilanka Ringwald 
1565666fff4SMilanka Ringwald static uint8_t media_sbc_codec_capabilities[] = {
1575666fff4SMilanka Ringwald     0xFF,//(AVDTP_SBC_44100 << 4) | AVDTP_SBC_STEREO,
1585666fff4SMilanka Ringwald     0xFF,//(AVDTP_SBC_BLOCK_LENGTH_16 << 4) | (AVDTP_SBC_SUBBANDS_8 << 2) | AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS,
1595666fff4SMilanka Ringwald     2, 53
1605666fff4SMilanka Ringwald };
1615666fff4SMilanka Ringwald 
1625666fff4SMilanka Ringwald static uint8_t media_sbc_codec_configuration[] = {
1635666fff4SMilanka Ringwald     (AVDTP_SBC_44100 << 4) | AVDTP_SBC_STEREO,
1645666fff4SMilanka Ringwald     (AVDTP_SBC_BLOCK_LENGTH_16 << 4) | (AVDTP_SBC_SUBBANDS_8 << 2) | AVDTP_SBC_ALLOCATION_METHOD_LOUDNESS,
1655666fff4SMilanka Ringwald     2, 53
1665666fff4SMilanka Ringwald };
1675666fff4SMilanka Ringwald 
1685666fff4SMilanka Ringwald static bool     browsing_query_active = false;
1695666fff4SMilanka Ringwald // static avrcp_media_item_context_t media_item_context;
1705666fff4SMilanka Ringwald 
1715666fff4SMilanka Ringwald static int playable_folder_index = 0;
1725666fff4SMilanka Ringwald 
1735666fff4SMilanka Ringwald 
1745666fff4SMilanka Ringwald static uint16_t browsing_uid_counter = 0;
175db79553aSMilanka Ringwald 
176db79553aSMilanka Ringwald static uint8_t  parent_folder_set = 0;
177db79553aSMilanka Ringwald static uint8_t  parent_folder_uid[8];
1785666fff4SMilanka Ringwald static char     parent_folder_name[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
1795666fff4SMilanka Ringwald 
1805666fff4SMilanka Ringwald static avrcp_browsable_item_t folders[AVRCP_BROWSING_MAX_FOLDERS];
181db79553aSMilanka Ringwald static int folder_index = -1;
1825666fff4SMilanka Ringwald 
1835666fff4SMilanka Ringwald static avrcp_browsable_media_element_item_t media_element_items[AVRCP_BROWSING_MAX_MEDIA_ITEMS];
1845666fff4SMilanka Ringwald static int media_element_item_index = -1;
1855666fff4SMilanka Ringwald 
1865666fff4SMilanka Ringwald static avrcp_browsable_media_player_item_t media_player_items[AVRCP_BROWSING_MAX_MEDIA_ITEMS];
1875666fff4SMilanka Ringwald static int media_player_item_index = -1;
188db79553aSMilanka Ringwald 
189462aa085SMilanka Ringwald static btstack_packet_callback_registration_t hci_event_callback_registration;
190462aa085SMilanka Ringwald 
191462aa085SMilanka Ringwald static uint8_t ertm_buffer[10000];
192462aa085SMilanka Ringwald static l2cap_ertm_config_t ertm_config = {
193462aa085SMilanka Ringwald     1,  // ertm mandatory
194462aa085SMilanka Ringwald     2,  // max transmit, some tests require > 1
195462aa085SMilanka Ringwald     2000,
196462aa085SMilanka Ringwald     12000,
197462aa085SMilanka Ringwald     144,    // l2cap ertm mtu
198462aa085SMilanka Ringwald     4,
199462aa085SMilanka Ringwald     4,
2007c7befacSMatthias Ringwald     1,     // 16-bit FCS
201462aa085SMilanka Ringwald };
202462aa085SMilanka Ringwald 
2035666fff4SMilanka Ringwald 
204db79553aSMilanka Ringwald static inline int next_index(int * index, int max_value){
205db79553aSMilanka Ringwald     if ((*index) < max_value){
206db79553aSMilanka Ringwald         (*index)++;
207db79553aSMilanka Ringwald     } else {
208db79553aSMilanka Ringwald         (*index) = 0;
209db79553aSMilanka Ringwald     }
210db79553aSMilanka Ringwald     return (*index);
211db79553aSMilanka Ringwald }
212db79553aSMilanka Ringwald 
2130bd372cfSMatthias Ringwald static int next_folder_index(void){
214db79553aSMilanka Ringwald     return next_index(&folder_index, AVRCP_BROWSING_MAX_FOLDERS);
215db79553aSMilanka Ringwald }
216db79553aSMilanka Ringwald 
2170bd372cfSMatthias Ringwald static int next_media_element_item_index(void){
2185666fff4SMilanka Ringwald     return next_index(&media_element_item_index, AVRCP_BROWSING_MAX_MEDIA_ITEMS);
219db79553aSMilanka Ringwald }
2200bd372cfSMatthias Ringwald static int next_media_player_item_index(void){
2215666fff4SMilanka Ringwald     return next_index(&media_player_item_index, AVRCP_BROWSING_MAX_MEDIA_ITEMS);
2225666fff4SMilanka Ringwald }
223db79553aSMilanka Ringwald 
224462aa085SMilanka Ringwald /* @section Main Application Setup
225462aa085SMilanka Ringwald  *
226462aa085SMilanka Ringwald  * @text The Listing MainConfiguration shows how to setup AVRCP Controller Browsing service.
227462aa085SMilanka Ringwald  * To announce AVRCP Controller Browsing service, you need to create corresponding
228462aa085SMilanka Ringwald  * SDP record and register it with the SDP service.
229462aa085SMilanka Ringwald  * You'll also need to register several packet handlers:
230462aa085SMilanka Ringwald  * - stdin_process callback - used to trigger AVRCP commands, such are get media players, playlists, albums, etc. Requires HAVE_BTSTACK_STDIN.
2315666fff4SMilanka Ringwald  * - avrcp_browsing_controller_packet_handler - used to receive answers for AVRCP commands.
232462aa085SMilanka Ringwald  *
233462aa085SMilanka Ringwald  */
234462aa085SMilanka Ringwald 
235462aa085SMilanka Ringwald /* LISTING_START(MainConfiguration): Setup Audio Sink and AVRCP Controller services */
2365666fff4SMilanka Ringwald static void avrcp_browsing_controller_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
23702d2f9caSMilanka Ringwald static void avrcp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
2385666fff4SMilanka Ringwald static void a2dp_sink_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
23902d2f9caSMilanka Ringwald 
240462aa085SMilanka Ringwald #ifdef HAVE_BTSTACK_STDIN
241462aa085SMilanka Ringwald static void stdin_process(char cmd);
242462aa085SMilanka Ringwald #endif
243462aa085SMilanka Ringwald 
244462aa085SMilanka Ringwald 
245462aa085SMilanka Ringwald int btstack_main(int argc, const char * argv[]);
246462aa085SMilanka Ringwald int btstack_main(int argc, const char * argv[]){
247462aa085SMilanka Ringwald     (void)argc;
248462aa085SMilanka Ringwald     (void)argv;
249462aa085SMilanka Ringwald 
250462aa085SMilanka Ringwald     // Initialize L2CAP.
251462aa085SMilanka Ringwald     l2cap_init();
252462aa085SMilanka Ringwald 
2538c9bb29eSMatthias Ringwald #ifdef ENABLE_BLE
2548c9bb29eSMatthias Ringwald     // Initialize LE Security Manager. Needed for cross-transport key derivation
2558c9bb29eSMatthias Ringwald     sm_init();
2568c9bb29eSMatthias Ringwald #endif
2578c9bb29eSMatthias Ringwald 
2585666fff4SMilanka Ringwald     a2dp_sink_init();
2595666fff4SMilanka Ringwald     a2dp_sink_register_packet_handler(&a2dp_sink_packet_handler);
2605666fff4SMilanka Ringwald 
2615666fff4SMilanka Ringwald     avdtp_stream_endpoint_t * local_stream_endpoint = a2dp_sink_create_stream_endpoint(AVDTP_AUDIO,
2625666fff4SMilanka Ringwald         AVDTP_CODEC_SBC, media_sbc_codec_capabilities, sizeof(media_sbc_codec_capabilities),
2635666fff4SMilanka Ringwald         media_sbc_codec_configuration, sizeof(media_sbc_codec_configuration));
2645666fff4SMilanka Ringwald     if (!local_stream_endpoint){
2655666fff4SMilanka Ringwald         printf("A2DP Sink: not enough memory to create local stream endpoint\n");
2665666fff4SMilanka Ringwald         return 1;
2675666fff4SMilanka Ringwald     }
2685666fff4SMilanka Ringwald     a2dp_local_seid = avdtp_local_seid(local_stream_endpoint);
2695666fff4SMilanka Ringwald 
27019691e0aSMilanka Ringwald     // Initialize AVRCP service.
27119691e0aSMilanka Ringwald     avrcp_init();
27202d2f9caSMilanka Ringwald     // Initialize AVRCP Controller & Target Service.
273462aa085SMilanka Ringwald     avrcp_controller_init();
27402d2f9caSMilanka Ringwald     avrcp_target_init();
27502d2f9caSMilanka Ringwald 
27602d2f9caSMilanka Ringwald     avrcp_register_packet_handler(&avrcp_packet_handler);
27702d2f9caSMilanka Ringwald     avrcp_controller_register_packet_handler(&avrcp_packet_handler);
27802d2f9caSMilanka Ringwald     avrcp_target_register_packet_handler(&avrcp_packet_handler);
279462aa085SMilanka Ringwald 
2800eebc132SMilanka Ringwald     // Initialize AVRCP Browsing Service.
2810eebc132SMilanka Ringwald     avrcp_browsing_init();
282462aa085SMilanka Ringwald     avrcp_browsing_controller_init();
2830eebc132SMilanka Ringwald     avrcp_browsing_target_init();
2840eebc132SMilanka Ringwald 
2850eebc132SMilanka Ringwald     // Register for HCI events.
2865666fff4SMilanka Ringwald     avrcp_browsing_controller_register_packet_handler(&avrcp_browsing_controller_packet_handler);
2875666fff4SMilanka Ringwald     avrcp_browsing_target_register_packet_handler(&avrcp_browsing_controller_packet_handler);
2885666fff4SMilanka Ringwald     avrcp_browsing_register_packet_handler(&avrcp_browsing_controller_packet_handler);
289462aa085SMilanka Ringwald 
290462aa085SMilanka Ringwald     // Initialize SDP.
291462aa085SMilanka Ringwald     sdp_init();
2925666fff4SMilanka Ringwald     // setup AVDTP sink
2935666fff4SMilanka Ringwald     memset(sdp_avdtp_sink_service_buffer, 0, sizeof(sdp_avdtp_sink_service_buffer));
2945666fff4SMilanka Ringwald     a2dp_sink_create_sdp_record(sdp_avdtp_sink_service_buffer, 0x10001, AVDTP_SINK_FEATURE_MASK_HEADPHONE, NULL, NULL);
2955666fff4SMilanka Ringwald     sdp_register_service(sdp_avdtp_sink_service_buffer);
296462aa085SMilanka Ringwald 
297462aa085SMilanka Ringwald     // Create AVRCP service record and register it with SDP.
298462aa085SMilanka Ringwald     memset(sdp_avrcp_browsing_controller_service_buffer, 0, sizeof(sdp_avrcp_browsing_controller_service_buffer));
29911014891SMilanka Ringwald 
300e2f25417SMilanka Ringwald     uint16_t supported_features = AVRCP_FEATURE_MASK_CATEGORY_PLAYER_OR_RECORDER;
3014f0111ebSMilanka Ringwald #ifdef AVRCP_BROWSING_ENABLED
302e2f25417SMilanka Ringwald     supported_features |= AVRCP_FEATURE_MASK_BROWSING;
3034f0111ebSMilanka Ringwald #endif
3045666fff4SMilanka Ringwald     avrcp_controller_create_sdp_record(sdp_avrcp_browsing_controller_service_buffer, 0x10002, supported_features, NULL, NULL);
305462aa085SMilanka Ringwald     sdp_register_service(sdp_avrcp_browsing_controller_service_buffer);
306462aa085SMilanka Ringwald 
307462aa085SMilanka Ringwald     // Set local name with a template Bluetooth address, that will be automatically
308462aa085SMilanka Ringwald     // replaced with a actual address once it is available, i.e. when BTstack boots
309462aa085SMilanka Ringwald     // up and starts talking to a Bluetooth module.
310462aa085SMilanka Ringwald     gap_set_local_name("AVRCP Browsing Client 00:00:00:00:00:00");
311462aa085SMilanka Ringwald     gap_discoverable_control(1);
312462aa085SMilanka Ringwald     gap_set_class_of_device(0x200408);
313462aa085SMilanka Ringwald 
314a4fe6467SMatthias Ringwald     // Register for HCI events.
3155666fff4SMilanka Ringwald     hci_event_callback_registration.callback = &avrcp_browsing_controller_packet_handler;
316a4fe6467SMatthias Ringwald     hci_add_event_handler(&hci_event_callback_registration);
317a4fe6467SMatthias Ringwald 
318a4fe6467SMatthias Ringwald 
319462aa085SMilanka Ringwald #ifdef HAVE_BTSTACK_STDIN
320462aa085SMilanka Ringwald     // Parse human readable Bluetooth address.
321462aa085SMilanka Ringwald     sscanf_bd_addr(device_addr_string, device_addr);
322462aa085SMilanka Ringwald     btstack_stdin_setup(stdin_process);
323462aa085SMilanka Ringwald #endif
324462aa085SMilanka Ringwald     printf("Starting BTstack ...\n");
325462aa085SMilanka Ringwald     hci_power_control(HCI_POWER_ON);
326462aa085SMilanka Ringwald     return 0;
327462aa085SMilanka Ringwald }
328462aa085SMilanka Ringwald /* LISTING_END */
329462aa085SMilanka Ringwald 
33002d2f9caSMilanka Ringwald static void avrcp_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
33102d2f9caSMilanka Ringwald     UNUSED(channel);
33202d2f9caSMilanka Ringwald     UNUSED(size);
33302d2f9caSMilanka Ringwald     uint16_t local_cid;
33402d2f9caSMilanka Ringwald     uint8_t  status = 0xFF;
33502d2f9caSMilanka Ringwald     bd_addr_t adress;
33602d2f9caSMilanka Ringwald 
33702d2f9caSMilanka Ringwald     if (packet_type != HCI_EVENT_PACKET) return;
33802d2f9caSMilanka Ringwald     if (hci_event_packet_get_type(packet) != HCI_EVENT_AVRCP_META) return;
33902d2f9caSMilanka Ringwald     switch (packet[2]){
34002d2f9caSMilanka Ringwald         case AVRCP_SUBEVENT_CONNECTION_ESTABLISHED: {
34102d2f9caSMilanka Ringwald             local_cid = avrcp_subevent_connection_established_get_avrcp_cid(packet);
34202d2f9caSMilanka Ringwald             status = avrcp_subevent_connection_established_get_status(packet);
34302d2f9caSMilanka Ringwald             if (status != ERROR_CODE_SUCCESS){
344*fcd55a0bSMilanka Ringwald                 printf("AVRCP: Connection failed, status 0x%02x\n", status);
34502d2f9caSMilanka Ringwald                 avrcp_cid = 0;
34602d2f9caSMilanka Ringwald                 return;
34702d2f9caSMilanka Ringwald             }
34802d2f9caSMilanka Ringwald 
34902d2f9caSMilanka Ringwald             avrcp_cid = local_cid;
350ace5ab2aSMilanka Ringwald             avrcp_connected = true;
35102d2f9caSMilanka Ringwald             avrcp_subevent_connection_established_get_bd_addr(packet, adress);
35202d2f9caSMilanka Ringwald             printf("AVRCP: Connected to %s, cid 0x%02x\n", bd_addr_to_str(adress), avrcp_cid);
35302d2f9caSMilanka Ringwald             return;
35402d2f9caSMilanka Ringwald         }
35502d2f9caSMilanka Ringwald 
35602d2f9caSMilanka Ringwald         case AVRCP_SUBEVENT_CONNECTION_RELEASED:
35702d2f9caSMilanka Ringwald             printf("AVRCP: Channel released: cid 0x%02x\n", avrcp_subevent_connection_released_get_avrcp_cid(packet));
35802d2f9caSMilanka Ringwald             avrcp_cid = 0;
359ace5ab2aSMilanka Ringwald             avrcp_connected = false;
36002d2f9caSMilanka Ringwald             return;
36102d2f9caSMilanka Ringwald         default:
36202d2f9caSMilanka Ringwald             break;
36302d2f9caSMilanka Ringwald     }
36402d2f9caSMilanka Ringwald }
365462aa085SMilanka Ringwald 
3665666fff4SMilanka Ringwald static void a2dp_sink_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
3675666fff4SMilanka Ringwald     UNUSED(channel);
3685666fff4SMilanka Ringwald     UNUSED(size);
3695666fff4SMilanka Ringwald     bd_addr_t address;
3705666fff4SMilanka Ringwald     uint8_t status;
3715666fff4SMilanka Ringwald 
3725666fff4SMilanka Ringwald     if (packet_type != HCI_EVENT_PACKET) return;
3735666fff4SMilanka Ringwald     if (hci_event_packet_get_type(packet) != HCI_EVENT_A2DP_META) return;
3745666fff4SMilanka Ringwald 
3755666fff4SMilanka Ringwald     switch (packet[2]){
3765666fff4SMilanka Ringwald         case A2DP_SUBEVENT_STREAM_ESTABLISHED:
3775666fff4SMilanka Ringwald             a2dp_subevent_stream_established_get_bd_addr(packet, address);
3785666fff4SMilanka Ringwald             status = a2dp_subevent_stream_established_get_status(packet);
3795666fff4SMilanka Ringwald 
3806058cb0dSMatthias Ringwald             if (status != ERROR_CODE_SUCCESS){
3815666fff4SMilanka Ringwald                 printf("A2DP  Sink      : Streaming connection failed, status 0x%02x\n", status);
3825666fff4SMilanka Ringwald                 break;
3835666fff4SMilanka Ringwald             }
3845666fff4SMilanka Ringwald 
3855666fff4SMilanka Ringwald             a2dp_cid = a2dp_subevent_stream_established_get_a2dp_cid(packet);
3865666fff4SMilanka Ringwald             memcpy(device_addr, address, 6);
3875666fff4SMilanka Ringwald             printf("A2DP  Sink: Connection established, address %s, a2dp_cid 0x%02x\n", bd_addr_to_str(address), a2dp_cid);
3885666fff4SMilanka Ringwald             break;
3895666fff4SMilanka Ringwald 
3905666fff4SMilanka Ringwald         case A2DP_SUBEVENT_SIGNALING_CONNECTION_RELEASED:
3915666fff4SMilanka Ringwald             printf("A2DP  Sink: Connection released\n");
3925666fff4SMilanka Ringwald             break;
3935666fff4SMilanka Ringwald 
3945666fff4SMilanka Ringwald         default:
3955666fff4SMilanka Ringwald             break;
3965666fff4SMilanka Ringwald     }
3975666fff4SMilanka Ringwald }
3985666fff4SMilanka Ringwald 
3995666fff4SMilanka Ringwald 
4005666fff4SMilanka Ringwald static void avrcp_browsing_controller_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
4015666fff4SMilanka Ringwald     UNUSED(channel);
4025666fff4SMilanka Ringwald     UNUSED(size);
4035666fff4SMilanka Ringwald     bd_addr_t address;
4045666fff4SMilanka Ringwald     uint8_t  status;
4055666fff4SMilanka Ringwald     uint16_t pos = 0;
4065666fff4SMilanka Ringwald     uint16_t local_cid;
4075666fff4SMilanka Ringwald 
4085666fff4SMilanka Ringwald     // printf("avrcp_browsing_controller_packet_handler packet type 0x%02X, subevent 0x%02X\n", packet_type, packet[2]);
409954cc391SMilanka Ringwald     switch (packet_type) {
4105666fff4SMilanka Ringwald         case HCI_EVENT_PACKET:
4115666fff4SMilanka Ringwald             switch (packet[0]){
4125666fff4SMilanka Ringwald                 case HCI_EVENT_AVRCP_META:
4135666fff4SMilanka Ringwald                      switch (packet[2]){
4145666fff4SMilanka Ringwald                         case AVRCP_SUBEVENT_BROWSING_CONNECTION_ESTABLISHED: {
4155666fff4SMilanka Ringwald                             local_cid = avrcp_subevent_browsing_connection_established_get_browsing_cid(packet);
4165666fff4SMilanka Ringwald                             status = avrcp_subevent_browsing_connection_established_get_status(packet);
4175666fff4SMilanka Ringwald                             if (status != ERROR_CODE_SUCCESS){
418*fcd55a0bSMilanka Ringwald                                 printf("AVRCP: Connection failed, status 0x%02x\n", status);
4195666fff4SMilanka Ringwald                                 browsing_cid = 0;
4205666fff4SMilanka Ringwald                                 return;
4215666fff4SMilanka Ringwald                             }
4225666fff4SMilanka Ringwald                             avrcp_subevent_browsing_connection_established_get_bd_addr(packet, address);
4235666fff4SMilanka Ringwald                             browsing_cid = local_cid;
4245666fff4SMilanka Ringwald                             browsing_connected = true;
4255666fff4SMilanka Ringwald                             printf("AVRCP Browsing: Connection established, address %s, browsing_cid 0x%02x\n", bd_addr_to_str(address), browsing_cid);
4265666fff4SMilanka Ringwald                             return;
4275666fff4SMilanka Ringwald                         }
4285666fff4SMilanka Ringwald 
4295666fff4SMilanka Ringwald                         case AVRCP_SUBEVENT_BROWSING_CONNECTION_RELEASED:
4305666fff4SMilanka Ringwald                             printf("AVRCP: Connection released, cid 0x%02x\n", avrcp_subevent_browsing_connection_released_get_browsing_cid(packet));
4315666fff4SMilanka Ringwald                             browsing_cid = 0;
4325666fff4SMilanka Ringwald                             browsing_connected = false;
4335666fff4SMilanka Ringwald                             return;
4345666fff4SMilanka Ringwald                         case AVRCP_SUBEVENT_BROWSING_DONE:
4355666fff4SMilanka Ringwald 
4365666fff4SMilanka Ringwald                             browsing_query_active = 0;
4375666fff4SMilanka Ringwald                             browsing_uid_counter = 0;
4385666fff4SMilanka Ringwald                             if (avrcp_subevent_browsing_done_get_browsing_status(packet) != AVRCP_BROWSING_ERROR_CODE_SUCCESS){
4395666fff4SMilanka Ringwald                                 printf("AVRCP Browsing query done with browsing status 0x%02x, bluetooth status 0x%02x.\n",
4405666fff4SMilanka Ringwald                                     avrcp_subevent_browsing_done_get_browsing_status(packet),
4415666fff4SMilanka Ringwald                                     avrcp_subevent_browsing_done_get_bluetooth_status(packet));
4425666fff4SMilanka Ringwald                                 return;
4435666fff4SMilanka Ringwald                             }
4445666fff4SMilanka Ringwald                             browsing_uid_counter = avrcp_subevent_browsing_done_get_uid_counter(packet);
4455666fff4SMilanka Ringwald                             printf("DONE, browsing_uid_counter %d.\n", browsing_uid_counter);
4465666fff4SMilanka Ringwald 
4475666fff4SMilanka Ringwald                             switch (browsing_state){
4485666fff4SMilanka Ringwald                                 case AVRCP_BROWSING_STATE_W4_GET_PLAYERS:
4495666fff4SMilanka Ringwald                                     if (media_player_item_index < 0) {
4505666fff4SMilanka Ringwald                                         printf("Get media players first\n");
4515666fff4SMilanka Ringwald                                         break;
4525666fff4SMilanka Ringwald                                     }
4535666fff4SMilanka Ringwald                                     printf("Set browsed player\n");
4545666fff4SMilanka Ringwald                                     browsing_state = AVRCP_BROWSING_STATE_W4_SET_PLAYER;
4555666fff4SMilanka Ringwald                                     status = avrcp_browsing_controller_set_browsed_player(browsing_cid, media_player_items[0].player_id);
4565666fff4SMilanka Ringwald                                     if (status != ERROR_CODE_SUCCESS){
457*fcd55a0bSMilanka Ringwald                                         printf("Could not set player, status 0x%02x\n", status);
4585666fff4SMilanka Ringwald                                         status = AVRCP_BROWSING_STATE_W4_GET_PLAYERS;
4595666fff4SMilanka Ringwald                                         break;
4605666fff4SMilanka Ringwald                                     }
4615666fff4SMilanka Ringwald                                     break;
4625666fff4SMilanka Ringwald                                 case AVRCP_BROWSING_STATE_W4_SET_PLAYER:
4635666fff4SMilanka Ringwald                                     browsing_state = AVRCP_BROWSING_STATE_READY;
4645666fff4SMilanka Ringwald                                     break;
4655666fff4SMilanka Ringwald                                 default:
4665666fff4SMilanka Ringwald                                     break;
4675666fff4SMilanka Ringwald                             }
4685666fff4SMilanka Ringwald                             break;
4695666fff4SMilanka Ringwald                         default:
4705666fff4SMilanka Ringwald                             break;
4715666fff4SMilanka Ringwald                     }
4725666fff4SMilanka Ringwald 
4735666fff4SMilanka Ringwald                 default:
4745666fff4SMilanka Ringwald                     break;
4755666fff4SMilanka Ringwald             }
4765666fff4SMilanka Ringwald             break;
4775666fff4SMilanka Ringwald 
478954cc391SMilanka Ringwald         case AVRCP_BROWSING_DATA_PACKET:
4795666fff4SMilanka Ringwald             browsing_query_active = 1;
480954cc391SMilanka Ringwald             avrcp_browsing_item_type_t data_type = (avrcp_browsing_item_type_t)packet[pos++];
481954cc391SMilanka Ringwald 
482954cc391SMilanka Ringwald             switch (data_type){
483db79553aSMilanka Ringwald                 case AVRCP_BROWSING_MEDIA_ROOT_FOLDER:{
4845666fff4SMilanka Ringwald                     uint16_t folder_name_length = size - pos;
4855666fff4SMilanka Ringwald                     char folder_name[AVRCP_MAX_FOLDER_NAME_SIZE];
4865666fff4SMilanka Ringwald                     memcpy(folder_name, &packet[pos], folder_name_length);
4875666fff4SMilanka Ringwald                     folder_name[folder_name_length] = 0;
4885666fff4SMilanka Ringwald                     printf("Found root folder: name %s \n", folder_name);
4895666fff4SMilanka Ringwald                     break;
4905666fff4SMilanka Ringwald                 }
4912f511ffbSMilanka Ringwald 
492954cc391SMilanka Ringwald                 case AVRCP_BROWSING_FOLDER_ITEM:{
493db79553aSMilanka Ringwald                     int index = next_folder_index();
494db79553aSMilanka Ringwald                     memcpy(folders[index].uid, packet+pos, 8);
4955666fff4SMilanka Ringwald 
496954cc391SMilanka Ringwald                     uint32_t folder_uid_high = big_endian_read_32(packet, pos);
497954cc391SMilanka Ringwald                     pos += 4;
498db79553aSMilanka Ringwald                     uint32_t folder_uid_low  = big_endian_read_32(packet, pos);
499954cc391SMilanka Ringwald                     pos += 4;
500954cc391SMilanka Ringwald                     avrcp_browsing_folder_type_t folder_type = packet[pos++];
501954cc391SMilanka Ringwald                     uint8_t  is_playable = packet[pos++];
502954cc391SMilanka Ringwald                     uint16_t charset = big_endian_read_16(packet, pos);
503954cc391SMilanka Ringwald                     pos += 2;
504954cc391SMilanka Ringwald                     uint16_t displayable_name_length = big_endian_read_16(packet, pos);
505954cc391SMilanka Ringwald                     pos += 2;
5062f511ffbSMilanka Ringwald 
5075666fff4SMilanka Ringwald                     char value[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
5085666fff4SMilanka Ringwald                     memset(value, 0, AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN);
5095666fff4SMilanka Ringwald                     uint16_t value_len = btstack_min(displayable_name_length, AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN - 1);
510954cc391SMilanka Ringwald                     memcpy(value, packet+pos, value_len);
511db79553aSMilanka Ringwald 
5125666fff4SMilanka Ringwald                     printf("Folder UID 0x%08" PRIx32 "%08" PRIx32 ", type 0x%02x, is_playable %d, charset 0x%02x, name %s\n",
5132f511ffbSMilanka Ringwald                         folder_uid_high, folder_uid_low, folder_type, is_playable, charset, value);
5145666fff4SMilanka Ringwald                     if (is_playable){
5155666fff4SMilanka Ringwald                         playable_folder_index = index;
5165666fff4SMilanka Ringwald                     }
517db79553aSMilanka Ringwald                     memcpy(folders[index].name, value, value_len);
518db79553aSMilanka Ringwald                     folders[index].name_len = value_len;
519954cc391SMilanka Ringwald                     break;
520954cc391SMilanka Ringwald                 }
521954cc391SMilanka Ringwald 
5225666fff4SMilanka Ringwald                 case AVRCP_BROWSING_MEDIA_PLAYER_ITEM:{
5235666fff4SMilanka Ringwald                     printf("Received media player item:  ");
5245666fff4SMilanka Ringwald                     uint16_t player_id = big_endian_read_16(packet, pos);
5255666fff4SMilanka Ringwald                     pos += 2;
5265666fff4SMilanka Ringwald                     avrcp_browsing_media_player_major_type_t major_type = packet[pos++];
5275666fff4SMilanka Ringwald                     avrcp_browsing_media_player_subtype_t subtype = big_endian_read_32(packet, pos);
528954cc391SMilanka Ringwald                     pos += 4;
5295666fff4SMilanka Ringwald                     status = packet[pos++];
5305666fff4SMilanka Ringwald                     uint8_t feature_bitmask[16];
5315666fff4SMilanka Ringwald                     memcpy(feature_bitmask, packet, 16);
5325666fff4SMilanka Ringwald                     pos += 16;
533*fcd55a0bSMilanka Ringwald                     printf("player ID 0x%04x, major_type %d, subtype %d, status 0x%02x\n", player_id, major_type, subtype, status);
5345666fff4SMilanka Ringwald                     media_player_items[next_media_player_item_index()].player_id = player_id;
5355666fff4SMilanka Ringwald                     break;
5365666fff4SMilanka Ringwald                 }
5375666fff4SMilanka Ringwald 
5385666fff4SMilanka Ringwald                 case AVRCP_BROWSING_MEDIA_ELEMENT_ITEM:{
5395666fff4SMilanka Ringwald                     int index = next_media_element_item_index();
5405666fff4SMilanka Ringwald                     memcpy(media_element_items[index].uid, packet+pos, 8);
5415666fff4SMilanka Ringwald                     printf("Received media element item UID (index %d): ", index);
5425666fff4SMilanka Ringwald 
5435666fff4SMilanka Ringwald                     // uint32_t media_uid_high = big_endian_read_32(packet, pos);
544954cc391SMilanka Ringwald                     pos += 4;
5455666fff4SMilanka Ringwald                     // uint32_t media_uid_low  = big_endian_read_32(packet, pos+4);
5465666fff4SMilanka Ringwald                     pos += 4;
5475666fff4SMilanka Ringwald 
548954cc391SMilanka Ringwald                     avrcp_browsing_media_type_t media_type = packet[pos++];
549954cc391SMilanka Ringwald                     uint16_t charset = big_endian_read_16(packet, pos);
550954cc391SMilanka Ringwald                     pos += 2;
551954cc391SMilanka Ringwald                     uint16_t displayable_name_length = big_endian_read_16(packet, pos);
552954cc391SMilanka Ringwald                     pos += 2;
5535666fff4SMilanka Ringwald 
5545666fff4SMilanka Ringwald                     char value[AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN];
5555666fff4SMilanka Ringwald                     memset(value, 0, AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN);
5565666fff4SMilanka Ringwald                     uint16_t value_len = btstack_min(displayable_name_length, AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN - 1);
5575666fff4SMilanka Ringwald                     memcpy(value, packet+pos, value_len);
5585666fff4SMilanka Ringwald                     memcpy(media_element_items[index].name, value, value_len);
5595666fff4SMilanka Ringwald 
560954cc391SMilanka Ringwald                     pos += displayable_name_length;
5615666fff4SMilanka Ringwald                     // printf("Media UID: 0x%08" PRIx32 "%08" PRIx32 ", media_type 0x%02x, charset 0x%02x, actual len %d, name %s\n", media_uid_high, media_uid_low, media_type, charset, value_len, value);
562954cc391SMilanka Ringwald 
5635666fff4SMilanka Ringwald                     printf_hexdump(media_element_items[index].uid, 8);
564954cc391SMilanka Ringwald                     uint8_t num_attributes = packet[pos++];
5655666fff4SMilanka Ringwald                     printf("  Media type 0x%02x, charset 0x%02x, actual len %d, name %s, num attributes %d:\n", media_type, charset, value_len, value, num_attributes);
566954cc391SMilanka Ringwald 
5675666fff4SMilanka Ringwald                     avrcp_media_item_context_t media_item_context;
568954cc391SMilanka Ringwald                     for (avrcp_media_item_iterator_init(&media_item_context, size-pos, packet+pos); avrcp_media_item_iterator_has_more(&media_item_context); avrcp_media_item_iterator_next(&media_item_context)){
569954cc391SMilanka Ringwald                         uint32_t attr_id            = avrcp_media_item_iterator_get_attr_id(&media_item_context);
570954cc391SMilanka Ringwald                         uint16_t attr_charset       = avrcp_media_item_iterator_get_attr_charset(&media_item_context);
571954cc391SMilanka Ringwald                         uint16_t attr_value_length  = avrcp_media_item_iterator_get_attr_value_len(&media_item_context);
572954cc391SMilanka Ringwald                         const uint8_t * attr_value  = avrcp_media_item_iterator_get_attr_value(&media_item_context);
573954cc391SMilanka Ringwald 
5745666fff4SMilanka Ringwald                         memset(value, 0, AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN);
5755666fff4SMilanka Ringwald                         value_len = btstack_min(attr_value_length, AVRCP_BROWSING_MAX_BROWSABLE_ITEM_NAME_LEN - 1);
5765666fff4SMilanka Ringwald                         memcpy(value, attr_value, value_len);
5775666fff4SMilanka Ringwald 
5785666fff4SMilanka Ringwald                         printf("    - attr ID 0x%08" PRIx32 ", charset 0x%02x, actual len %d, name %s\n", attr_id, attr_charset, value_len, value);
579954cc391SMilanka Ringwald                     }
580954cc391SMilanka Ringwald                     break;
581954cc391SMilanka Ringwald                 }
582954cc391SMilanka Ringwald 
5835666fff4SMilanka Ringwald                 case AVRCP_BROWSING_MEDIA_ELEMENT_ITEM_ATTRIBUTE:{
5845666fff4SMilanka Ringwald                     uint8_t num_attributes = packet[pos++];
5855666fff4SMilanka Ringwald                     printf("Num media attributes %d:\n", num_attributes);
5865666fff4SMilanka Ringwald                     avrcp_media_item_context_t media_item_context;
5875666fff4SMilanka Ringwald                     for (avrcp_media_item_iterator_init(&media_item_context, size-pos, packet+pos); avrcp_media_item_iterator_has_more(&media_item_context); avrcp_media_item_iterator_next(&media_item_context)){
5885666fff4SMilanka Ringwald                         uint32_t attr_id            = avrcp_media_item_iterator_get_attr_id(&media_item_context);
5895666fff4SMilanka Ringwald                         uint16_t attr_charset       = avrcp_media_item_iterator_get_attr_charset(&media_item_context);
5905666fff4SMilanka Ringwald                         uint16_t attr_value_length  = avrcp_media_item_iterator_get_attr_value_len(&media_item_context);
5915666fff4SMilanka Ringwald                         const uint8_t * attr_value  = avrcp_media_item_iterator_get_attr_value(&media_item_context);
5925666fff4SMilanka Ringwald                         printf("    - attr ID 0x%08" PRIx32 ", charset 0x%02x, actual len %d, name %s\n", attr_id, attr_charset, attr_value_length, attr_value);
5935666fff4SMilanka Ringwald                     }
5945666fff4SMilanka Ringwald                 }
595954cc391SMilanka Ringwald                 default:
5965666fff4SMilanka Ringwald                     printf("AVRCP browsing: unknown browsable item type 0%02x\n", data_type);
597954cc391SMilanka Ringwald                     break;
598954cc391SMilanka Ringwald             }
599954cc391SMilanka Ringwald             break;
600462aa085SMilanka Ringwald         default:
601462aa085SMilanka Ringwald             break;
602462aa085SMilanka Ringwald     }
603462aa085SMilanka Ringwald }
604462aa085SMilanka Ringwald 
605462aa085SMilanka Ringwald #ifdef HAVE_BTSTACK_STDIN
606462aa085SMilanka Ringwald static void show_usage(void){
607462aa085SMilanka Ringwald     bd_addr_t      iut_address;
608462aa085SMilanka Ringwald     gap_local_bd_addr(iut_address);
6098ee7c9daSMilanka Ringwald     printf("\n--- Bluetooth AVRCP Browsing Service Test Console %s ---\n", bd_addr_to_str(iut_address));
6108ee7c9daSMilanka Ringwald     printf("c      - AVRCP Service create connection to addr %s\n", bd_addr_to_str(device_addr));
6118ee7c9daSMilanka Ringwald     printf("C      - AVRCP Service disconnect\n");
6128ee7c9daSMilanka Ringwald     printf("e      - AVRCP Browsing Service create connection to addr %s\n", bd_addr_to_str(device_addr));
6138ee7c9daSMilanka Ringwald     printf("E      - AVRCP Browsing Service disconnect\n");
614db79553aSMilanka Ringwald 
615c0a054f6SMilanka Ringwald     printf("I      - Set first found player as addressed player\n");
616c0a054f6SMilanka Ringwald     printf("O      - Set first found player as browsed player\n");
617db79553aSMilanka Ringwald 
618c0a054f6SMilanka Ringwald     printf("p      - Get media players\n");
619c0a054f6SMilanka Ringwald     printf("Q      - Browse folders\n");
620c0a054f6SMilanka Ringwald     printf("P      - Go up one level\n");
621c0a054f6SMilanka Ringwald     printf("W      - Go down one level\n");
622c0a054f6SMilanka Ringwald     printf("T      - Browse media items\n");
623462aa085SMilanka Ringwald     printf("---\n");
624462aa085SMilanka Ringwald }
625462aa085SMilanka Ringwald #endif
626462aa085SMilanka Ringwald 
627ed0df7b2SMilanka Ringwald 
628462aa085SMilanka Ringwald #ifdef HAVE_BTSTACK_STDIN
629462aa085SMilanka Ringwald static void stdin_process(char cmd){
630462aa085SMilanka Ringwald     uint8_t status = ERROR_CODE_SUCCESS;
631462aa085SMilanka Ringwald 
632954cc391SMilanka Ringwald     if (cmd != 'a' && cmd != 'A' && cmd != 'c' && cmd != 'C'){
633954cc391SMilanka Ringwald         if (browsing_query_active){
634954cc391SMilanka Ringwald             printf("Query active, try later!\n");
635954cc391SMilanka Ringwald             return;
636954cc391SMilanka Ringwald         }
637954cc391SMilanka Ringwald     }
638954cc391SMilanka Ringwald 
639462aa085SMilanka Ringwald     switch (cmd){
6405666fff4SMilanka Ringwald         case 'b':
6415666fff4SMilanka Ringwald             status = a2dp_sink_establish_stream(device_addr, a2dp_local_seid, &a2dp_cid);
6425666fff4SMilanka Ringwald             printf(" - Create AVDTP connection to addr %s, and local seid %d, expected cid 0x%02x.\n", bd_addr_to_str(device_addr), a2dp_local_seid, a2dp_cid);
6435666fff4SMilanka Ringwald             break;
6445666fff4SMilanka Ringwald         case 'B':
6455666fff4SMilanka Ringwald             printf(" - AVDTP disconnect from addr %s.\n", bd_addr_to_str(device_addr));
6465666fff4SMilanka Ringwald             a2dp_sink_disconnect(a2dp_cid);
6475666fff4SMilanka Ringwald             break;
6485666fff4SMilanka Ringwald 
649c0a054f6SMilanka Ringwald         case 'c':
6508ee7c9daSMilanka Ringwald             printf(" - Connect to AVRCP Service on addr %s.\n", bd_addr_to_str(device_addr));
651fe10780bSMilanka Ringwald             status = avrcp_connect(device_addr, &avrcp_cid);
652462aa085SMilanka Ringwald             break;
653c0a054f6SMilanka Ringwald         case 'C':
654247956eaSMilanka Ringwald             if (avrcp_connected){
6558ee7c9daSMilanka Ringwald                 printf(" - AVRCP Service disconnect from addr %s.\n", bd_addr_to_str(device_addr));
656fe10780bSMilanka Ringwald                 status = avrcp_disconnect(avrcp_cid);
657247956eaSMilanka Ringwald                 break;
658247956eaSMilanka Ringwald             }
6598ee7c9daSMilanka Ringwald             printf("AVRCP Service already disconnected\n");
660247956eaSMilanka Ringwald             break;
661247956eaSMilanka Ringwald 
662c0a054f6SMilanka Ringwald         case 'e':
663462aa085SMilanka Ringwald             if (!avrcp_connected) {
6648ee7c9daSMilanka Ringwald                 printf(" You must first connect to AVRCP Service on addr %s.\n", bd_addr_to_str(device_addr));
665462aa085SMilanka Ringwald                 break;
666462aa085SMilanka Ringwald             }
6678ee7c9daSMilanka Ringwald             printf(" - Connect to AVRCP Browsing Service at addr %s.\n", bd_addr_to_str(device_addr));
6683121b998SMilanka Ringwald             status = avrcp_browsing_connect(device_addr, ertm_buffer, sizeof(ertm_buffer), &ertm_config, &browsing_cid);
669462aa085SMilanka Ringwald             break;
670c0a054f6SMilanka Ringwald         case 'E':
6715666fff4SMilanka Ringwald             if (browsing_connected){
6728ee7c9daSMilanka Ringwald                 printf(" - AVRCP Browsing Service disconnect from addr %s.\n", bd_addr_to_str(device_addr));
6738ee7c9daSMilanka Ringwald                 status = avrcp_browsing_disconnect(browsing_cid);
674462aa085SMilanka Ringwald                 break;
675247956eaSMilanka Ringwald             }
6768ee7c9daSMilanka Ringwald             printf("AVRCP Browsing Service already disconnected\n");
677247956eaSMilanka Ringwald             break;
678462aa085SMilanka Ringwald         case '\n':
679462aa085SMilanka Ringwald         case '\r':
680462aa085SMilanka Ringwald             break;
681db79553aSMilanka Ringwald 
682db79553aSMilanka Ringwald         default:
6835666fff4SMilanka Ringwald             if (!browsing_connected){
684db79553aSMilanka Ringwald                 show_usage();
685db79553aSMilanka Ringwald                 break;
686db79553aSMilanka Ringwald             }
687db79553aSMilanka Ringwald 
688db79553aSMilanka Ringwald             switch (cmd) {
689c0a054f6SMilanka Ringwald                 case 'I':
6905666fff4SMilanka Ringwald                     if (media_player_item_index < 0) {
6918ee7c9daSMilanka Ringwald                         printf("AVRCP Browsing:Get media players first\n");
692db79553aSMilanka Ringwald                         break;
693db79553aSMilanka Ringwald                     }
6948ee7c9daSMilanka Ringwald                     printf("AVRCP Browsing:Set addressed player\n");
6955666fff4SMilanka Ringwald                     status = avrcp_controller_set_addressed_player(avrcp_cid, media_player_items[0].player_id);
696db79553aSMilanka Ringwald                     break;
697c0a054f6SMilanka Ringwald                 case 'O':
6985666fff4SMilanka Ringwald                     if (media_player_item_index < 0) {
6998ee7c9daSMilanka Ringwald                         printf("AVRCP Browsing:Get media players first\n");
700db79553aSMilanka Ringwald                         break;
701db79553aSMilanka Ringwald                     }
702db79553aSMilanka Ringwald                     printf("Set browsed player\n");
7035666fff4SMilanka Ringwald                     status = avrcp_browsing_controller_set_browsed_player(browsing_cid, media_player_items[0].player_id);
704db79553aSMilanka Ringwald                     break;
705c0a054f6SMilanka Ringwald                 case 'p':
706db79553aSMilanka Ringwald                     printf("AVRCP Browsing: get media players\n");
7075666fff4SMilanka Ringwald                     media_player_item_index = -1;
7082f511ffbSMilanka Ringwald                     status = avrcp_browsing_controller_get_media_players(browsing_cid, 0, 0xFFFFFFFF, AVRCP_MEDIA_ATTR_ALL);
709db79553aSMilanka Ringwald                     break;
710c0a054f6SMilanka Ringwald                 case 'Q':
711db79553aSMilanka Ringwald                     printf("AVRCP Browsing: browse folders\n");
712db79553aSMilanka Ringwald                     folder_index = -1;
7132f511ffbSMilanka Ringwald                     status = avrcp_browsing_controller_browse_file_system(browsing_cid, 0, 0xFFFFFFFF, AVRCP_MEDIA_ATTR_ALL);
714db79553aSMilanka Ringwald                     break;
715c0a054f6SMilanka Ringwald                 case 'P':
716db79553aSMilanka Ringwald                     printf("AVRCP Browsing: browse media items\n");
7172f511ffbSMilanka Ringwald                     avrcp_browsing_controller_browse_media(browsing_cid, 0, 0xFFFFFFFF, AVRCP_MEDIA_ATTR_ALL);
718db79553aSMilanka Ringwald                     break;
719c0a054f6SMilanka Ringwald                 case 'W':
720e5b24613SMilanka Ringwald                     printf("AVRCP Browsing: go up one level\n");
721e5b24613SMilanka Ringwald                     status = avrcp_browsing_controller_go_up_one_level(browsing_cid);
722db79553aSMilanka Ringwald                     folder_index = -1;
723db79553aSMilanka Ringwald                     break;
724c0a054f6SMilanka Ringwald                 case 'T':
725db79553aSMilanka Ringwald                     if (folder_index < 0 && !parent_folder_set){
726db79553aSMilanka Ringwald                         printf("AVRCP Browsing: no folders available\n");
727db79553aSMilanka Ringwald                         break;
728db79553aSMilanka Ringwald                     }
729db79553aSMilanka Ringwald                     if (!parent_folder_set){
730db79553aSMilanka Ringwald                         parent_folder_set = 1;
731db79553aSMilanka Ringwald                         memcpy(parent_folder_name, folders[0].name, folders[0].name_len);
732db79553aSMilanka Ringwald                         memcpy(parent_folder_uid, folders[0].uid, 8);
733db79553aSMilanka Ringwald                     }
734db79553aSMilanka Ringwald                     printf("AVRCP Browsing: go down one level of %s\n", (char *)parent_folder_name);
735db79553aSMilanka Ringwald                     status = avrcp_browsing_controller_go_down_one_level(browsing_cid, parent_folder_uid);
736db79553aSMilanka Ringwald                     folder_index = -1;
737db79553aSMilanka Ringwald                     break;
738462aa085SMilanka Ringwald                 default:
739462aa085SMilanka Ringwald                     show_usage();
740db79553aSMilanka Ringwald                     break;
741db79553aSMilanka Ringwald             }
742db79553aSMilanka Ringwald             break;
743462aa085SMilanka Ringwald     }
744ed0df7b2SMilanka Ringwald 
745462aa085SMilanka Ringwald     if (status != ERROR_CODE_SUCCESS){
7468ee7c9daSMilanka Ringwald         printf("AVRCP Browsing: Could not perform command, status 0x%2x\n", status);
747462aa085SMilanka Ringwald     }
748462aa085SMilanka Ringwald }
749462aa085SMilanka Ringwald #endif
750462aa085SMilanka Ringwald /* EXAMPLE_END */
751