101dc6e35SMilanka Ringwald /* 201dc6e35SMilanka Ringwald * Copyright (C) 2016 BlueKitchen GmbH 301dc6e35SMilanka Ringwald * 401dc6e35SMilanka Ringwald * Redistribution and use in source and binary forms, with or without 501dc6e35SMilanka Ringwald * modification, are permitted provided that the following conditions 601dc6e35SMilanka Ringwald * are met: 701dc6e35SMilanka Ringwald * 801dc6e35SMilanka Ringwald * 1. Redistributions of source code must retain the above copyright 901dc6e35SMilanka Ringwald * notice, this list of conditions and the following disclaimer. 1001dc6e35SMilanka Ringwald * 2. Redistributions in binary form must reproduce the above copyright 1101dc6e35SMilanka Ringwald * notice, this list of conditions and the following disclaimer in the 1201dc6e35SMilanka Ringwald * documentation and/or other materials provided with the distribution. 1301dc6e35SMilanka Ringwald * 3. Neither the name of the copyright holders nor the names of 1401dc6e35SMilanka Ringwald * contributors may be used to endorse or promote products derived 1501dc6e35SMilanka Ringwald * from this software without specific prior written permission. 1601dc6e35SMilanka Ringwald * 4. Any redistribution, use, or modification is done solely for 1701dc6e35SMilanka Ringwald * personal benefit and not for any commercial purpose or for 1801dc6e35SMilanka Ringwald * monetary gain. 1901dc6e35SMilanka Ringwald * 2001dc6e35SMilanka Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 2101dc6e35SMilanka Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 2201dc6e35SMilanka 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, 2501dc6e35SMilanka Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 2601dc6e35SMilanka Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 2701dc6e35SMilanka Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 2801dc6e35SMilanka Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 2901dc6e35SMilanka Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 3001dc6e35SMilanka Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 3101dc6e35SMilanka Ringwald * SUCH DAMAGE. 3201dc6e35SMilanka Ringwald * 3301dc6e35SMilanka Ringwald * Please inquire about commercial licensing options at 3401dc6e35SMilanka Ringwald * [email protected] 3501dc6e35SMilanka Ringwald * 3601dc6e35SMilanka Ringwald */ 3701dc6e35SMilanka Ringwald 38e501bae0SMatthias Ringwald #define BTSTACK_FILE__ "avrcp_target.c" 3901dc6e35SMilanka Ringwald 4001dc6e35SMilanka Ringwald #include <stdint.h> 413cfa4086SMatthias Ringwald #include <stdio.h> 4201dc6e35SMilanka Ringwald #include <string.h> 43adaba9f3SMatthias Ringwald #include <inttypes.h> 4401dc6e35SMilanka Ringwald 4501dc6e35SMilanka Ringwald #include "classic/avrcp.h" 467c76cd61SMatthias Ringwald #include "classic/avrcp_target.h" 477c76cd61SMatthias Ringwald 487c76cd61SMatthias Ringwald #include "bluetooth_sdp.h" 497c76cd61SMatthias Ringwald #include "btstack_debug.h" 507c76cd61SMatthias Ringwald #include "btstack_event.h" 517c76cd61SMatthias Ringwald #include "btstack_util.h" 527c76cd61SMatthias Ringwald #include "l2cap.h" 5301dc6e35SMilanka Ringwald 54aeb99916SMilanka Ringwald #include <stdio.h> 55d1207cd8SMilanka Ringwald #define AVRCP_ATTR_HEADER_LEN 8 56d1207cd8SMilanka Ringwald 57d1207cd8SMilanka Ringwald static const uint8_t AVRCP_NOTIFICATION_TRACK_SELECTED[] = {0,0,0,0,0,0,0,0}; 58d1207cd8SMilanka Ringwald static const uint8_t AVRCP_NOTIFICATION_TRACK_NOT_SELECTED[] = {0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF}; 59d1207cd8SMilanka Ringwald 60141dff40SMatthias Ringwald static const char * avrcp_default_target_service_name = "AVRCP Target"; 61141dff40SMatthias Ringwald static const char * avrcp_default_target_service_provider_name = "BlueKitchen"; 62141dff40SMatthias Ringwald 63e3d57ee2SMilanka Ringwald avrcp_context_t avrcp_target_context; 6401dc6e35SMilanka Ringwald 65aeb99916SMilanka Ringwald static uint32_t default_companies[] = { 66aeb99916SMilanka Ringwald 0x581900 //BT SIG registered CompanyID 67aeb99916SMilanka Ringwald }; 68aeb99916SMilanka Ringwald 6911014891SMilanka Ringwald static int avrcp_target_supports_browsing(uint16_t target_supported_features){ 70e2f25417SMilanka Ringwald return target_supported_features & AVRCP_FEATURE_MASK_BROWSING; 7111014891SMilanka Ringwald } 7211014891SMilanka Ringwald 734f0111ebSMilanka Ringwald void avrcp_target_create_sdp_record(uint8_t * service, uint32_t service_record_handle, uint16_t supported_features, const char * service_name, const char * service_provider_name){ 74141dff40SMatthias Ringwald if (service_name == NULL){ 75141dff40SMatthias Ringwald service_name = avrcp_default_target_service_name; 76141dff40SMatthias Ringwald } 77141dff40SMatthias Ringwald if (service_provider_name == NULL){ 78141dff40SMatthias Ringwald service_provider_name = avrcp_default_target_service_provider_name; 79141dff40SMatthias Ringwald } 80c7831204SMatthias Ringwald avrcp_create_sdp_record(false, service, service_record_handle, avrcp_target_supports_browsing(supported_features), supported_features, service_name, service_provider_name); 8101dc6e35SMilanka Ringwald } 8201dc6e35SMilanka Ringwald 83319131f8SMatthias Ringwald static void 84319131f8SMatthias Ringwald avrcp_target_emit_operation(btstack_packet_handler_t callback, uint16_t avrcp_cid, avrcp_operation_id_t operation_id, 85319131f8SMatthias Ringwald bool button_pressed, uint8_t operands_length, uint8_t operand) { 86be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 87be4cc80aSMilanka Ringwald 88319131f8SMatthias Ringwald uint8_t event[9]; 89831d3fd5SMilanka Ringwald int pos = 0; 90831d3fd5SMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META; 91831d3fd5SMilanka Ringwald event[pos++] = sizeof(event) - 2; 92831d3fd5SMilanka Ringwald event[pos++] = AVRCP_SUBEVENT_OPERATION; 93831d3fd5SMilanka Ringwald little_endian_store_16(event, pos, avrcp_cid); 94831d3fd5SMilanka Ringwald pos += 2; 95831d3fd5SMilanka Ringwald event[pos++] = operation_id; 96319131f8SMatthias Ringwald event[pos++] = button_pressed ? 1 : 0; 97831d3fd5SMilanka Ringwald event[pos++] = operands_length; 98831d3fd5SMilanka Ringwald event[pos++] = operand; 99831d3fd5SMilanka Ringwald (*callback)(HCI_EVENT_PACKET, 0, event, sizeof(event)); 100831d3fd5SMilanka Ringwald } 101831d3fd5SMilanka Ringwald 1020ec79bd0SMilanka Ringwald static void avrcp_target_emit_volume_changed(btstack_packet_handler_t callback, uint16_t avrcp_cid, uint8_t absolute_volume){ 103be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 104be4cc80aSMilanka Ringwald 1050ec79bd0SMilanka Ringwald uint8_t event[7]; 1060ec79bd0SMilanka Ringwald int offset = 0; 1070ec79bd0SMilanka Ringwald event[offset++] = HCI_EVENT_AVRCP_META; 1080ec79bd0SMilanka Ringwald event[offset++] = sizeof(event) - 2; 1090ec79bd0SMilanka Ringwald event[offset++] = AVRCP_SUBEVENT_NOTIFICATION_VOLUME_CHANGED; 1100ec79bd0SMilanka Ringwald little_endian_store_16(event, offset, avrcp_cid); 1110ec79bd0SMilanka Ringwald offset += 2; 1120ec79bd0SMilanka Ringwald event[offset++] = AVRCP_CTYPE_NOTIFY; 1130ec79bd0SMilanka Ringwald event[offset++] = absolute_volume; 1140ec79bd0SMilanka Ringwald (*callback)(HCI_EVENT_PACKET, 0, event, sizeof(event)); 1150ec79bd0SMilanka Ringwald } 1160ec79bd0SMilanka Ringwald 117c045af99SMilanka Ringwald static void avrcp_target_emit_respond_vendor_dependent_query(btstack_packet_handler_t callback, uint16_t avrcp_cid, uint8_t subevent_id){ 118be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 119be4cc80aSMilanka Ringwald 120c045af99SMilanka Ringwald uint8_t event[5]; 121e0bbf3edSMilanka Ringwald int pos = 0; 122e0bbf3edSMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META; 123e0bbf3edSMilanka Ringwald event[pos++] = sizeof(event) - 2; 124e0bbf3edSMilanka Ringwald event[pos++] = subevent_id; 125e0bbf3edSMilanka Ringwald little_endian_store_16(event, pos, avrcp_cid); 1264b338011SMilanka Ringwald (*callback)(HCI_EVENT_PACKET, 0, event, sizeof(event)); 1274b338011SMilanka Ringwald } 1284b338011SMilanka Ringwald 129*1e4db407SMilanka Ringwald static void avrcp_target_emit_respond_play_item(btstack_packet_handler_t callback, uint16_t avrcp_cid, uint16_t uid_counter, avrcp_browsing_scope_t scope, uint8_t * uid){ 130*1e4db407SMilanka Ringwald btstack_assert(callback != NULL); 131*1e4db407SMilanka Ringwald 132*1e4db407SMilanka Ringwald uint8_t event[16]; 133*1e4db407SMilanka Ringwald int pos = 0; 134*1e4db407SMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META; 135*1e4db407SMilanka Ringwald event[pos++] = sizeof(event) - 2; 136*1e4db407SMilanka Ringwald event[pos++] = AVRCP_SUBEVENT_PLAY_ITEM; 137*1e4db407SMilanka Ringwald little_endian_store_16(event, pos, avrcp_cid); 138*1e4db407SMilanka Ringwald pos += 2; 139*1e4db407SMilanka Ringwald little_endian_store_16(event, pos, uid_counter); 140*1e4db407SMilanka Ringwald pos += 2; 141*1e4db407SMilanka Ringwald event[pos++] = (uint8_t) scope; 142*1e4db407SMilanka Ringwald memcpy(&event[pos], uid, 8); 143*1e4db407SMilanka Ringwald pos += 8; 144*1e4db407SMilanka Ringwald 145*1e4db407SMilanka Ringwald (*callback)(HCI_EVENT_PACKET, 0, event, pos); 146*1e4db407SMilanka Ringwald } 147*1e4db407SMilanka Ringwald 1485e845fceSMatthias Ringwald // returns number of bytes stored 149aeb99916SMilanka Ringwald static uint16_t avrcp_target_pack_single_element_header(uint8_t * buffer, avrcp_media_attribute_id_t attr_id, uint16_t attr_value_size){ 150aeb99916SMilanka Ringwald btstack_assert(attr_id > AVRCP_MEDIA_ATTR_ALL); 15120c6ab95SMilanka Ringwald btstack_assert(attr_id < AVRCP_MEDIA_ATTR_NUM); 152aeb99916SMilanka Ringwald uint16_t pos = 0; 153aeb99916SMilanka Ringwald big_endian_store_32(buffer, pos, attr_id); 154aeb99916SMilanka Ringwald big_endian_store_16(buffer, pos + 4, RFC2978_CHARSET_MIB_UTF8); 155aeb99916SMilanka Ringwald big_endian_store_16(buffer, pos + 6, attr_value_size); 1565e845fceSMatthias Ringwald return 8; 1575e845fceSMatthias Ringwald } 1585e845fceSMatthias Ringwald 159aeb99916SMilanka Ringwald static uint16_t avrcp_now_playing_info_attr_id_value_len(avrcp_connection_t * connection, avrcp_media_attribute_id_t attr_id){ 160aeb99916SMilanka Ringwald char buffer[AVRCP_MAX_ATTRIBUTE_SIZE]; 161aeb99916SMilanka Ringwald uint16_t str_len; 1625e3d4d2bSMilanka Ringwald switch (attr_id) { 1632f511ffbSMilanka Ringwald case AVRCP_MEDIA_ATTR_ALL: 164d1207cd8SMilanka Ringwald case AVRCP_MEDIA_ATTR_NONE: 165aeb99916SMilanka Ringwald return 0; 1665e3d4d2bSMilanka Ringwald case AVRCP_MEDIA_ATTR_TRACK: 1672e2ef1f1SMatthias Ringwald str_len = btstack_snprintf_assert_complete(buffer, sizeof(buffer), "%" PRIu32, connection->target_track_nr); 1685e3d4d2bSMilanka Ringwald break; 169864d08b0SMilanka Ringwald case AVRCP_MEDIA_ATTR_TOTAL_NUM_ITEMS: 1702e2ef1f1SMatthias Ringwald str_len = btstack_snprintf_assert_complete(buffer, sizeof(buffer), "%" PRIu32, connection->target_total_tracks); 1715e3d4d2bSMilanka Ringwald break; 1722f511ffbSMilanka Ringwald case AVRCP_MEDIA_ATTR_SONG_LENGTH_MS: 1732e2ef1f1SMatthias Ringwald str_len = btstack_snprintf_assert_complete(buffer, sizeof(buffer), "%" PRIu32, connection->target_song_length_ms); 1745e3d4d2bSMilanka Ringwald break; 1755e3d4d2bSMilanka Ringwald default: 176393f5724SMilanka Ringwald str_len = connection->target_now_playing_info[(uint16_t)attr_id - 1].len; 1775e3d4d2bSMilanka Ringwald break; 1785e3d4d2bSMilanka Ringwald } 179aeb99916SMilanka Ringwald return str_len; 1805e3d4d2bSMilanka Ringwald } 1815e3d4d2bSMilanka Ringwald 182aeb99916SMilanka Ringwald static uint16_t avrcp_now_playing_info_value_len_with_headers(avrcp_connection_t * connection){ 183aeb99916SMilanka Ringwald uint16_t playing_info_len = 0; 184d1207cd8SMilanka Ringwald 185aeb99916SMilanka Ringwald uint8_t i; 18620c6ab95SMilanka Ringwald for ( i = (uint8_t)AVRCP_MEDIA_ATTR_ALL + 1; i < (uint8_t) AVRCP_MEDIA_ATTR_NUM; i++){ 187aeb99916SMilanka Ringwald avrcp_media_attribute_id_t attr_id = (avrcp_media_attribute_id_t) i; 1885e3d4d2bSMilanka Ringwald 189393f5724SMilanka Ringwald if ((connection->target_now_playing_info_attr_bitmap & (1 << attr_id)) == 0) { 190aeb99916SMilanka Ringwald continue; 191aeb99916SMilanka Ringwald } 192aeb99916SMilanka Ringwald 193aeb99916SMilanka Ringwald switch (attr_id) { 194aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_ALL: 195aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_NONE: 196aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_DEFAULT_COVER_ART: 197aeb99916SMilanka Ringwald break; 198aeb99916SMilanka Ringwald default: 199aeb99916SMilanka Ringwald playing_info_len += AVRCP_ATTR_HEADER_LEN + avrcp_now_playing_info_attr_id_value_len(connection, attr_id); 200aeb99916SMilanka Ringwald break; 201aeb99916SMilanka Ringwald } 202aeb99916SMilanka Ringwald } 203ccfcd81eSMilanka Ringwald // for total num bytes that of the attributes + headers 204ccfcd81eSMilanka Ringwald playing_info_len += 1; 205aeb99916SMilanka Ringwald return playing_info_len; 206aeb99916SMilanka Ringwald } 207aeb99916SMilanka Ringwald 208aeb99916SMilanka Ringwald static uint8_t * avrcp_get_attribute_value_from_u32(avrcp_connection_t * connection, uint32_t value, uint16_t * num_bytes_to_copy){ 209aeb99916SMilanka Ringwald *num_bytes_to_copy = 0; 210aeb99916SMilanka Ringwald 211aeb99916SMilanka Ringwald if (connection->attribute_value_len == 0){ 212ab2445a0SMatthias Ringwald // "4294967296" = 10 chars + \0 2132e2ef1f1SMatthias Ringwald connection->attribute_value_len = btstack_snprintf_assert_complete((char *)connection->attribute_value, 11, "%" PRIu32, value); 214aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 215aeb99916SMilanka Ringwald } 216aeb99916SMilanka Ringwald *num_bytes_to_copy = connection->attribute_value_len - connection->attribute_value_offset; 217aeb99916SMilanka Ringwald return connection->attribute_value + connection->attribute_value_offset; 218aeb99916SMilanka Ringwald } 219aeb99916SMilanka Ringwald 220aeb99916SMilanka Ringwald static uint8_t * avrcp_get_next_value_fragment_for_attribute_id(avrcp_connection_t * connection, avrcp_media_attribute_id_t attr_id, uint16_t * num_bytes_to_copy){ 221aeb99916SMilanka Ringwald switch (attr_id){ 222aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_TRACK: 223393f5724SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->target_track_nr, num_bytes_to_copy); 224aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_TOTAL_NUM_ITEMS: 225393f5724SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->target_total_tracks, num_bytes_to_copy); 226aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_SONG_LENGTH_MS: 227393f5724SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->target_song_length_ms, num_bytes_to_copy); 228aeb99916SMilanka Ringwald default: 229aeb99916SMilanka Ringwald break; 230aeb99916SMilanka Ringwald } 231aeb99916SMilanka Ringwald int attr_index = attr_id - 1; 232aeb99916SMilanka Ringwald if (connection->attribute_value_len == 0){ 233aeb99916SMilanka Ringwald connection->attribute_value_len = avrcp_now_playing_info_attr_id_value_len(connection, attr_id); 234aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 235aeb99916SMilanka Ringwald } 236393f5724SMilanka Ringwald *num_bytes_to_copy = connection->target_now_playing_info[attr_index].len - connection->attribute_value_offset; 237393f5724SMilanka Ringwald return (uint8_t *) (connection->target_now_playing_info[attr_index].value + connection->attribute_value_offset); 238aeb99916SMilanka Ringwald } 239aeb99916SMilanka Ringwald 240aeb99916SMilanka Ringwald // TODO Review 241aeb99916SMilanka Ringwald static uint16_t avrcp_store_avctp_now_playing_info_fragment(avrcp_connection_t * connection, uint16_t packet_size, uint8_t * packet){ 242aeb99916SMilanka Ringwald uint16_t num_free_bytes = packet_size; 243aeb99916SMilanka Ringwald 244aeb99916SMilanka Ringwald uint16_t bytes_stored = 0; 245aeb99916SMilanka Ringwald 246aeb99916SMilanka Ringwald while ((num_free_bytes > 0) && (connection->next_attr_id <= AVRCP_MEDIA_ATTR_SONG_LENGTH_MS)){ 247393f5724SMilanka Ringwald if ((connection->target_now_playing_info_attr_bitmap & (1 << (uint8_t)connection->next_attr_id)) == 0) { 24892ae15eaSMilanka Ringwald connection->next_attr_id = (avrcp_media_attribute_id_t) (((int) connection->next_attr_id) + 1); 249aeb99916SMilanka Ringwald continue; 250aeb99916SMilanka Ringwald } 251aeb99916SMilanka Ringwald 252ccfcd81eSMilanka Ringwald // prepare attribute value 253ccfcd81eSMilanka Ringwald uint16_t num_bytes_to_copy; 254ccfcd81eSMilanka Ringwald uint8_t * attr_value_with_offset = avrcp_get_next_value_fragment_for_attribute_id(connection, 255ccfcd81eSMilanka Ringwald connection->next_attr_id, 256ccfcd81eSMilanka Ringwald &num_bytes_to_copy); 257ccfcd81eSMilanka Ringwald 258aeb99916SMilanka Ringwald // store header 259aeb99916SMilanka Ringwald if (connection->attribute_value_offset == 0){ 260aeb99916SMilanka Ringwald // pack the whole attribute value header 261aeb99916SMilanka Ringwald if (connection->parser_attribute_header_pos == 0) { 262aeb99916SMilanka Ringwald avrcp_target_pack_single_element_header(connection->parser_attribute_header, connection->next_attr_id, 263aeb99916SMilanka Ringwald connection->attribute_value_len); 264ccfcd81eSMilanka Ringwald } 265aeb99916SMilanka Ringwald } 266aeb99916SMilanka Ringwald 267aeb99916SMilanka Ringwald if (connection->parser_attribute_header_pos < AVRCP_ATTRIBUTE_HEADER_LEN){ 268aeb99916SMilanka Ringwald uint16_t num_header_bytes_to_store = btstack_min(num_free_bytes, AVRCP_ATTRIBUTE_HEADER_LEN - connection->parser_attribute_header_pos); 269aeb99916SMilanka Ringwald memcpy(packet + bytes_stored, connection->parser_attribute_header + connection->parser_attribute_header_pos, num_header_bytes_to_store); 270aeb99916SMilanka Ringwald connection->parser_attribute_header_pos += num_header_bytes_to_store; 271aeb99916SMilanka Ringwald bytes_stored += num_header_bytes_to_store; 272aeb99916SMilanka Ringwald num_free_bytes -= num_header_bytes_to_store; 273aeb99916SMilanka Ringwald connection->data_offset += num_header_bytes_to_store; 274aeb99916SMilanka Ringwald 275aeb99916SMilanka Ringwald if (num_free_bytes == 0){ 276aeb99916SMilanka Ringwald continue; 277aeb99916SMilanka Ringwald } 278aeb99916SMilanka Ringwald } 279aeb99916SMilanka Ringwald 280aeb99916SMilanka Ringwald // store value 281aeb99916SMilanka Ringwald uint16_t num_attr_value_bytes_to_store = btstack_min(num_free_bytes, connection->attribute_value_len - connection->attribute_value_offset); 282aeb99916SMilanka Ringwald memcpy(packet + bytes_stored, attr_value_with_offset, num_attr_value_bytes_to_store); 283aeb99916SMilanka Ringwald bytes_stored += num_attr_value_bytes_to_store; 284aeb99916SMilanka Ringwald num_free_bytes -= num_attr_value_bytes_to_store; 285aeb99916SMilanka Ringwald connection->attribute_value_offset += num_attr_value_bytes_to_store; 286aeb99916SMilanka Ringwald connection->data_offset += num_attr_value_bytes_to_store; 287aeb99916SMilanka Ringwald 288aeb99916SMilanka Ringwald if (connection->attribute_value_offset == connection->attribute_value_len){ 289aeb99916SMilanka Ringwald // C++ compatible version of connection->next_attr_id++ 290aeb99916SMilanka Ringwald connection->next_attr_id = (avrcp_media_attribute_id_t) (((int) connection->next_attr_id) + 1); 291aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 292aeb99916SMilanka Ringwald connection->attribute_value_len = 0; 293aeb99916SMilanka Ringwald connection->parser_attribute_header_pos = 0; 294aeb99916SMilanka Ringwald } 295aeb99916SMilanka Ringwald } 296aeb99916SMilanka Ringwald return bytes_stored; 297aeb99916SMilanka Ringwald } 298aeb99916SMilanka Ringwald 299aeb99916SMilanka Ringwald static void avrcp_send_response_with_avctp_fragmentation(avrcp_connection_t * connection){ 3005e3d4d2bSMilanka Ringwald l2cap_reserve_packet_buffer(); 3015e3d4d2bSMilanka Ringwald uint8_t * packet = l2cap_get_outgoing_buffer(); 3025e3d4d2bSMilanka Ringwald 3035e3d4d2bSMilanka Ringwald // transport header 3045e3d4d2bSMilanka Ringwald // Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 305d1207cd8SMilanka Ringwald 306aeb99916SMilanka Ringwald uint16_t max_payload_size; 307aeb99916SMilanka Ringwald connection->avctp_packet_type = avctp_get_packet_type(connection, &max_payload_size); 308393f5724SMilanka Ringwald connection->avrcp_packet_type = avrcp_get_packet_type(connection); 309d1207cd8SMilanka Ringwald 310aeb99916SMilanka Ringwald // AVCTP header 311aeb99916SMilanka Ringwald // transport header : transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 312aeb99916SMilanka Ringwald uint16_t pos = 0; 313aeb99916SMilanka Ringwald packet[pos++] = (connection->transaction_id << 4) | (connection->avctp_packet_type << 2) | (AVRCP_RESPONSE_FRAME << 1) | 0; 314aeb99916SMilanka Ringwald 31518f5a148SMilanka Ringwald uint16_t param_len = connection->data_len; 316aeb99916SMilanka Ringwald 317aeb99916SMilanka Ringwald if (connection->avctp_packet_type == AVCTP_START_PACKET){ 31818f5a148SMilanka Ringwald uint16_t max_frame_size = btstack_min(connection->l2cap_mtu, AVRCP_MAX_AV_C_MESSAGE_FRAME_SIZE); 319aeb99916SMilanka Ringwald // first packet: max_payload_size 320aeb99916SMilanka Ringwald // rest packets 321aeb99916SMilanka Ringwald uint16_t num_payload_bytes = param_len - max_payload_size; 322aeb99916SMilanka Ringwald uint16_t frame_size_for_continue_packet = max_frame_size - avctp_get_num_bytes_for_header(AVCTP_CONTINUE_PACKET); 323aeb99916SMilanka Ringwald uint16_t num_avctp_packets = (num_payload_bytes + frame_size_for_continue_packet - 1)/frame_size_for_continue_packet + 1; 324ab2445a0SMatthias Ringwald btstack_assert(num_avctp_packets <= 255); 325ab2445a0SMatthias Ringwald packet[pos++] = (uint8_t) num_avctp_packets; 326aeb99916SMilanka Ringwald } 327aeb99916SMilanka Ringwald 328aeb99916SMilanka Ringwald uint16_t bytes_stored = 0; 329aeb99916SMilanka Ringwald uint8_t i; 330aeb99916SMilanka Ringwald 331aeb99916SMilanka Ringwald switch (connection->avctp_packet_type) { 332aeb99916SMilanka Ringwald case AVCTP_SINGLE_PACKET: 333aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 3345e3d4d2bSMilanka Ringwald // Profile IDentifier (PID) 3355e3d4d2bSMilanka Ringwald packet[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8; 3365e3d4d2bSMilanka Ringwald packet[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF; 337aeb99916SMilanka Ringwald 338aeb99916SMilanka Ringwald // AVRCP message 3395e3d4d2bSMilanka Ringwald // command_type 3405e3d4d2bSMilanka Ringwald packet[pos++] = connection->command_type; 3415e3d4d2bSMilanka Ringwald // subunit_type | subunit ID 3425e3d4d2bSMilanka Ringwald packet[pos++] = (connection->subunit_type << 3) | connection->subunit_id; 3435e3d4d2bSMilanka Ringwald // opcode 3445e3d4d2bSMilanka Ringwald packet[pos++] = (uint8_t) connection->command_opcode; 3450ec79bd0SMilanka Ringwald 346aeb99916SMilanka Ringwald switch (connection->command_opcode) { 347aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_VENDOR_DEPENDENT: 348aeb99916SMilanka Ringwald big_endian_store_24(packet, pos, connection->company_id); 349aeb99916SMilanka Ringwald pos += 3; 350aeb99916SMilanka Ringwald packet[pos++] = connection->pdu_id; 351aeb99916SMilanka Ringwald // AVRCP packet type 3520ec79bd0SMilanka Ringwald 353393f5724SMilanka Ringwald packet[pos++] = (uint8_t)connection->avrcp_packet_type; 354aeb99916SMilanka Ringwald // parameter length 355aeb99916SMilanka Ringwald big_endian_store_16(packet, pos, param_len); 356aeb99916SMilanka Ringwald pos += 2; 357aeb99916SMilanka Ringwald 358aeb99916SMilanka Ringwald switch (connection->pdu_id) { 359aeb99916SMilanka Ringwald // message is small enough to fit the single packet, no need for extra check 360aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_CAPABILITIES: 361ef07fa3cSMilanka Ringwald // capability ID 362aeb99916SMilanka Ringwald packet[pos++] = connection->data[0]; 363ef07fa3cSMilanka Ringwald // num_capabilities 364aeb99916SMilanka Ringwald packet[pos++] = connection->data[1]; 365aeb99916SMilanka Ringwald 366aeb99916SMilanka Ringwald switch ((avrcp_capability_id_t) connection->data[0]) { 367aeb99916SMilanka Ringwald case AVRCP_CAPABILITY_ID_EVENT: 368aeb99916SMilanka Ringwald for (i = (uint8_t) AVRCP_NOTIFICATION_EVENT_FIRST_INDEX; 369aeb99916SMilanka Ringwald i < (uint8_t) AVRCP_NOTIFICATION_EVENT_LAST_INDEX; i++) { 370393f5724SMilanka Ringwald if ((connection->notifications_supported_by_target & (1 << i)) == 0) { 371aeb99916SMilanka Ringwald continue; 372aeb99916SMilanka Ringwald } 373aeb99916SMilanka Ringwald packet[pos++] = i; 374aeb99916SMilanka Ringwald } 375aeb99916SMilanka Ringwald break; 376aeb99916SMilanka Ringwald case AVRCP_CAPABILITY_ID_COMPANY: 377aeb99916SMilanka Ringwald // use Bluetooth SIG as default company 378aeb99916SMilanka Ringwald for (i = 0; i < connection->data[1]; i++) { 379aeb99916SMilanka Ringwald little_endian_store_24(packet, pos, 380aeb99916SMilanka Ringwald connection->target_supported_companies[i]); 381aeb99916SMilanka Ringwald pos += 3; 382aeb99916SMilanka Ringwald } 383aeb99916SMilanka Ringwald break; 384aeb99916SMilanka Ringwald default: 385ef07fa3cSMilanka Ringwald // error response 386ef07fa3cSMilanka Ringwald break; 387aeb99916SMilanka Ringwald } 388aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos); 389aeb99916SMilanka Ringwald return; 390aeb99916SMilanka Ringwald 391aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES: 392f3bb6622SMilanka Ringwald packet[pos++] = count_set_bits_uint32(connection->target_now_playing_info_attr_bitmap); 393ccfcd81eSMilanka Ringwald max_payload_size--; 394ccfcd81eSMilanka Ringwald 395aeb99916SMilanka Ringwald bytes_stored = avrcp_store_avctp_now_playing_info_fragment(connection, max_payload_size, packet + pos); 396aeb99916SMilanka Ringwald 397aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += bytes_stored + pos; 398ccfcd81eSMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos + bytes_stored); 399aeb99916SMilanka Ringwald return; 400aeb99916SMilanka Ringwald 401aeb99916SMilanka Ringwald default: 402ef8ea8e4SMilanka Ringwald // error response and other OPCODEs 403aeb99916SMilanka Ringwald break; 404aeb99916SMilanka Ringwald } 405aeb99916SMilanka Ringwald break; 406aeb99916SMilanka Ringwald 407aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_PASS_THROUGH: 408aeb99916SMilanka Ringwald packet[pos++] = connection->operation_id; 409aeb99916SMilanka Ringwald // parameter length 410aeb99916SMilanka Ringwald packet[pos++] = (uint8_t) connection->data_len; 411aeb99916SMilanka Ringwald pos += 2; 412aeb99916SMilanka Ringwald break; 413aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_UNIT_INFO: 414aeb99916SMilanka Ringwald break; 415aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_SUBUNIT_INFO: 416aeb99916SMilanka Ringwald break; 417aeb99916SMilanka Ringwald default: 418aeb99916SMilanka Ringwald btstack_assert(false); 419aeb99916SMilanka Ringwald return; 420aeb99916SMilanka Ringwald } 421aeb99916SMilanka Ringwald break; 422aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 423aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 424aeb99916SMilanka Ringwald switch (connection->pdu_id) { 425aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES: 426aeb99916SMilanka Ringwald bytes_stored = avrcp_store_avctp_now_playing_info_fragment(connection, max_payload_size, packet + pos); 427aeb99916SMilanka Ringwald 428aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += bytes_stored + pos; 429ccfcd81eSMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos + bytes_stored); 430aeb99916SMilanka Ringwald return; 431aeb99916SMilanka Ringwald 432aeb99916SMilanka Ringwald default: 433aeb99916SMilanka Ringwald break; 434aeb99916SMilanka Ringwald } 435aeb99916SMilanka Ringwald break; 436aeb99916SMilanka Ringwald default: 437aeb99916SMilanka Ringwald btstack_assert(false); 438aeb99916SMilanka Ringwald return; 4394b338011SMilanka Ringwald } 4404b338011SMilanka Ringwald 441aeb99916SMilanka Ringwald // compare number of bytes to store with the remaining buffer size 442aeb99916SMilanka Ringwald uint16_t bytes_to_copy = btstack_min(connection->data_len - connection->data_offset, max_payload_size - pos); 443aeb99916SMilanka Ringwald 444aeb99916SMilanka Ringwald (void)memcpy(packet + pos, &connection->data[connection->data_offset], bytes_to_copy); 445aeb99916SMilanka Ringwald pos += bytes_to_copy; 446aeb99916SMilanka Ringwald connection->data_offset += bytes_to_copy; 447aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += pos; 448aeb99916SMilanka Ringwald 449aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos); 450aeb99916SMilanka Ringwald } 451aeb99916SMilanka Ringwald 4523b899686SMilanka Ringwald static void avctp_send_reject_cmd_wrong_pid(avrcp_connection_t * connection){ 453aeb99916SMilanka Ringwald l2cap_reserve_packet_buffer(); 454aeb99916SMilanka Ringwald uint8_t * packet = l2cap_get_outgoing_buffer(); 455aeb99916SMilanka Ringwald 456aeb99916SMilanka Ringwald // AVCTP header 457aeb99916SMilanka Ringwald // transport header : transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 4583b899686SMilanka Ringwald packet[0] = (connection->transaction_id << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_RESPONSE_FRAME << 1) | 1; 459393f5724SMilanka Ringwald big_endian_store_16(packet, 1, connection->message_body[0]); 460aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, 3); 461aeb99916SMilanka Ringwald } 462aeb99916SMilanka Ringwald 463c3c3461eSMilanka Ringwald static void avrcp_target_custom_command_data_init(avrcp_connection_t * connection, 464aeb99916SMilanka Ringwald avrcp_command_opcode_t opcode, avrcp_command_type_t command_type, 465aeb99916SMilanka Ringwald avrcp_subunit_type_t subunit_type, avrcp_subunit_id_t subunit_id, 466aeb99916SMilanka Ringwald avrcp_pdu_id_t pdu_id, uint32_t company_id){ 467aeb99916SMilanka Ringwald 468aeb99916SMilanka Ringwald connection->command_opcode = opcode; 4696649facbSMatthias Ringwald connection->command_type = command_type; 47012b7c8ecSMilanka Ringwald connection->subunit_type = subunit_type; 47112b7c8ecSMilanka Ringwald connection->subunit_id = subunit_id; 472aeb99916SMilanka Ringwald connection->company_id = company_id << 16; 473aeb99916SMilanka Ringwald connection->pdu_id = pdu_id; 474aeb99916SMilanka Ringwald connection->data = NULL; 475aeb99916SMilanka Ringwald connection->data_offset = 0; 476aeb99916SMilanka Ringwald connection->data_len = 0; 4771e852c68SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 4786649facbSMatthias Ringwald } 4796649facbSMatthias Ringwald 480aeb99916SMilanka Ringwald static void avrcp_target_vendor_dependent_response_data_init(avrcp_connection_t * connection, avrcp_command_type_t command_type, avrcp_pdu_id_t pdu_id){ 481aeb99916SMilanka Ringwald connection->command_opcode = AVRCP_CMD_OPCODE_VENDOR_DEPENDENT; 482aeb99916SMilanka Ringwald connection->subunit_type = AVRCP_SUBUNIT_TYPE_PANEL; 483aeb99916SMilanka Ringwald connection->subunit_id = AVRCP_SUBUNIT_ID; 484aeb99916SMilanka Ringwald connection->company_id = BT_SIG_COMPANY_ID; 485aeb99916SMilanka Ringwald 486aeb99916SMilanka Ringwald connection->command_type = command_type; 487aeb99916SMilanka Ringwald connection->pdu_id = pdu_id; 488393f5724SMilanka Ringwald connection->data = connection->message_body; 489aeb99916SMilanka Ringwald connection->data_offset = 0; 4901e852c68SMilanka Ringwald connection->data_len = 0; 491aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 492aeb99916SMilanka Ringwald } 493aeb99916SMilanka Ringwald 494aeb99916SMilanka Ringwald static void avrcp_target_pass_through_command_data_init(avrcp_connection_t * connection, avrcp_command_type_t command_type, avrcp_operation_id_t opid){ 495aeb99916SMilanka Ringwald connection->command_opcode = AVRCP_CMD_OPCODE_PASS_THROUGH; 496aeb99916SMilanka Ringwald connection->subunit_type = AVRCP_SUBUNIT_TYPE_PANEL; 497aeb99916SMilanka Ringwald connection->subunit_id = AVRCP_SUBUNIT_ID; 498aeb99916SMilanka Ringwald 499aeb99916SMilanka Ringwald connection->command_type = command_type; 500aeb99916SMilanka Ringwald connection->company_id = 0; 501ccfcd81eSMilanka Ringwald connection->pdu_id = AVRCP_PDU_ID_UNDEFINED; 502aeb99916SMilanka Ringwald connection->operation_id = opid; 503aeb99916SMilanka Ringwald 504393f5724SMilanka Ringwald connection->data = connection->message_body; 505aeb99916SMilanka Ringwald connection->data_offset = 0; 506aeb99916SMilanka Ringwald connection->data_len = 0; 5071e852c68SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 508aeb99916SMilanka Ringwald } 509aeb99916SMilanka Ringwald 510aeb99916SMilanka Ringwald 511aeb99916SMilanka Ringwald static uint8_t avrcp_target_vendor_dependent_response_accept(avrcp_connection_t * connection, avrcp_pdu_id_t pdu_id, uint8_t status){ 512aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, pdu_id); 513aeb99916SMilanka Ringwald connection->data_len = 1; 514aeb99916SMilanka Ringwald connection->data[0] = status; 515aeb99916SMilanka Ringwald 516393f5724SMilanka Ringwald connection->target_accept_response = true; 517ef8ea8e4SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 51812b7c8ecSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 51912b7c8ecSMilanka Ringwald return ERROR_CODE_SUCCESS; 52012b7c8ecSMilanka Ringwald } 52112b7c8ecSMilanka Ringwald 522aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_reject(avrcp_connection_t * connection, avrcp_pdu_id_t pdu_id, avrcp_status_code_t status){ 523aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_REJECTED, pdu_id); 524aeb99916SMilanka Ringwald connection->data_len = 1; 525aeb99916SMilanka Ringwald connection->data[0] = status; 526aeb99916SMilanka Ringwald 527d1207cd8SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 528d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 529d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 530d1207cd8SMilanka Ringwald } 531d1207cd8SMilanka Ringwald 532aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_not_implemented(avrcp_connection_t * connection, avrcp_pdu_id_t pdu_id, uint8_t event_id){ 533aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_NOT_IMPLEMENTED, pdu_id); 534aeb99916SMilanka Ringwald connection->data_len = 1; 535aeb99916SMilanka Ringwald connection->data[0] = event_id; 536d1386928SMilanka Ringwald 537d1386928SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 538d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 539d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 540e0bbf3edSMilanka Ringwald } 541e0bbf3edSMilanka Ringwald 542d8133c09SMatthias Ringwald static uint8_t avrcp_target_response_vendor_dependent_interim(avrcp_connection_t * connection, avrcp_pdu_id_t pdu_id, uint8_t event_id, const uint8_t * value, uint16_t value_len){ 543aeb99916SMilanka Ringwald btstack_assert(value_len + 1 < AVRCP_MAX_COMMAND_PARAMETER_LENGTH); 544aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_INTERIM, pdu_id); 545aeb99916SMilanka Ringwald connection->data_len = 1 + value_len; 546aeb99916SMilanka Ringwald connection->data[0] = event_id; 547aeb99916SMilanka Ringwald 548c1ab6cc1SMatthias Ringwald if (value && (value_len > 0)){ 549aeb99916SMilanka Ringwald (void)memcpy(connection->data + 1, value, value_len); 550d1207cd8SMilanka Ringwald } 551aeb99916SMilanka Ringwald 552d1207cd8SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 553d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 554d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 555d1207cd8SMilanka Ringwald } 556d1207cd8SMilanka Ringwald 557aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_addressed_player_changed_interim(avrcp_connection_t * connection, avrcp_pdu_id_t pdu_id, uint8_t event_id){ 558aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_INTERIM, pdu_id); 5590a2feef4SMilanka Ringwald 560aeb99916SMilanka Ringwald connection->data_len = 5; 561aeb99916SMilanka Ringwald connection->data[0] = event_id; 562393f5724SMilanka Ringwald big_endian_store_16(connection->data, 1, connection->target_addressed_player_id); 563393f5724SMilanka Ringwald big_endian_store_16(connection->data, 3, connection->target_uid_counter); 5646568eb54SMilanka Ringwald 5656568eb54SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 5666568eb54SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 5676568eb54SMilanka Ringwald return ERROR_CODE_SUCCESS; 5686568eb54SMilanka Ringwald } 5696568eb54SMilanka Ringwald 570aeb99916SMilanka Ringwald static uint8_t avrcp_target_pass_through_response(uint16_t avrcp_cid, avrcp_command_type_t ctype, avrcp_operation_id_t opid, uint8_t operands_length, uint8_t operand){ 5711945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 572831d3fd5SMilanka Ringwald if (!connection){ 573831d3fd5SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 574831d3fd5SMilanka Ringwald } 575aeb99916SMilanka Ringwald avrcp_target_pass_through_command_data_init(connection, ctype, opid); 576831d3fd5SMilanka Ringwald 577831d3fd5SMilanka Ringwald if (operands_length == 1){ 578aeb99916SMilanka Ringwald connection->data_len = 1; 579393f5724SMilanka Ringwald connection->message_body[0] = operand; 580831d3fd5SMilanka Ringwald } 581831d3fd5SMilanka Ringwald 582831d3fd5SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 583831d3fd5SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 584831d3fd5SMilanka Ringwald return ERROR_CODE_SUCCESS; 585831d3fd5SMilanka Ringwald } 586831d3fd5SMilanka Ringwald 587831d3fd5SMilanka Ringwald uint8_t avrcp_target_operation_rejected(uint16_t avrcp_cid, avrcp_operation_id_t opid, uint8_t operands_length, uint8_t operand){ 588831d3fd5SMilanka Ringwald return avrcp_target_pass_through_response(avrcp_cid, AVRCP_CTYPE_RESPONSE_REJECTED, opid, operands_length, operand); 589831d3fd5SMilanka Ringwald } 590831d3fd5SMilanka Ringwald 591831d3fd5SMilanka Ringwald uint8_t avrcp_target_operation_accepted(uint16_t avrcp_cid, avrcp_operation_id_t opid, uint8_t operands_length, uint8_t operand){ 592831d3fd5SMilanka Ringwald return avrcp_target_pass_through_response(avrcp_cid, AVRCP_CTYPE_RESPONSE_ACCEPTED, opid, operands_length, operand); 593831d3fd5SMilanka Ringwald } 594831d3fd5SMilanka Ringwald 595831d3fd5SMilanka Ringwald uint8_t avrcp_target_operation_not_implemented(uint16_t avrcp_cid, avrcp_operation_id_t opid, uint8_t operands_length, uint8_t operand){ 596831d3fd5SMilanka Ringwald return avrcp_target_pass_through_response(avrcp_cid, AVRCP_CTYPE_RESPONSE_ACCEPTED, opid, operands_length, operand); 597831d3fd5SMilanka Ringwald } 598831d3fd5SMilanka Ringwald 59961a3437bSMilanka Ringwald uint8_t avrcp_target_set_unit_info(uint16_t avrcp_cid, avrcp_subunit_type_t unit_type, uint32_t company_id){ 6001945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 6014b338011SMilanka Ringwald if (!connection){ 60261a3437bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 6034b338011SMilanka Ringwald } 604393f5724SMilanka Ringwald connection->target_unit_type = unit_type; 605d1207cd8SMilanka Ringwald connection->company_id = company_id; 60661a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 607d1207cd8SMilanka Ringwald } 608d1207cd8SMilanka Ringwald 60961a3437bSMilanka Ringwald uint8_t avrcp_target_set_subunit_info(uint16_t avrcp_cid, avrcp_subunit_type_t subunit_type, const uint8_t * subunit_info_data, uint16_t subunit_info_data_size){ 6101945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 611d1207cd8SMilanka Ringwald if (!connection){ 61261a3437bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 613d1207cd8SMilanka Ringwald } 614393f5724SMilanka Ringwald connection->target_subunit_info_type = subunit_type; 615393f5724SMilanka Ringwald connection->target_subunit_info_data = subunit_info_data; 616393f5724SMilanka Ringwald connection->target_subunit_info_data_size = subunit_info_data_size; 61761a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 618d1207cd8SMilanka Ringwald } 619d1207cd8SMilanka Ringwald 620d1207cd8SMilanka Ringwald static uint8_t avrcp_target_unit_info(avrcp_connection_t * connection){ 621aeb99916SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED){ 622aeb99916SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 623aeb99916SMilanka Ringwald } 624aeb99916SMilanka Ringwald 6257bccc69fSMilanka Ringwald avrcp_target_custom_command_data_init(connection, 626aeb99916SMilanka Ringwald AVRCP_CMD_OPCODE_UNIT_INFO, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, 6277bccc69fSMilanka Ringwald AVRCP_SUBUNIT_TYPE_UNIT, AVRCP_SUBUNIT_ID_IGNORE, AVRCP_PDU_ID_UNDEFINED, 6287bccc69fSMilanka Ringwald connection->company_id); 6294b338011SMilanka Ringwald 6304b338011SMilanka Ringwald uint8_t unit = 0; 631393f5724SMilanka Ringwald connection->data = connection->message_body; 632aeb99916SMilanka Ringwald connection->data_len = 5; 633aeb99916SMilanka Ringwald connection->data[0] = 0x07; 634393f5724SMilanka Ringwald connection->data[1] = (connection->target_unit_type << 4) | unit; 6354b338011SMilanka Ringwald // company id is 3 bytes long 636aeb99916SMilanka Ringwald big_endian_store_24(connection->data, 2, connection->company_id); 637d1386928SMilanka Ringwald 638d3db1226SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 639d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 640d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 6414b338011SMilanka Ringwald } 6424b338011SMilanka Ringwald 643d1207cd8SMilanka Ringwald static uint8_t avrcp_target_subunit_info(avrcp_connection_t * connection, uint8_t offset){ 6444b338011SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED) return ERROR_CODE_COMMAND_DISALLOWED; 645fda344fcSMatthias Ringwald if (offset >= 32) return AVRCP_STATUS_INVALID_PARAMETER; 6464b338011SMilanka Ringwald 6477bccc69fSMilanka Ringwald avrcp_target_custom_command_data_init(connection, AVRCP_CMD_OPCODE_SUBUNIT_INFO, 6487bccc69fSMilanka Ringwald AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, 6497bccc69fSMilanka Ringwald AVRCP_SUBUNIT_TYPE_UNIT, AVRCP_SUBUNIT_ID_IGNORE, AVRCP_PDU_ID_UNDEFINED, 6507bccc69fSMilanka Ringwald connection->company_id); 6514b338011SMilanka Ringwald 6524b338011SMilanka Ringwald uint8_t page = offset / 4; 6534b338011SMilanka Ringwald uint8_t extension_code = 7; 654393f5724SMilanka Ringwald connection->data = connection->message_body; 655aeb99916SMilanka Ringwald connection->data_len = 5; 656aeb99916SMilanka Ringwald connection->data[0] = (page << 4) | extension_code; 657d1207cd8SMilanka Ringwald 658f0af2234SMatthias Ringwald // mark non-existent entries with 0xff 659393f5724SMilanka Ringwald memset(&connection->message_body[1], 0xFF, 4); 660393f5724SMilanka Ringwald if ((connection->data != NULL) && (offset < connection->target_subunit_info_data_size)){ 661393f5724SMilanka Ringwald uint8_t bytes_to_copy = btstack_min(connection->target_subunit_info_data_size - offset, 4); 662393f5724SMilanka Ringwald memcpy(&connection->data[1], &connection->target_subunit_info_data[offset], bytes_to_copy); 663fda344fcSMatthias Ringwald } 664d1386928SMilanka Ringwald 665d3db1226SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 666d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 667d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 6684b338011SMilanka Ringwald } 6694b338011SMilanka Ringwald 670aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_supported_events(avrcp_connection_t * connection){ 671aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_CAPABILITIES); 67293259c07SMilanka Ringwald 67389042326SMilanka Ringwald uint8_t event_id; 674aeb99916SMilanka Ringwald uint8_t num_events = 0; 67589042326SMilanka Ringwald for (event_id = (uint8_t) AVRCP_NOTIFICATION_EVENT_FIRST_INDEX; event_id < (uint8_t) AVRCP_NOTIFICATION_EVENT_LAST_INDEX; event_id++){ 676393f5724SMilanka Ringwald if ((connection->notifications_supported_by_target & (1 << event_id)) == 0){ 67789042326SMilanka Ringwald continue; 67889042326SMilanka Ringwald } 67989042326SMilanka Ringwald num_events++; 68089042326SMilanka Ringwald } 68189042326SMilanka Ringwald 682aeb99916SMilanka Ringwald connection->data[0] = AVRCP_CAPABILITY_ID_EVENT; 683aeb99916SMilanka Ringwald connection->data[1] = num_events; 684aeb99916SMilanka Ringwald connection->data_len = 2 + num_events; 685e0bbf3edSMilanka Ringwald 686aeb99916SMilanka Ringwald // fill the data later directly to the L2CAP outgoing buffer 687ae5447c3SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 688d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 689d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 690e0bbf3edSMilanka Ringwald } 691e0bbf3edSMilanka Ringwald 692aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_supported_companies(avrcp_connection_t * connection){ 693aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_CAPABILITIES); 694aeb99916SMilanka Ringwald 695aeb99916SMilanka Ringwald connection->data[0] = AVRCP_CAPABILITY_ID_COMPANY; 696ef07fa3cSMilanka Ringwald if (connection->target_supported_companies_num == 0){ 697ef07fa3cSMilanka Ringwald connection->target_supported_companies_num = 1; 698ef07fa3cSMilanka Ringwald connection->target_supported_companies = default_companies; 699ef07fa3cSMilanka Ringwald } 700ef07fa3cSMilanka Ringwald 701aeb99916SMilanka Ringwald connection->data[1] = connection->target_supported_companies_num; 702ef07fa3cSMilanka Ringwald connection->data_len = 2 + connection->data[1] * 3; 703aeb99916SMilanka Ringwald 704aeb99916SMilanka Ringwald // fill the data later directly to the L2CAP outgoing buffer and 705aeb99916SMilanka Ringwald // use Bluetooth SIG as default company 706ae5447c3SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 70789042326SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 70889042326SMilanka Ringwald return ERROR_CODE_SUCCESS; 70989042326SMilanka Ringwald } 71089042326SMilanka Ringwald 7110346f11dSMilanka Ringwald uint8_t avrcp_target_support_event(uint16_t avrcp_cid, avrcp_notification_event_id_t event_id){ 7120346f11dSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7130346f11dSMilanka Ringwald if (!connection){ 7140346f11dSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 71589042326SMilanka Ringwald } 71689042326SMilanka Ringwald 7170346f11dSMilanka Ringwald if ((event_id < (uint8_t)AVRCP_NOTIFICATION_EVENT_FIRST_INDEX) || (event_id > (uint8_t)AVRCP_NOTIFICATION_EVENT_LAST_INDEX)){ 7180346f11dSMilanka Ringwald return ERROR_CODE_UNSUPPORTED_FEATURE_OR_PARAMETER_VALUE; 7190346f11dSMilanka Ringwald } 7200346f11dSMilanka Ringwald 721393f5724SMilanka Ringwald connection->notifications_supported_by_target |= (1 << (uint8_t)event_id); 7220346f11dSMilanka Ringwald return ERROR_CODE_SUCCESS; 7230346f11dSMilanka Ringwald } 7240346f11dSMilanka Ringwald 7250346f11dSMilanka Ringwald uint8_t avrcp_target_support_companies(uint16_t avrcp_cid, uint8_t num_companies, const uint32_t *companies){ 7260346f11dSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7270346f11dSMilanka Ringwald if (!connection){ 7280346f11dSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7290346f11dSMilanka Ringwald } 7300346f11dSMilanka Ringwald 7310346f11dSMilanka Ringwald connection->target_supported_companies_num = num_companies; 7320346f11dSMilanka Ringwald connection->target_supported_companies = companies; 7330346f11dSMilanka Ringwald return ERROR_CODE_SUCCESS; 7345e3d4d2bSMilanka Ringwald } 7355e3d4d2bSMilanka Ringwald 736aeb99916SMilanka Ringwald // TODO Review (use flags) 737d1207cd8SMilanka Ringwald uint8_t avrcp_target_play_status(uint16_t avrcp_cid, uint32_t song_length_ms, uint32_t song_position_ms, avrcp_playback_status_t play_status){ 738aeb99916SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 739aeb99916SMilanka Ringwald if (!connection){ 740aeb99916SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 741aeb99916SMilanka Ringwald } 742aeb99916SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED){ 743aeb99916SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 7445e3d4d2bSMilanka Ringwald } 7455e3d4d2bSMilanka Ringwald 746aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_PLAY_STATUS); 747aeb99916SMilanka Ringwald connection->data_len = 9; 748aeb99916SMilanka Ringwald big_endian_store_32(connection->data, 0, song_length_ms); 749aeb99916SMilanka Ringwald big_endian_store_32(connection->data, 4, song_position_ms); 750aeb99916SMilanka Ringwald connection->data[8] = play_status; 7515e3d4d2bSMilanka Ringwald 7525e3d4d2bSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 7535e3d4d2bSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7545e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7555e3d4d2bSMilanka Ringwald } 7565e3d4d2bSMilanka Ringwald 757d1207cd8SMilanka Ringwald static uint8_t avrcp_target_store_media_attr(avrcp_connection_t * connection, avrcp_media_attribute_id_t attr_id, const char * value){ 7585e3d4d2bSMilanka Ringwald int index = attr_id - 1; 759d1207cd8SMilanka Ringwald if (!value) return AVRCP_STATUS_INVALID_PARAMETER; 760ab2445a0SMatthias Ringwald uint16_t value_len = (uint16_t)strlen(value); 761ab2445a0SMatthias Ringwald btstack_assert(value_len <= 255); 762393f5724SMilanka Ringwald connection->target_now_playing_info[index].value = (uint8_t*)value; 763ab2445a0SMatthias Ringwald connection->target_now_playing_info[index].len = value_len; 7645e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7655e3d4d2bSMilanka Ringwald } 7665e3d4d2bSMilanka Ringwald 767d1207cd8SMilanka Ringwald uint8_t avrcp_target_set_playback_status(uint16_t avrcp_cid, avrcp_playback_status_t playback_status){ 7681945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7695e3d4d2bSMilanka Ringwald if (!connection){ 7705e3d4d2bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7715e3d4d2bSMilanka Ringwald } 772393f5724SMilanka Ringwald if (connection->target_playback_status == playback_status){ 77361a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 77461a3437bSMilanka Ringwald } 775d1207cd8SMilanka Ringwald 776393f5724SMilanka Ringwald connection->target_playback_status = playback_status; 7774b5213b0SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED)) { 778393f5724SMilanka Ringwald connection->target_playback_status_changed = true; 779d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7804b5213b0SMilanka Ringwald } 7815e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7825e3d4d2bSMilanka Ringwald } 7835e3d4d2bSMilanka Ringwald 784276c795aSMilanka Ringwald static void avrcp_target_register_track_changed(avrcp_connection_t * connection, const uint8_t * track_id){ 7857d1b72e5SMilanka Ringwald if (track_id == NULL){ 7867d1b72e5SMilanka Ringwald memset(connection->target_track_id, 0xFF, 8); 7877d1b72e5SMilanka Ringwald connection->target_track_selected = false; 7887d1b72e5SMilanka Ringwald } else { 7897d1b72e5SMilanka Ringwald (void)memcpy(connection->target_track_id, track_id, 8); 7907d1b72e5SMilanka Ringwald connection->target_track_selected = true; 7917d1b72e5SMilanka Ringwald } 7927d1b72e5SMilanka Ringwald 7937d1b72e5SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED)) { 7947d1b72e5SMilanka Ringwald connection->target_track_changed = true; 7957d1b72e5SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7967d1b72e5SMilanka Ringwald } 7977d1b72e5SMilanka Ringwald } 7987d1b72e5SMilanka Ringwald 7997d1b72e5SMilanka Ringwald uint8_t avrcp_target_track_changed(uint16_t avrcp_cid, uint8_t * track_id){ 8007d1b72e5SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8017d1b72e5SMilanka Ringwald if (!connection){ 8027d1b72e5SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8037d1b72e5SMilanka Ringwald } 8047d1b72e5SMilanka Ringwald avrcp_target_register_track_changed(connection, track_id); 8057d1b72e5SMilanka Ringwald return ERROR_CODE_SUCCESS; 8067d1b72e5SMilanka Ringwald } 8077d1b72e5SMilanka Ringwald 80861a3437bSMilanka Ringwald uint8_t avrcp_target_set_now_playing_info(uint16_t avrcp_cid, const avrcp_track_t * current_track, uint16_t total_tracks){ 8091945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8105e3d4d2bSMilanka Ringwald if (!connection){ 81161a3437bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8125e3d4d2bSMilanka Ringwald } 813d1207cd8SMilanka Ringwald if (!current_track){ 81461a3437bSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 815d1207cd8SMilanka Ringwald } 81661a3437bSMilanka Ringwald 817393f5724SMilanka Ringwald (void)memcpy(connection->target_track_id, current_track->track_id, 8); 818393f5724SMilanka Ringwald connection->target_song_length_ms = current_track->song_length_ms; 819393f5724SMilanka Ringwald connection->target_track_nr = current_track->track_nr; 820393f5724SMilanka Ringwald connection->target_total_tracks = total_tracks; 821d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_TITLE, current_track->title); 822d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_ARTIST, current_track->artist); 823d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_ALBUM, current_track->album); 824d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_GENRE, current_track->genre); 825aeb99916SMilanka Ringwald 8267d1b72e5SMilanka Ringwald avrcp_target_register_track_changed(connection, current_track->track_id); 82761a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 8285e3d4d2bSMilanka Ringwald } 8295e3d4d2bSMilanka Ringwald 83012b7c8ecSMilanka Ringwald 831d1207cd8SMilanka Ringwald uint8_t avrcp_target_playing_content_changed(uint16_t avrcp_cid){ 8321945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8335e3d4d2bSMilanka Ringwald if (!connection){ 8345e3d4d2bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8355e3d4d2bSMilanka Ringwald } 83612b7c8ecSMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED)) { 837393f5724SMilanka Ringwald connection->target_playing_content_changed = true; 838d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 83912b7c8ecSMilanka Ringwald } 8405e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 8415e3d4d2bSMilanka Ringwald } 842e0bbf3edSMilanka Ringwald 8436568eb54SMilanka Ringwald uint8_t avrcp_target_addressed_player_changed(uint16_t avrcp_cid, uint16_t player_id, uint16_t uid_counter){ 8441945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8456568eb54SMilanka Ringwald if (!connection){ 8466568eb54SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8476568eb54SMilanka Ringwald } 84861a3437bSMilanka Ringwald 849393f5724SMilanka Ringwald if (connection->target_addressed_player_id == player_id){ 85061a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 85161a3437bSMilanka Ringwald } 852db818650SMilanka Ringwald 853393f5724SMilanka Ringwald connection->target_uid_counter = uid_counter; 854393f5724SMilanka Ringwald connection->target_addressed_player_id = player_id; 855db818650SMilanka Ringwald 856db818650SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED)) { 857393f5724SMilanka Ringwald connection->target_addressed_player_changed = true; 858db818650SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 8596568eb54SMilanka Ringwald } 8606568eb54SMilanka Ringwald return ERROR_CODE_SUCCESS; 8616568eb54SMilanka Ringwald } 8626568eb54SMilanka Ringwald 8636a2e66beSMilanka Ringwald uint8_t avrcp_target_uids_changed(uint16_t avrcp_cid, uint16_t uid_counter){ 8646a2e66beSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8656a2e66beSMilanka Ringwald if (!connection){ 8666a2e66beSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8676a2e66beSMilanka Ringwald } 8686a2e66beSMilanka Ringwald 8696a2e66beSMilanka Ringwald connection->target_uid_counter = uid_counter; 8706a2e66beSMilanka Ringwald 8716a2e66beSMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_UIDS_CHANGED)) { 8726a2e66beSMilanka Ringwald connection->target_uids_changed = true; 8736a2e66beSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 8746a2e66beSMilanka Ringwald } 8756a2e66beSMilanka Ringwald return ERROR_CODE_SUCCESS; 8766a2e66beSMilanka Ringwald } 8776a2e66beSMilanka Ringwald 878d1207cd8SMilanka Ringwald uint8_t avrcp_target_battery_status_changed(uint16_t avrcp_cid, avrcp_battery_status_t battery_status){ 8791945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 880d1207cd8SMilanka Ringwald if (!connection){ 881d1207cd8SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 882d1207cd8SMilanka Ringwald } 883393f5724SMilanka Ringwald if (connection->target_battery_status == battery_status){ 88461a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 88561a3437bSMilanka Ringwald } 886db818650SMilanka Ringwald 887393f5724SMilanka Ringwald connection->target_battery_status = battery_status; 888db818650SMilanka Ringwald 889db818650SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED)) { 890393f5724SMilanka Ringwald connection->target_battery_status_changed = true; 891d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 89212b7c8ecSMilanka Ringwald } 893d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 894d1207cd8SMilanka Ringwald } 895d1207cd8SMilanka Ringwald 896b772a0d7SMilanka Ringwald uint8_t avrcp_target_adjust_absolute_volume(uint16_t avrcp_cid, uint8_t absolute_volume){ 897b772a0d7SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 898b772a0d7SMilanka Ringwald if (!connection){ 899b772a0d7SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 900b772a0d7SMilanka Ringwald } 901b772a0d7SMilanka Ringwald 902393f5724SMilanka Ringwald connection->target_absolute_volume = absolute_volume; 903b772a0d7SMilanka Ringwald return ERROR_CODE_SUCCESS; 904b772a0d7SMilanka Ringwald } 905b772a0d7SMilanka Ringwald 906984bbc3eSMilanka Ringwald uint8_t avrcp_target_volume_changed(uint16_t avrcp_cid, uint8_t absolute_volume){ 9071945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 908d1207cd8SMilanka Ringwald if (!connection){ 909d1207cd8SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 910d1207cd8SMilanka Ringwald } 911393f5724SMilanka Ringwald if (connection->target_absolute_volume == absolute_volume){ 91261a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 91361a3437bSMilanka Ringwald } 91461a3437bSMilanka Ringwald 915393f5724SMilanka Ringwald connection->target_absolute_volume = absolute_volume; 91661a3437bSMilanka Ringwald 9173e5c3c5bSMatthias Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED )) { 918393f5724SMilanka Ringwald connection->target_notify_absolute_volume_changed = true; 919d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 92012b7c8ecSMilanka Ringwald } 921d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 922d1207cd8SMilanka Ringwald } 923d1207cd8SMilanka Ringwald 9240ec79bd0SMilanka Ringwald static void avrcp_target_set_transaction_label_for_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification, uint8_t transaction_label){ 925fe934656SMatthias Ringwald if (notification > AVRCP_NOTIFICATION_EVENT_MAX_VALUE) return; 926393f5724SMilanka Ringwald connection->target_notifications_transaction_label[notification] = transaction_label; 9270ec79bd0SMilanka Ringwald } 9280ec79bd0SMilanka Ringwald 9290ec79bd0SMilanka Ringwald static uint8_t avrcp_target_get_transaction_label_for_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification){ 930fe934656SMatthias Ringwald if (notification > AVRCP_NOTIFICATION_EVENT_MAX_VALUE) return 0; 931393f5724SMilanka Ringwald return connection->target_notifications_transaction_label[notification]; 9320ec79bd0SMilanka Ringwald } 933d1207cd8SMilanka Ringwald 9340b578d06SMilanka Ringwald static bool avcrp_operation_id_is_valid(avrcp_operation_id_t operation_id){ 9350b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_1) return true; 9360b578d06SMilanka Ringwald 9370b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_0) return false; 9380b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_2) return true; 9390b578d06SMilanka Ringwald 9400b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_CHANNEL_UP) return false; 9410b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_3) return true; 9420b578d06SMilanka Ringwald 9430b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_CHANNEL_UP) return false; 9440b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_3) return true; 9450b578d06SMilanka Ringwald 9460b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_SKIP) return false; 9470b578d06SMilanka Ringwald if (operation_id == AVRCP_OPERATION_ID_SKIP) return true; 9480b578d06SMilanka Ringwald 9490b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_POWER) return false; 9500b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_4) return true; 9510b578d06SMilanka Ringwald 9520b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_ANGLE) return false; 9530b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_5) return true; 9540b578d06SMilanka Ringwald 9550b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_F1) return false; 9560b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_6) return true; 9571f2f3fd1SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_7) return true; 9580b578d06SMilanka Ringwald return false; 9590b578d06SMilanka Ringwald } 9600b578d06SMilanka Ringwald 961aeb99916SMilanka Ringwald 962aeb99916SMilanka Ringwald #ifdef ENABLE_AVCTP_FRAGMENTATION 963aeb99916SMilanka Ringwald static void avctp_reassemble_message(avrcp_connection_t * connection, avctp_packet_type_t packet_type, uint8_t *packet, uint16_t size){ 964aeb99916SMilanka Ringwald // after header (transaction label and packet type) 965aeb99916SMilanka Ringwald uint16_t pos; 966aeb99916SMilanka Ringwald uint16_t bytes_to_store; 967aeb99916SMilanka Ringwald 968aeb99916SMilanka Ringwald switch (packet_type){ 969aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 970aeb99916SMilanka Ringwald if (size < 2) return; 971aeb99916SMilanka Ringwald 972aeb99916SMilanka Ringwald // store header 973aeb99916SMilanka Ringwald pos = 0; 974aeb99916SMilanka Ringwald connection->avctp_reassembly_buffer[pos] = packet[pos]; 975aeb99916SMilanka Ringwald pos++; 976aeb99916SMilanka Ringwald connection->avctp_reassembly_size = pos; 977aeb99916SMilanka Ringwald 978aeb99916SMilanka Ringwald // NOTE: num packets not needed for reassembly, ignoring it does not pose security risk -> no need to store it 979aeb99916SMilanka Ringwald pos++; 980aeb99916SMilanka Ringwald 981aeb99916SMilanka Ringwald // PID in reassembled packet is at offset 1, it will be read later after the avctp_reassemble_message with AVCTP_END_PACKET is called 982aeb99916SMilanka Ringwald 983aeb99916SMilanka Ringwald bytes_to_store = btstack_min(size - pos, sizeof(connection->avctp_reassembly_buffer) - connection->avctp_reassembly_size); 984aeb99916SMilanka Ringwald memcpy(&connection->avctp_reassembly_buffer[connection->avctp_reassembly_size], &packet[pos], bytes_to_store); 985aeb99916SMilanka Ringwald connection->avctp_reassembly_size += bytes_to_store; 986aeb99916SMilanka Ringwald break; 987aeb99916SMilanka Ringwald 988aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 989aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 990aeb99916SMilanka Ringwald if (size < 1) return; 991aeb99916SMilanka Ringwald 992aeb99916SMilanka Ringwald // store remaining data, ignore header 993aeb99916SMilanka Ringwald pos = 1; 994aeb99916SMilanka Ringwald bytes_to_store = btstack_min(size - pos, sizeof(connection->avctp_reassembly_buffer) - connection->avctp_reassembly_size); 995aeb99916SMilanka Ringwald memcpy(&connection->avctp_reassembly_buffer[connection->avctp_reassembly_size], &packet[pos], bytes_to_store); 996aeb99916SMilanka Ringwald connection->avctp_reassembly_size += bytes_to_store; 997aeb99916SMilanka Ringwald break; 998aeb99916SMilanka Ringwald 999aeb99916SMilanka Ringwald default: 1000aeb99916SMilanka Ringwald return; 1001aeb99916SMilanka Ringwald } 1002aeb99916SMilanka Ringwald } 1003aeb99916SMilanka Ringwald #endif 1004aeb99916SMilanka Ringwald 100501dc6e35SMilanka Ringwald static void avrcp_handle_l2cap_data_packet_for_signaling_connection(avrcp_connection_t * connection, uint8_t * packet, uint16_t size){ 1006aeb99916SMilanka Ringwald uint8_t avctp_header = packet[0]; 1007aeb99916SMilanka Ringwald connection->transaction_id = avctp_header >> 4; 1008aeb99916SMilanka Ringwald 1009aeb99916SMilanka Ringwald avctp_packet_type_t avctp_packet_type = (avctp_packet_type_t) ((avctp_header & 0x0F) >> 2); 1010aeb99916SMilanka Ringwald switch (avctp_packet_type){ 1011aeb99916SMilanka Ringwald case AVCTP_SINGLE_PACKET: 1012aeb99916SMilanka Ringwald break; 1013aeb99916SMilanka Ringwald 1014aeb99916SMilanka Ringwald #ifdef ENABLE_AVCTP_FRAGMENTATION 1015aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 1016aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 1017aeb99916SMilanka Ringwald avctp_reassemble_message(connection, avctp_packet_type, packet, size); 1018aeb99916SMilanka Ringwald return; 1019aeb99916SMilanka Ringwald 1020aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 1021aeb99916SMilanka Ringwald avctp_reassemble_message(connection, avctp_packet_type, packet, size); 1022aeb99916SMilanka Ringwald 1023aeb99916SMilanka Ringwald packet = connection->avctp_reassembly_buffer; 1024aeb99916SMilanka Ringwald size = connection->avctp_reassembly_size; 1025aeb99916SMilanka Ringwald break; 1026aeb99916SMilanka Ringwald #endif 1027aeb99916SMilanka Ringwald default: 1028aeb99916SMilanka Ringwald return; 1029aeb99916SMilanka Ringwald } 1030697b823eSMatthias Ringwald 1031697b823eSMatthias Ringwald if (size < 6u) return; 10324b338011SMilanka Ringwald 1033aeb99916SMilanka Ringwald uint16_t pid = big_endian_read_16(packet, 1); 1034be4cc80aSMilanka Ringwald 103567049435SMilanka Ringwald if (pid != BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL){ 10363b899686SMilanka Ringwald log_info("Invalid pid 0x%02x, expected 0x%02x", pid, BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL); 1037393f5724SMilanka Ringwald connection->target_reject_transport_header = true; 1038393f5724SMilanka Ringwald connection->target_invalid_pid = pid; 10394e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 104067049435SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 104167049435SMilanka Ringwald return; 104267049435SMilanka Ringwald } 104367049435SMilanka Ringwald 1044aeb99916SMilanka Ringwald // avrcp_subunit_type_t subunit_type = (avrcp_subunit_type_t) (packet[4] >> 3); 1045aeb99916SMilanka Ringwald // avrcp_subunit_id_t subunit_id = (avrcp_subunit_id_t) (packet[4] & 0x07); 10464b338011SMilanka Ringwald 1047dfbbc26bSMatthias Ringwald avrcp_command_opcode_t opcode = (avrcp_command_opcode_t) avrcp_cmd_opcode(packet,size); 1048e0bbf3edSMilanka Ringwald 1049e8ad6d13SMatthias Ringwald int pos = 6; 1050dfbbc26bSMatthias Ringwald uint16_t length; 1051dfbbc26bSMatthias Ringwald avrcp_pdu_id_t pdu_id; 1052aeb99916SMilanka Ringwald // connection->data_len = 0; 10532c197cf1SMilanka Ringwald uint8_t offset; 10540b578d06SMilanka Ringwald uint8_t operand; 1055367e3c55SMilanka Ringwald uint16_t event_mask; 105638410698SMilanka Ringwald avrcp_operation_id_t operation_id; 1057e0bbf3edSMilanka Ringwald 1058e0bbf3edSMilanka Ringwald switch (opcode){ 10594b338011SMilanka Ringwald case AVRCP_CMD_OPCODE_UNIT_INFO: 1060d1207cd8SMilanka Ringwald avrcp_target_unit_info(connection); 10614b338011SMilanka Ringwald break; 10622c197cf1SMilanka Ringwald case AVRCP_CMD_OPCODE_SUBUNIT_INFO: 106399702381SMatthias Ringwald if ((size - pos) < 3) return; 10642c197cf1SMilanka Ringwald // page: packet[pos] >> 4, 10652c197cf1SMilanka Ringwald offset = 4 * (packet[pos]>>4); 10662c197cf1SMilanka Ringwald // extension code (fixed 7) = packet[pos] & 0x0F 10672c197cf1SMilanka Ringwald // 4 bytes paga data, all 0xFF 1068d1207cd8SMilanka Ringwald avrcp_target_subunit_info(connection, offset); 1069e0bbf3edSMilanka Ringwald break; 10702c197cf1SMilanka Ringwald 10710b578d06SMilanka Ringwald case AVRCP_CMD_OPCODE_PASS_THROUGH: 1072810b0d84SMatthias Ringwald if (size < 8) return; 1073810b0d84SMatthias Ringwald log_info("AVRCP_OPERATION_ID 0x%02x, operands length %d", packet[6], packet[7]); 107438410698SMilanka Ringwald operation_id = (avrcp_operation_id_t) (packet[6] & 0x7f); 10750b578d06SMilanka Ringwald operand = 0; 1076810b0d84SMatthias Ringwald if ((packet[7] >= 1) && (size >= 9)){ 1077810b0d84SMatthias Ringwald operand = packet[8]; 1078810b0d84SMatthias Ringwald } 1079d1207cd8SMilanka Ringwald 10800b578d06SMilanka Ringwald if (avcrp_operation_id_is_valid(operation_id)){ 10810b578d06SMilanka Ringwald bool button_pressed = (packet[6] & 0x80) == 0; 10820b578d06SMilanka Ringwald 1083810b0d84SMatthias Ringwald avrcp_target_operation_accepted(connection->avrcp_cid, (avrcp_operation_id_t) packet[6], packet[7], operand); 1084319131f8SMatthias Ringwald avrcp_target_emit_operation(avrcp_target_context.avrcp_callback, connection->avrcp_cid, 1085927fd23fSMatthias Ringwald operation_id, button_pressed, packet[7], operand); 10860b578d06SMilanka Ringwald } else { 1087810b0d84SMatthias Ringwald avrcp_target_operation_not_implemented(connection->avrcp_cid, (avrcp_operation_id_t) packet[6], packet[7], operand); 1088d1207cd8SMilanka Ringwald } 1089d1207cd8SMilanka Ringwald break; 10900b578d06SMilanka Ringwald 1091d1207cd8SMilanka Ringwald 1092e0bbf3edSMilanka Ringwald case AVRCP_CMD_OPCODE_VENDOR_DEPENDENT: 109399702381SMatthias Ringwald 109499702381SMatthias Ringwald if (size < 13) return; 109599702381SMatthias Ringwald 109699702381SMatthias Ringwald // pos = 6 - company id 1097393f5724SMilanka Ringwald (void)memcpy(connection->message_body, &packet[pos], 3); 1098aeb99916SMilanka Ringwald // connection->data_len = 3; 109999702381SMatthias Ringwald pos += 3; 110099702381SMatthias Ringwald // pos = 9 110199702381SMatthias Ringwald pdu_id = (avrcp_pdu_id_t) packet[pos++]; 110299702381SMatthias Ringwald // 1 - reserved 110399702381SMatthias Ringwald pos++; 110499702381SMatthias Ringwald // 2-3 param length, 110599702381SMatthias Ringwald length = big_endian_read_16(packet, pos); 110699702381SMatthias Ringwald pos += 2; 110799702381SMatthias Ringwald // pos = 13 1108e0bbf3edSMilanka Ringwald switch (pdu_id){ 1109*1e4db407SMilanka Ringwald case AVRCP_PDU_ID_PLAY_ITEM: 1110*1e4db407SMilanka Ringwald if ( (pos + 11) > size){ 1111*1e4db407SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 1112*1e4db407SMilanka Ringwald return; 1113*1e4db407SMilanka Ringwald } 1114*1e4db407SMilanka Ringwald 1115*1e4db407SMilanka Ringwald connection->target_scope = packet[pos++]; 1116*1e4db407SMilanka Ringwald if (connection->target_scope >= AVRCP_BROWSING_RFU){ 1117*1e4db407SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 1118*1e4db407SMilanka Ringwald return; 1119*1e4db407SMilanka Ringwald } 1120*1e4db407SMilanka Ringwald memcpy(connection->target_track_id, &packet[pos], 8); 1121*1e4db407SMilanka Ringwald pos += 8; 1122*1e4db407SMilanka Ringwald connection->target_uid_counter = big_endian_read_16(packet,pos); 1123*1e4db407SMilanka Ringwald connection->state = AVCTP_W2_CHECK_DATABASE; 1124*1e4db407SMilanka Ringwald avrcp_target_emit_respond_play_item(avrcp_target_context.avrcp_callback, connection->avrcp_cid, connection->target_uid_counter, connection->target_scope,connection->target_track_id); 1125*1e4db407SMilanka Ringwald break; 1126*1e4db407SMilanka Ringwald 11276568eb54SMilanka Ringwald case AVRCP_PDU_ID_SET_ADDRESSED_PLAYER:{ 112899702381SMatthias Ringwald if ((pos + 2) > size) return; 1129d1cf25b1SMatthias Ringwald bool ok = length == 4; 1130d1cf25b1SMatthias Ringwald if (avrcp_target_context.set_addressed_player_callback != NULL){ 113199702381SMatthias Ringwald uint16_t player_id = big_endian_read_16(packet, pos); 1132d1cf25b1SMatthias Ringwald ok = avrcp_target_context.set_addressed_player_callback(player_id); 11336568eb54SMilanka Ringwald } 1134d1cf25b1SMatthias Ringwald if (ok){ 1135aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, AVRCP_STATUS_SUCCESS); 1136d1cf25b1SMatthias Ringwald } else { 1137aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PLAYER_ID); 1138d1cf25b1SMatthias Ringwald } 11396568eb54SMilanka Ringwald break; 11406568eb54SMilanka Ringwald } 1141e0bbf3edSMilanka Ringwald case AVRCP_PDU_ID_GET_CAPABILITIES:{ 114299702381SMatthias Ringwald avrcp_capability_id_t capability_id = (avrcp_capability_id_t) packet[pos]; 1143e0bbf3edSMilanka Ringwald switch (capability_id){ 1144e0bbf3edSMilanka Ringwald case AVRCP_CAPABILITY_ID_EVENT: 1145aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_supported_events(connection); 1146e0bbf3edSMilanka Ringwald break; 1147e0bbf3edSMilanka Ringwald case AVRCP_CAPABILITY_ID_COMPANY: 1148aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_supported_companies(connection); 1149e0bbf3edSMilanka Ringwald break; 1150e0bbf3edSMilanka Ringwald default: 1151aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 1152e0bbf3edSMilanka Ringwald break; 1153e0bbf3edSMilanka Ringwald } 1154e0bbf3edSMilanka Ringwald break; 1155e0bbf3edSMilanka Ringwald } 1156c045af99SMilanka Ringwald case AVRCP_PDU_ID_GET_PLAY_STATUS: 1157c045af99SMilanka Ringwald avrcp_target_emit_respond_vendor_dependent_query(avrcp_target_context.avrcp_callback, connection->avrcp_cid, AVRCP_SUBEVENT_PLAY_STATUS_QUERY); 1158c045af99SMilanka Ringwald break; 11591bf7a74fSMilanka Ringwald case AVRCP_PDU_ID_REQUEST_ABORT_CONTINUING_RESPONSE: 116099702381SMatthias Ringwald if ((pos + 1) > size) return; 1161393f5724SMilanka Ringwald connection->target_abort_continue_response = true; 11624e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 11631bf7a74fSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 11641bf7a74fSMilanka Ringwald break; 1165d1207cd8SMilanka Ringwald case AVRCP_PDU_ID_REQUEST_CONTINUING_RESPONSE: 116699702381SMatthias Ringwald if ((pos + 1) > size) return; 116799702381SMatthias Ringwald if (packet[pos] != AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES){ 1168aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 1169d1207cd8SMilanka Ringwald return; 1170d1207cd8SMilanka Ringwald } 11714b47bcb9SMilanka Ringwald connection->target_continue_response = true; 1172393f5724SMilanka Ringwald connection->target_now_playing_info_response = true; 11734e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 1174d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1175d1207cd8SMilanka Ringwald break; 11765e3d4d2bSMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES:{ 117799702381SMatthias Ringwald if ((pos + 9) > size) return; 11785e3d4d2bSMilanka Ringwald uint8_t play_identifier[8]; 11795e3d4d2bSMilanka Ringwald memset(play_identifier, 0, 8); 118099702381SMatthias Ringwald if (memcmp(&packet[pos], play_identifier, 8) != 0) { 1181aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 11825e3d4d2bSMilanka Ringwald return; 11835e3d4d2bSMilanka Ringwald } 1184d1207cd8SMilanka Ringwald pos += 8; 118599702381SMatthias Ringwald uint8_t attribute_count = packet[pos++]; 1186d1207cd8SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_NONE; 1187d1207cd8SMilanka Ringwald if (!attribute_count){ 1188aeb99916SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_TITLE; 1189393f5724SMilanka Ringwald connection->target_now_playing_info_attr_bitmap = 0xFE; 1190d1207cd8SMilanka Ringwald } else { 1191d1207cd8SMilanka Ringwald int i; 1192aeb99916SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_TITLE; 1193393f5724SMilanka Ringwald connection->target_now_playing_info_attr_bitmap = 0; 1194aeb99916SMilanka Ringwald if ((pos + attribute_count * 4) > size) return; 1195d1207cd8SMilanka Ringwald for (i=0; i < attribute_count; i++){ 1196aeb99916SMilanka Ringwald uint32_t attr_id = big_endian_read_32(packet, pos); 1197393f5724SMilanka Ringwald connection->target_now_playing_info_attr_bitmap |= (1 << attr_id); 1198aeb99916SMilanka Ringwald pos += 4; 1199d1207cd8SMilanka Ringwald } 1200d1207cd8SMilanka Ringwald } 1201393f5724SMilanka Ringwald log_info("target_now_playing_info_attr_bitmap 0x%02x", connection->target_now_playing_info_attr_bitmap); 1202393f5724SMilanka Ringwald connection->target_now_playing_info_response = true; 12034e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 1204aeb99916SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1205d1207cd8SMilanka Ringwald break; 1206d1207cd8SMilanka Ringwald } 1207d1207cd8SMilanka Ringwald case AVRCP_PDU_ID_REGISTER_NOTIFICATION:{ 120899702381SMatthias Ringwald if ((pos + 1) > size) return; 120999702381SMatthias Ringwald avrcp_notification_event_id_t event_id = (avrcp_notification_event_id_t) packet[pos]; 1210367e3c55SMilanka Ringwald 1211022b77fcSMilanka Ringwald avrcp_target_set_transaction_label_for_notification(connection, event_id, connection->transaction_id); 1212d1207cd8SMilanka Ringwald 1213367e3c55SMilanka Ringwald if (event_id < AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED || 1214367e3c55SMilanka Ringwald event_id > AVRCP_NOTIFICATION_EVENT_MAX_VALUE){ 1215aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 1216367e3c55SMilanka Ringwald return; 1217367e3c55SMilanka Ringwald } 1218367e3c55SMilanka Ringwald 1219367e3c55SMilanka Ringwald switch (event_id){ 1220367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_AVAILABLE_PLAYERS_CHANGED: 1221367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYER_APPLICATION_SETTING_CHANGED: 12224b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_REACHED_END: 12234b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_REACHED_START: 12244b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYBACK_POS_CHANGED: 12254b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_SYSTEM_STATUS_CHANGED: 12264b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_MAX_VALUE: 1227aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_not_implemented(connection, pdu_id, event_id); 1228367e3c55SMilanka Ringwald return; 1229367e3c55SMilanka Ringwald default: 1230367e3c55SMilanka Ringwald break; 1231367e3c55SMilanka Ringwald } 1232367e3c55SMilanka Ringwald 1233367e3c55SMilanka Ringwald event_mask = (1 << event_id); 1234367e3c55SMilanka Ringwald connection->notifications_enabled |= event_mask; 1235367e3c55SMilanka Ringwald 1236d1207cd8SMilanka Ringwald switch (event_id){ 12376a2e66beSMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_UIDS_CHANGED:{ 12386a2e66beSMilanka Ringwald if (connection->target_uid_counter == 0){ 12396a2e66beSMilanka Ringwald connection->target_uid_counter = 1; 12406a2e66beSMilanka Ringwald } 12416a2e66beSMilanka Ringwald uint8_t value[2]; 12426a2e66beSMilanka Ringwald big_endian_store_16(value, 0, connection->target_uid_counter); 12436a2e66beSMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, value, 2); 12446a2e66beSMilanka Ringwald break; 12456a2e66beSMilanka Ringwald } 12466a2e66beSMilanka Ringwald 1247d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED: 1248393f5724SMilanka Ringwald if (connection->target_track_selected){ 1249bbc50dcbSMilanka Ringwald if (connection->target_total_tracks > 0){ 1250bbc50dcbSMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, connection->target_track_id, 8); 1251bbc50dcbSMilanka Ringwald break; 1252bbc50dcbSMilanka Ringwald } 1253d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, AVRCP_NOTIFICATION_TRACK_SELECTED, 8); 1254d1207cd8SMilanka Ringwald } else { 1255d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, AVRCP_NOTIFICATION_TRACK_NOT_SELECTED, 8); 1256d1207cd8SMilanka Ringwald } 1257d1207cd8SMilanka Ringwald break; 1258d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED: 1259393f5724SMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->target_playback_status, 1); 1260d1207cd8SMilanka Ringwald break; 1261d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED: 1262d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, NULL, 0); 1263d1207cd8SMilanka Ringwald break; 1264d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED: 1265393f5724SMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->target_absolute_volume, 1); 1266d1207cd8SMilanka Ringwald break; 1267d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED: 1268393f5724SMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->target_battery_status, 1); 1269d1207cd8SMilanka Ringwald break; 12706568eb54SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED: 1271aeb99916SMilanka Ringwald avrcp_target_response_addressed_player_changed_interim(connection, pdu_id, event_id); 12726568eb54SMilanka Ringwald return; 1273d1207cd8SMilanka Ringwald default: 1274367e3c55SMilanka Ringwald btstack_assert(false); 1275d1207cd8SMilanka Ringwald return; 1276d1207cd8SMilanka Ringwald } 12775e3d4d2bSMilanka Ringwald break; 12785e3d4d2bSMilanka Ringwald } 127912b7c8ecSMilanka Ringwald case AVRCP_PDU_ID_SET_ABSOLUTE_VOLUME: { 1280a215e8c0SMilanka Ringwald if ((pos + 1) > size ){ 1281aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 128212b7c8ecSMilanka Ringwald break; 128312b7c8ecSMilanka Ringwald } 128412b7c8ecSMilanka Ringwald 1285a215e8c0SMilanka Ringwald if (length != 1){ 1286a215e8c0SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_SPECIFIED_PARAMETER_NOT_FOUND); 1287a215e8c0SMilanka Ringwald break; 12880a2feef4SMilanka Ringwald } 1289a215e8c0SMilanka Ringwald 1290a215e8c0SMilanka Ringwald uint8_t absolute_volume = packet[pos]; 1291a215e8c0SMilanka Ringwald if (absolute_volume >= 0x80) { 1292a215e8c0SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_SPECIFIED_PARAMETER_NOT_FOUND); 1293a215e8c0SMilanka Ringwald break; 1294a215e8c0SMilanka Ringwald } 1295a215e8c0SMilanka Ringwald connection->target_absolute_volume = absolute_volume; 1296a215e8c0SMilanka Ringwald 1297393f5724SMilanka Ringwald avrcp_target_emit_volume_changed(avrcp_target_context.avrcp_callback, connection->avrcp_cid, connection->target_absolute_volume); 1298393f5724SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, connection->target_absolute_volume); 129912b7c8ecSMilanka Ringwald break; 130012b7c8ecSMilanka Ringwald } 1301e0bbf3edSMilanka Ringwald default: 13021bf7a74fSMilanka Ringwald log_info("AVRCP target: unhandled pdu id 0x%02x", pdu_id); 1303aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 1304e0bbf3edSMilanka Ringwald break; 1305e0bbf3edSMilanka Ringwald } 13064b338011SMilanka Ringwald break; 13074b338011SMilanka Ringwald default: 13081bf7a74fSMilanka Ringwald log_info("AVRCP target: opcode 0x%02x not implemented", avrcp_cmd_opcode(packet,size)); 13094b338011SMilanka Ringwald break; 13104b338011SMilanka Ringwald } 131101dc6e35SMilanka Ringwald } 131201dc6e35SMilanka Ringwald 1313aeb99916SMilanka Ringwald static void avrcp_target_notification_init(avrcp_connection_t * connection, avrcp_notification_event_id_t notification_id, uint8_t * value, uint16_t value_len){ 13148b631c13SMatthias Ringwald btstack_assert(value_len + 1 < AVRCP_MAX_COMMAND_PARAMETER_LENGTH); 1315aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_CHANGED_STABLE, AVRCP_PDU_ID_REGISTER_NOTIFICATION); 1316022b77fcSMilanka Ringwald connection->transaction_id = avrcp_target_get_transaction_label_for_notification(connection, notification_id); 1317d1207cd8SMilanka Ringwald 1318aeb99916SMilanka Ringwald connection->data_len = 1 + value_len; 1319aeb99916SMilanka Ringwald connection->data[0] = notification_id; 1320276c795aSMilanka Ringwald if (value != NULL){ 1321aeb99916SMilanka Ringwald (void)memcpy(connection->data + 1, value, value_len); 1322aeb99916SMilanka Ringwald } 1323df842b64SMatthias Ringwald } 1324bf67b2dbSMatthias Ringwald 1325aeb99916SMilanka Ringwald static void avrcp_target_notification_addressed_player_changed_init(avrcp_connection_t * connection){ 1326aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_CHANGED_STABLE, AVRCP_PDU_ID_REGISTER_NOTIFICATION); 1327aeb99916SMilanka Ringwald connection->transaction_id = avrcp_target_get_transaction_label_for_notification(connection, AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED); 1328aeb99916SMilanka Ringwald 1329aeb99916SMilanka Ringwald connection->data_len = 5; 1330aeb99916SMilanka Ringwald connection->data[0] = AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED; 1331393f5724SMilanka Ringwald big_endian_store_16(connection->data, 1, connection->target_addressed_player_id); 1332393f5724SMilanka Ringwald big_endian_store_16(connection->data, 3, connection->target_uid_counter); 1333d1207cd8SMilanka Ringwald } 1334d1207cd8SMilanka Ringwald 1335aeb99916SMilanka Ringwald 13360ec79bd0SMilanka Ringwald static void avrcp_target_reset_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification_id){ 1337aeb99916SMilanka Ringwald if (notification_id < AVRCP_NOTIFICATION_EVENT_FIRST_INDEX || notification_id > AVRCP_NOTIFICATION_EVENT_LAST_INDEX){ 133812b7c8ecSMilanka Ringwald return; 133912b7c8ecSMilanka Ringwald } 134012b7c8ecSMilanka Ringwald connection->notifications_enabled &= ~(1 << notification_id); 1341393f5724SMilanka Ringwald connection->target_notifications_transaction_label[notification_id] = 0; 1342aeb99916SMilanka Ringwald } 1343aeb99916SMilanka Ringwald 1344aeb99916SMilanka Ringwald static void avrcp_request_next_avctp_segment(avrcp_connection_t * connection){ 1345aeb99916SMilanka Ringwald // AVCTP 1346aeb99916SMilanka Ringwald switch (connection->avctp_packet_type){ 1347aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 1348ccfcd81eSMilanka Ringwald case AVCTP_SINGLE_PACKET: 1349aeb99916SMilanka Ringwald connection->state = AVCTP_CONNECTION_OPENED; 1350aeb99916SMilanka Ringwald break; 1351aeb99916SMilanka Ringwald default: 1352ffe8f3acSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 1353aeb99916SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1354aeb99916SMilanka Ringwald break; 1355aeb99916SMilanka Ringwald } 135612b7c8ecSMilanka Ringwald } 135712b7c8ecSMilanka Ringwald 1358654724deSMilanka Ringwald static void avrcp_target_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 135901dc6e35SMilanka Ringwald avrcp_connection_t * connection; 1360ffe8f3acSMilanka Ringwald avrcp_notification_event_id_t notification_id = AVRCP_NOTIFICATION_EVENT_NONE; 1361aeb99916SMilanka Ringwald 136201dc6e35SMilanka Ringwald switch (packet_type){ 136301dc6e35SMilanka Ringwald case L2CAP_DATA_PACKET: 13641945fe3eSMilanka Ringwald connection = avrcp_get_connection_for_l2cap_signaling_cid_for_role(AVRCP_TARGET, channel); 136501dc6e35SMilanka Ringwald avrcp_handle_l2cap_data_packet_for_signaling_connection(connection, packet, size); 1366ffe8f3acSMilanka Ringwald return; 1367ffe8f3acSMilanka Ringwald 136801dc6e35SMilanka Ringwald case HCI_EVENT_PACKET: 1369ffe8f3acSMilanka Ringwald if (hci_event_packet_get_type(packet) != L2CAP_EVENT_CAN_SEND_NOW){ 1370ffe8f3acSMilanka Ringwald return; 1371ffe8f3acSMilanka Ringwald } 1372ffe8f3acSMilanka Ringwald 13731945fe3eSMilanka Ringwald connection = avrcp_get_connection_for_l2cap_signaling_cid_for_role(AVRCP_TARGET, channel); 1374ffe8f3acSMilanka Ringwald if (connection == NULL){ 1375ffe8f3acSMilanka Ringwald return; 1376ffe8f3acSMilanka Ringwald } 1377ffe8f3acSMilanka Ringwald 13784e0673dbSMilanka Ringwald if (connection->state == AVCTP_W2_SEND_RESPONSE){ 1379d3db1226SMilanka Ringwald // start AVCTP 1380393f5724SMilanka Ringwald if (connection->target_reject_transport_header){ 1381393f5724SMilanka Ringwald connection->target_reject_transport_header = false; 1382ffe8f3acSMilanka Ringwald avctp_send_reject_cmd_wrong_pid(connection); 1383ffe8f3acSMilanka Ringwald connection->state = AVCTP_CONNECTION_OPENED; 1384ffe8f3acSMilanka Ringwald return; 1385ffe8f3acSMilanka Ringwald } 1386d3db1226SMilanka Ringwald // end AVCTP 1387d1207cd8SMilanka Ringwald 1388d3db1226SMilanka Ringwald // start AVRCP 1389393f5724SMilanka Ringwald if (connection->target_abort_continue_response){ 1390393f5724SMilanka Ringwald connection->target_abort_continue_response = false; 1391aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, AVRCP_PDU_ID_REQUEST_ABORT_CONTINUING_RESPONSE); 1392ffe8f3acSMilanka Ringwald break; 13931bf7a74fSMilanka Ringwald } 13941bf7a74fSMilanka Ringwald 1395393f5724SMilanka Ringwald if (connection->target_now_playing_info_response){ 1396393f5724SMilanka Ringwald connection->target_now_playing_info_response = false; 13974b47bcb9SMilanka Ringwald if (connection->target_continue_response){ 13984e0673dbSMilanka Ringwald connection->target_continue_response = false; 13991e852c68SMilanka Ringwald if (connection->data_len == 0){ 14001e852c68SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, connection->pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 14011e852c68SMilanka Ringwald return; 14021e852c68SMilanka Ringwald } 14034b47bcb9SMilanka Ringwald } else { 1404e2ceb2dfSMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES); 1405aeb99916SMilanka Ringwald connection->data_len = avrcp_now_playing_info_value_len_with_headers(connection); 14064b47bcb9SMilanka Ringwald } 1407ffe8f3acSMilanka Ringwald break; 14085e3d4d2bSMilanka Ringwald } 1409d1207cd8SMilanka Ringwald 14104e0673dbSMilanka Ringwald // data already prepared 14114e0673dbSMilanka Ringwald break; 14124e0673dbSMilanka Ringwald } 14134e0673dbSMilanka Ringwald 14144e0673dbSMilanka Ringwald // Notifications 14154e0673dbSMilanka Ringwald 1416393f5724SMilanka Ringwald if (connection->target_track_changed){ 1417393f5724SMilanka Ringwald connection->target_track_changed = false; 1418ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED; 1419393f5724SMilanka Ringwald avrcp_target_notification_init(connection, notification_id, connection->target_track_id, 8); 1420ffe8f3acSMilanka Ringwald break; 1421d1207cd8SMilanka Ringwald } 1422d1207cd8SMilanka Ringwald 1423393f5724SMilanka Ringwald if (connection->target_playback_status_changed){ 1424393f5724SMilanka Ringwald connection->target_playback_status_changed = false; 1425ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED; 1426393f5724SMilanka Ringwald uint8_t playback_status = (uint8_t) connection->target_playback_status; 1427ffe8f3acSMilanka Ringwald avrcp_target_notification_init(connection, notification_id, &playback_status, 1); 1428ffe8f3acSMilanka Ringwald break; 1429d1207cd8SMilanka Ringwald } 1430d1207cd8SMilanka Ringwald 1431393f5724SMilanka Ringwald if (connection->target_playing_content_changed){ 1432393f5724SMilanka Ringwald connection->target_playing_content_changed = false; 1433ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED; 1434276c795aSMilanka Ringwald avrcp_target_notification_init(connection, notification_id, NULL, 0); 1435ffe8f3acSMilanka Ringwald break; 1436d1207cd8SMilanka Ringwald } 1437d1207cd8SMilanka Ringwald 1438393f5724SMilanka Ringwald if (connection->target_battery_status_changed){ 1439393f5724SMilanka Ringwald connection->target_battery_status_changed = false; 1440ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED; 1441393f5724SMilanka Ringwald avrcp_target_notification_init(connection, notification_id, (uint8_t *)&connection->target_battery_status, 1); 1442ffe8f3acSMilanka Ringwald break; 1443d1207cd8SMilanka Ringwald } 1444d1207cd8SMilanka Ringwald 1445393f5724SMilanka Ringwald if (connection->target_notify_absolute_volume_changed){ 1446393f5724SMilanka Ringwald connection->target_notify_absolute_volume_changed = false; 1447ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED; 1448393f5724SMilanka Ringwald avrcp_target_notification_init(connection, notification_id, &connection->target_absolute_volume, 1); 1449ffe8f3acSMilanka Ringwald break; 1450d1207cd8SMilanka Ringwald } 1451d1207cd8SMilanka Ringwald 1452393f5724SMilanka Ringwald if (connection->target_addressed_player_changed){ 1453393f5724SMilanka Ringwald connection->target_addressed_player_changed = false; 1454ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED; 1455aeb99916SMilanka Ringwald avrcp_target_notification_addressed_player_changed_init(connection); 1456ffe8f3acSMilanka Ringwald break; 1457ffe8f3acSMilanka Ringwald } 1458ffe8f3acSMilanka Ringwald 14596a2e66beSMilanka Ringwald if (connection->target_uids_changed){ 14606a2e66beSMilanka Ringwald connection->target_uids_changed = false; 14616a2e66beSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_UIDS_CHANGED; 14626a2e66beSMilanka Ringwald uint8_t value[2]; 14636a2e66beSMilanka Ringwald big_endian_store_16(value, 0, connection->target_uid_counter); 14646a2e66beSMilanka Ringwald avrcp_target_notification_init(connection, notification_id, value, 2); 14656a2e66beSMilanka Ringwald break; 14666a2e66beSMilanka Ringwald } 146779774d78SMilanka Ringwald // nothing to send, exit 146879774d78SMilanka Ringwald return; 1469ffe8f3acSMilanka Ringwald 1470ffe8f3acSMilanka Ringwald default: 1471f1028954SMilanka Ringwald return; 1472f1028954SMilanka Ringwald } 1473f1028954SMilanka Ringwald 1474aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 1475ffe8f3acSMilanka Ringwald avrcp_target_reset_notification(connection, notification_id); 14763b899686SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 147701dc6e35SMilanka Ringwald } 147801dc6e35SMilanka Ringwald 147901dc6e35SMilanka Ringwald void avrcp_target_init(void){ 148001dc6e35SMilanka Ringwald avrcp_target_context.role = AVRCP_TARGET; 1481654724deSMilanka Ringwald avrcp_target_context.packet_handler = avrcp_target_packet_handler; 14820ec79bd0SMilanka Ringwald avrcp_register_target_packet_handler(&avrcp_target_packet_handler); 148301dc6e35SMilanka Ringwald } 148401dc6e35SMilanka Ringwald 1485680af5dcSMatthias Ringwald void avrcp_target_deinit(void){ 1486680af5dcSMatthias Ringwald memset(&avrcp_target_context, 0, sizeof(avrcp_context_t)); 1487680af5dcSMatthias Ringwald } 1488680af5dcSMatthias Ringwald 148901dc6e35SMilanka Ringwald void avrcp_target_register_packet_handler(btstack_packet_handler_t callback){ 1490be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 149101dc6e35SMilanka Ringwald avrcp_target_context.avrcp_callback = callback; 149201dc6e35SMilanka Ringwald } 149301dc6e35SMilanka Ringwald 1494d1cf25b1SMatthias Ringwald void avrcp_target_register_set_addressed_player_handler(bool (*callback)(uint16_t player_id)){ 149595f00c7eSMilanka Ringwald btstack_assert(callback != NULL); 1496d1cf25b1SMatthias Ringwald avrcp_target_context.set_addressed_player_callback = callback; 1497d1cf25b1SMatthias Ringwald } 1498d1cf25b1SMatthias Ringwald 149901dc6e35SMilanka Ringwald 1500*1e4db407SMilanka Ringwald uint8_t avrcp_send_response_to_play_item(uint16_t avrcp_cid, avrcp_status_code_t status){ 1501*1e4db407SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(avrcp_cid, AVRCP_TARGET); 1502*1e4db407SMilanka Ringwald if (connection == NULL){ 1503*1e4db407SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1504*1e4db407SMilanka Ringwald } 1505*1e4db407SMilanka Ringwald if (connection->state != AVCTP_W2_CHECK_DATABASE){ 1506*1e4db407SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 1507*1e4db407SMilanka Ringwald } 1508*1e4db407SMilanka Ringwald if (status != AVRCP_STATUS_SUCCESS){ 1509*1e4db407SMilanka Ringwald return avrcp_target_response_vendor_dependent_reject(connection, AVRCP_PDU_ID_PLAY_ITEM, status); 1510*1e4db407SMilanka Ringwald } 1511*1e4db407SMilanka Ringwald return avrcp_target_vendor_dependent_response_accept(connection, AVRCP_PDU_ID_PLAY_ITEM, status); 1512*1e4db407SMilanka Ringwald } 1513