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 60e3d57ee2SMilanka Ringwald avrcp_context_t avrcp_target_context; 6101dc6e35SMilanka Ringwald 62aeb99916SMilanka Ringwald static uint32_t default_companies[] = { 63aeb99916SMilanka Ringwald 0x581900 //BT SIG registered CompanyID 64aeb99916SMilanka Ringwald }; 65aeb99916SMilanka Ringwald 6611014891SMilanka Ringwald static int avrcp_target_supports_browsing(uint16_t target_supported_features){ 67e2f25417SMilanka Ringwald return target_supported_features & AVRCP_FEATURE_MASK_BROWSING; 6811014891SMilanka Ringwald } 6911014891SMilanka Ringwald 704f0111ebSMilanka 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){ 7111014891SMilanka Ringwald avrcp_create_sdp_record(0, service, service_record_handle, avrcp_target_supports_browsing(supported_features), supported_features, service_name, service_provider_name); 7201dc6e35SMilanka Ringwald } 7301dc6e35SMilanka Ringwald 74319131f8SMatthias Ringwald static void 75319131f8SMatthias Ringwald avrcp_target_emit_operation(btstack_packet_handler_t callback, uint16_t avrcp_cid, avrcp_operation_id_t operation_id, 76319131f8SMatthias Ringwald bool button_pressed, uint8_t operands_length, uint8_t operand) { 77be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 78be4cc80aSMilanka Ringwald 79319131f8SMatthias Ringwald uint8_t event[9]; 80831d3fd5SMilanka Ringwald int pos = 0; 81831d3fd5SMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META; 82831d3fd5SMilanka Ringwald event[pos++] = sizeof(event) - 2; 83831d3fd5SMilanka Ringwald event[pos++] = AVRCP_SUBEVENT_OPERATION; 84831d3fd5SMilanka Ringwald little_endian_store_16(event, pos, avrcp_cid); 85831d3fd5SMilanka Ringwald pos += 2; 86831d3fd5SMilanka Ringwald event[pos++] = operation_id; 87319131f8SMatthias Ringwald event[pos++] = button_pressed ? 1 : 0; 88831d3fd5SMilanka Ringwald event[pos++] = operands_length; 89831d3fd5SMilanka Ringwald event[pos++] = operand; 90831d3fd5SMilanka Ringwald (*callback)(HCI_EVENT_PACKET, 0, event, sizeof(event)); 91831d3fd5SMilanka Ringwald } 92831d3fd5SMilanka Ringwald 930ec79bd0SMilanka Ringwald static void avrcp_target_emit_volume_changed(btstack_packet_handler_t callback, uint16_t avrcp_cid, uint8_t absolute_volume){ 94be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 95be4cc80aSMilanka Ringwald 960ec79bd0SMilanka Ringwald uint8_t event[7]; 970ec79bd0SMilanka Ringwald int offset = 0; 980ec79bd0SMilanka Ringwald event[offset++] = HCI_EVENT_AVRCP_META; 990ec79bd0SMilanka Ringwald event[offset++] = sizeof(event) - 2; 1000ec79bd0SMilanka Ringwald event[offset++] = AVRCP_SUBEVENT_NOTIFICATION_VOLUME_CHANGED; 1010ec79bd0SMilanka Ringwald little_endian_store_16(event, offset, avrcp_cid); 1020ec79bd0SMilanka Ringwald offset += 2; 1030ec79bd0SMilanka Ringwald event[offset++] = AVRCP_CTYPE_NOTIFY; 1040ec79bd0SMilanka Ringwald event[offset++] = absolute_volume; 1050ec79bd0SMilanka Ringwald (*callback)(HCI_EVENT_PACKET, 0, event, sizeof(event)); 1060ec79bd0SMilanka Ringwald } 1070ec79bd0SMilanka Ringwald 108c045af99SMilanka Ringwald static void avrcp_target_emit_respond_vendor_dependent_query(btstack_packet_handler_t callback, uint16_t avrcp_cid, uint8_t subevent_id){ 109be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 110be4cc80aSMilanka Ringwald 111c045af99SMilanka Ringwald uint8_t event[5]; 112e0bbf3edSMilanka Ringwald int pos = 0; 113e0bbf3edSMilanka Ringwald event[pos++] = HCI_EVENT_AVRCP_META; 114e0bbf3edSMilanka Ringwald event[pos++] = sizeof(event) - 2; 115e0bbf3edSMilanka Ringwald event[pos++] = subevent_id; 116e0bbf3edSMilanka Ringwald little_endian_store_16(event, pos, avrcp_cid); 1174b338011SMilanka Ringwald (*callback)(HCI_EVENT_PACKET, 0, event, sizeof(event)); 1184b338011SMilanka Ringwald } 1194b338011SMilanka Ringwald 1205e845fceSMatthias Ringwald // returns number of bytes stored 121aeb99916SMilanka 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){ 122aeb99916SMilanka Ringwald btstack_assert(attr_id > AVRCP_MEDIA_ATTR_ALL); 123aeb99916SMilanka Ringwald btstack_assert(attr_id < AVRCP_MEDIA_ATTR_RESERVED); 124aeb99916SMilanka Ringwald uint16_t pos = 0; 125aeb99916SMilanka Ringwald big_endian_store_32(buffer, pos, attr_id); 126aeb99916SMilanka Ringwald big_endian_store_16(buffer, pos + 4, RFC2978_CHARSET_MIB_UTF8); 127aeb99916SMilanka Ringwald big_endian_store_16(buffer, pos + 6, attr_value_size); 1285e845fceSMatthias Ringwald return 8; 1295e845fceSMatthias Ringwald } 1305e845fceSMatthias Ringwald 131aeb99916SMilanka Ringwald static uint16_t avrcp_now_playing_info_attr_id_value_len(avrcp_connection_t * connection, avrcp_media_attribute_id_t attr_id){ 132aeb99916SMilanka Ringwald char buffer[AVRCP_MAX_ATTRIBUTE_SIZE]; 133aeb99916SMilanka Ringwald uint16_t str_len; 1345e3d4d2bSMilanka Ringwald switch (attr_id) { 1352f511ffbSMilanka Ringwald case AVRCP_MEDIA_ATTR_ALL: 136d1207cd8SMilanka Ringwald case AVRCP_MEDIA_ATTR_NONE: 137aeb99916SMilanka Ringwald return 0; 1385e3d4d2bSMilanka Ringwald case AVRCP_MEDIA_ATTR_TRACK: 139*ab2445a0SMatthias Ringwald str_len = snprintf(buffer, sizeof(buffer), "%" PRIu32, connection->target_track_nr); 1405e3d4d2bSMilanka Ringwald break; 141864d08b0SMilanka Ringwald case AVRCP_MEDIA_ATTR_TOTAL_NUM_ITEMS: 142*ab2445a0SMatthias Ringwald str_len = snprintf(buffer, sizeof(buffer), "%" PRIu32, connection->target_total_tracks); 1435e3d4d2bSMilanka Ringwald break; 1442f511ffbSMilanka Ringwald case AVRCP_MEDIA_ATTR_SONG_LENGTH_MS: 145*ab2445a0SMatthias Ringwald str_len = snprintf(buffer, sizeof(buffer), "%" PRIu32, connection->target_song_length_ms); 1465e3d4d2bSMilanka Ringwald break; 1475e3d4d2bSMilanka Ringwald default: 148393f5724SMilanka Ringwald str_len = connection->target_now_playing_info[(uint16_t)attr_id - 1].len; 1495e3d4d2bSMilanka Ringwald break; 1505e3d4d2bSMilanka Ringwald } 151aeb99916SMilanka Ringwald return str_len; 1525e3d4d2bSMilanka Ringwald } 1535e3d4d2bSMilanka Ringwald 154aeb99916SMilanka Ringwald static uint16_t avrcp_now_playing_info_value_len_with_headers(avrcp_connection_t * connection){ 155aeb99916SMilanka Ringwald uint16_t playing_info_len = 0; 156d1207cd8SMilanka Ringwald 157aeb99916SMilanka Ringwald uint8_t i; 158aeb99916SMilanka Ringwald for ( i = (uint8_t)AVRCP_MEDIA_ATTR_ALL + 1; i < (uint8_t) AVRCP_MEDIA_ATTR_RESERVED; i++){ 159aeb99916SMilanka Ringwald avrcp_media_attribute_id_t attr_id = (avrcp_media_attribute_id_t) i; 1605e3d4d2bSMilanka Ringwald 161393f5724SMilanka Ringwald if ((connection->target_now_playing_info_attr_bitmap & (1 << attr_id)) == 0) { 162aeb99916SMilanka Ringwald continue; 163aeb99916SMilanka Ringwald } 164aeb99916SMilanka Ringwald 165aeb99916SMilanka Ringwald switch (attr_id) { 166aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_ALL: 167aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_NONE: 168aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_DEFAULT_COVER_ART: 169aeb99916SMilanka Ringwald break; 170aeb99916SMilanka Ringwald default: 171aeb99916SMilanka Ringwald playing_info_len += AVRCP_ATTR_HEADER_LEN + avrcp_now_playing_info_attr_id_value_len(connection, attr_id); 172aeb99916SMilanka Ringwald break; 173aeb99916SMilanka Ringwald } 174aeb99916SMilanka Ringwald } 175ccfcd81eSMilanka Ringwald // for total num bytes that of the attributes + headers 176ccfcd81eSMilanka Ringwald playing_info_len += 1; 177aeb99916SMilanka Ringwald return playing_info_len; 178aeb99916SMilanka Ringwald } 179aeb99916SMilanka Ringwald 180aeb99916SMilanka Ringwald static uint8_t * avrcp_get_attribute_value_from_u32(avrcp_connection_t * connection, uint32_t value, uint16_t * num_bytes_to_copy){ 181aeb99916SMilanka Ringwald *num_bytes_to_copy = 0; 182aeb99916SMilanka Ringwald 183aeb99916SMilanka Ringwald if (connection->attribute_value_len == 0){ 184*ab2445a0SMatthias Ringwald // "4294967296" = 10 chars + \0 185*ab2445a0SMatthias Ringwald connection->attribute_value_len = snprintf((char *)connection->attribute_value, 11, "%" PRIu32, value); 186aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 187aeb99916SMilanka Ringwald } 188aeb99916SMilanka Ringwald *num_bytes_to_copy = connection->attribute_value_len - connection->attribute_value_offset; 189aeb99916SMilanka Ringwald return connection->attribute_value + connection->attribute_value_offset; 190aeb99916SMilanka Ringwald } 191aeb99916SMilanka Ringwald 192aeb99916SMilanka 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){ 193aeb99916SMilanka Ringwald switch (attr_id){ 194aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_TRACK: 195393f5724SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->target_track_nr, num_bytes_to_copy); 196aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_TOTAL_NUM_ITEMS: 197393f5724SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->target_total_tracks, num_bytes_to_copy); 198aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_SONG_LENGTH_MS: 199393f5724SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->target_song_length_ms, num_bytes_to_copy); 200aeb99916SMilanka Ringwald default: 201aeb99916SMilanka Ringwald break; 202aeb99916SMilanka Ringwald } 203aeb99916SMilanka Ringwald int attr_index = attr_id - 1; 204aeb99916SMilanka Ringwald if (connection->attribute_value_len == 0){ 205aeb99916SMilanka Ringwald connection->attribute_value_len = avrcp_now_playing_info_attr_id_value_len(connection, attr_id); 206aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 207aeb99916SMilanka Ringwald } 208393f5724SMilanka Ringwald *num_bytes_to_copy = connection->target_now_playing_info[attr_index].len - connection->attribute_value_offset; 209393f5724SMilanka Ringwald return (uint8_t *) (connection->target_now_playing_info[attr_index].value + connection->attribute_value_offset); 210aeb99916SMilanka Ringwald } 211aeb99916SMilanka Ringwald 212aeb99916SMilanka Ringwald // TODO Review 213aeb99916SMilanka Ringwald static uint16_t avrcp_store_avctp_now_playing_info_fragment(avrcp_connection_t * connection, uint16_t packet_size, uint8_t * packet){ 214aeb99916SMilanka Ringwald uint16_t num_free_bytes = packet_size; 215aeb99916SMilanka Ringwald 216aeb99916SMilanka Ringwald uint16_t bytes_stored = 0; 217aeb99916SMilanka Ringwald 218aeb99916SMilanka Ringwald while ((num_free_bytes > 0) && (connection->next_attr_id <= AVRCP_MEDIA_ATTR_SONG_LENGTH_MS)){ 219393f5724SMilanka Ringwald if ((connection->target_now_playing_info_attr_bitmap & (1 << (uint8_t)connection->next_attr_id)) == 0) { 22092ae15eaSMilanka Ringwald connection->next_attr_id = (avrcp_media_attribute_id_t) (((int) connection->next_attr_id) + 1); 221aeb99916SMilanka Ringwald continue; 222aeb99916SMilanka Ringwald } 223aeb99916SMilanka Ringwald 224ccfcd81eSMilanka Ringwald // prepare attribute value 225ccfcd81eSMilanka Ringwald uint16_t num_bytes_to_copy; 226ccfcd81eSMilanka Ringwald uint8_t * attr_value_with_offset = avrcp_get_next_value_fragment_for_attribute_id(connection, 227ccfcd81eSMilanka Ringwald connection->next_attr_id, 228ccfcd81eSMilanka Ringwald &num_bytes_to_copy); 229ccfcd81eSMilanka Ringwald 230aeb99916SMilanka Ringwald // store header 231aeb99916SMilanka Ringwald if (connection->attribute_value_offset == 0){ 232aeb99916SMilanka Ringwald // pack the whole attribute value header 233aeb99916SMilanka Ringwald if (connection->parser_attribute_header_pos == 0) { 234aeb99916SMilanka Ringwald avrcp_target_pack_single_element_header(connection->parser_attribute_header, connection->next_attr_id, 235aeb99916SMilanka Ringwald connection->attribute_value_len); 236ccfcd81eSMilanka Ringwald } 237aeb99916SMilanka Ringwald } 238aeb99916SMilanka Ringwald 239aeb99916SMilanka Ringwald if (connection->parser_attribute_header_pos < AVRCP_ATTRIBUTE_HEADER_LEN){ 240aeb99916SMilanka Ringwald uint16_t num_header_bytes_to_store = btstack_min(num_free_bytes, AVRCP_ATTRIBUTE_HEADER_LEN - connection->parser_attribute_header_pos); 241aeb99916SMilanka Ringwald memcpy(packet + bytes_stored, connection->parser_attribute_header + connection->parser_attribute_header_pos, num_header_bytes_to_store); 242aeb99916SMilanka Ringwald connection->parser_attribute_header_pos += num_header_bytes_to_store; 243aeb99916SMilanka Ringwald bytes_stored += num_header_bytes_to_store; 244aeb99916SMilanka Ringwald num_free_bytes -= num_header_bytes_to_store; 245aeb99916SMilanka Ringwald connection->data_offset += num_header_bytes_to_store; 246aeb99916SMilanka Ringwald 247aeb99916SMilanka Ringwald if (num_free_bytes == 0){ 248aeb99916SMilanka Ringwald continue; 249aeb99916SMilanka Ringwald } 250aeb99916SMilanka Ringwald } 251aeb99916SMilanka Ringwald 252aeb99916SMilanka Ringwald // store value 253aeb99916SMilanka Ringwald uint16_t num_attr_value_bytes_to_store = btstack_min(num_free_bytes, connection->attribute_value_len - connection->attribute_value_offset); 254aeb99916SMilanka Ringwald memcpy(packet + bytes_stored, attr_value_with_offset, num_attr_value_bytes_to_store); 255aeb99916SMilanka Ringwald bytes_stored += num_attr_value_bytes_to_store; 256aeb99916SMilanka Ringwald num_free_bytes -= num_attr_value_bytes_to_store; 257aeb99916SMilanka Ringwald connection->attribute_value_offset += num_attr_value_bytes_to_store; 258aeb99916SMilanka Ringwald connection->data_offset += num_attr_value_bytes_to_store; 259aeb99916SMilanka Ringwald 260aeb99916SMilanka Ringwald if (connection->attribute_value_offset == connection->attribute_value_len){ 261aeb99916SMilanka Ringwald // C++ compatible version of connection->next_attr_id++ 262aeb99916SMilanka Ringwald connection->next_attr_id = (avrcp_media_attribute_id_t) (((int) connection->next_attr_id) + 1); 263aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 264aeb99916SMilanka Ringwald connection->attribute_value_len = 0; 265aeb99916SMilanka Ringwald connection->parser_attribute_header_pos = 0; 266aeb99916SMilanka Ringwald } 267aeb99916SMilanka Ringwald } 268aeb99916SMilanka Ringwald return bytes_stored; 269aeb99916SMilanka Ringwald } 270aeb99916SMilanka Ringwald 271aeb99916SMilanka Ringwald static void avrcp_send_response_with_avctp_fragmentation(avrcp_connection_t * connection){ 2725e3d4d2bSMilanka Ringwald l2cap_reserve_packet_buffer(); 2735e3d4d2bSMilanka Ringwald uint8_t * packet = l2cap_get_outgoing_buffer(); 2745e3d4d2bSMilanka Ringwald 2755e3d4d2bSMilanka Ringwald // transport header 2765e3d4d2bSMilanka Ringwald // Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 277d1207cd8SMilanka Ringwald 278aeb99916SMilanka Ringwald uint16_t max_payload_size; 279aeb99916SMilanka Ringwald connection->avctp_packet_type = avctp_get_packet_type(connection, &max_payload_size); 280393f5724SMilanka Ringwald connection->avrcp_packet_type = avrcp_get_packet_type(connection); 281d1207cd8SMilanka Ringwald 282aeb99916SMilanka Ringwald // AVCTP header 283aeb99916SMilanka Ringwald // transport header : transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 284aeb99916SMilanka Ringwald uint16_t pos = 0; 285aeb99916SMilanka Ringwald packet[pos++] = (connection->transaction_id << 4) | (connection->avctp_packet_type << 2) | (AVRCP_RESPONSE_FRAME << 1) | 0; 286aeb99916SMilanka Ringwald 28718f5a148SMilanka Ringwald uint16_t param_len = connection->data_len; 288aeb99916SMilanka Ringwald 289aeb99916SMilanka Ringwald if (connection->avctp_packet_type == AVCTP_START_PACKET){ 29018f5a148SMilanka Ringwald uint16_t max_frame_size = btstack_min(connection->l2cap_mtu, AVRCP_MAX_AV_C_MESSAGE_FRAME_SIZE); 291aeb99916SMilanka Ringwald // first packet: max_payload_size 292aeb99916SMilanka Ringwald // rest packets 293aeb99916SMilanka Ringwald uint16_t num_payload_bytes = param_len - max_payload_size; 294aeb99916SMilanka Ringwald uint16_t frame_size_for_continue_packet = max_frame_size - avctp_get_num_bytes_for_header(AVCTP_CONTINUE_PACKET); 295aeb99916SMilanka Ringwald uint16_t num_avctp_packets = (num_payload_bytes + frame_size_for_continue_packet - 1)/frame_size_for_continue_packet + 1; 296*ab2445a0SMatthias Ringwald btstack_assert(num_avctp_packets <= 255); 297*ab2445a0SMatthias Ringwald packet[pos++] = (uint8_t) num_avctp_packets; 298aeb99916SMilanka Ringwald } 299aeb99916SMilanka Ringwald 300aeb99916SMilanka Ringwald uint16_t bytes_stored = 0; 301aeb99916SMilanka Ringwald uint8_t i; 302aeb99916SMilanka Ringwald 303aeb99916SMilanka Ringwald switch (connection->avctp_packet_type) { 304aeb99916SMilanka Ringwald case AVCTP_SINGLE_PACKET: 305aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 3065e3d4d2bSMilanka Ringwald // Profile IDentifier (PID) 3075e3d4d2bSMilanka Ringwald packet[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8; 3085e3d4d2bSMilanka Ringwald packet[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF; 309aeb99916SMilanka Ringwald 310aeb99916SMilanka Ringwald // AVRCP message 3115e3d4d2bSMilanka Ringwald // command_type 3125e3d4d2bSMilanka Ringwald packet[pos++] = connection->command_type; 3135e3d4d2bSMilanka Ringwald // subunit_type | subunit ID 3145e3d4d2bSMilanka Ringwald packet[pos++] = (connection->subunit_type << 3) | connection->subunit_id; 3155e3d4d2bSMilanka Ringwald // opcode 3165e3d4d2bSMilanka Ringwald packet[pos++] = (uint8_t) connection->command_opcode; 3170ec79bd0SMilanka Ringwald 318aeb99916SMilanka Ringwald switch (connection->command_opcode) { 319aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_VENDOR_DEPENDENT: 320aeb99916SMilanka Ringwald big_endian_store_24(packet, pos, connection->company_id); 321aeb99916SMilanka Ringwald pos += 3; 322aeb99916SMilanka Ringwald packet[pos++] = connection->pdu_id; 323aeb99916SMilanka Ringwald // AVRCP packet type 3240ec79bd0SMilanka Ringwald 325393f5724SMilanka Ringwald packet[pos++] = (uint8_t)connection->avrcp_packet_type; 326aeb99916SMilanka Ringwald // parameter length 327aeb99916SMilanka Ringwald big_endian_store_16(packet, pos, param_len); 328aeb99916SMilanka Ringwald pos += 2; 329aeb99916SMilanka Ringwald 330aeb99916SMilanka Ringwald switch (connection->pdu_id) { 331aeb99916SMilanka Ringwald // message is small enough to fit the single packet, no need for extra check 332aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_CAPABILITIES: 333ef07fa3cSMilanka Ringwald // capability ID 334aeb99916SMilanka Ringwald packet[pos++] = connection->data[0]; 335ef07fa3cSMilanka Ringwald // num_capabilities 336aeb99916SMilanka Ringwald packet[pos++] = connection->data[1]; 337aeb99916SMilanka Ringwald 338aeb99916SMilanka Ringwald switch ((avrcp_capability_id_t) connection->data[0]) { 339aeb99916SMilanka Ringwald case AVRCP_CAPABILITY_ID_EVENT: 340aeb99916SMilanka Ringwald for (i = (uint8_t) AVRCP_NOTIFICATION_EVENT_FIRST_INDEX; 341aeb99916SMilanka Ringwald i < (uint8_t) AVRCP_NOTIFICATION_EVENT_LAST_INDEX; i++) { 342393f5724SMilanka Ringwald if ((connection->notifications_supported_by_target & (1 << i)) == 0) { 343aeb99916SMilanka Ringwald continue; 344aeb99916SMilanka Ringwald } 345aeb99916SMilanka Ringwald packet[pos++] = i; 346aeb99916SMilanka Ringwald } 347aeb99916SMilanka Ringwald break; 348aeb99916SMilanka Ringwald case AVRCP_CAPABILITY_ID_COMPANY: 349aeb99916SMilanka Ringwald // use Bluetooth SIG as default company 350aeb99916SMilanka Ringwald for (i = 0; i < connection->data[1]; i++) { 351aeb99916SMilanka Ringwald little_endian_store_24(packet, pos, 352aeb99916SMilanka Ringwald connection->target_supported_companies[i]); 353aeb99916SMilanka Ringwald pos += 3; 354aeb99916SMilanka Ringwald } 355aeb99916SMilanka Ringwald break; 356aeb99916SMilanka Ringwald default: 357ef07fa3cSMilanka Ringwald // error response 358ef07fa3cSMilanka Ringwald break; 359aeb99916SMilanka Ringwald } 360aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos); 361aeb99916SMilanka Ringwald return; 362aeb99916SMilanka Ringwald 363aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES: 364f3bb6622SMilanka Ringwald packet[pos++] = count_set_bits_uint32(connection->target_now_playing_info_attr_bitmap); 365ccfcd81eSMilanka Ringwald max_payload_size--; 366ccfcd81eSMilanka Ringwald 367aeb99916SMilanka Ringwald bytes_stored = avrcp_store_avctp_now_playing_info_fragment(connection, max_payload_size, packet + pos); 368aeb99916SMilanka Ringwald 369aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += bytes_stored + pos; 370ccfcd81eSMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos + bytes_stored); 371aeb99916SMilanka Ringwald return; 372aeb99916SMilanka Ringwald 373aeb99916SMilanka Ringwald default: 374ef8ea8e4SMilanka Ringwald // error response and other OPCODEs 375aeb99916SMilanka Ringwald break; 376aeb99916SMilanka Ringwald } 377aeb99916SMilanka Ringwald break; 378aeb99916SMilanka Ringwald 379aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_PASS_THROUGH: 380aeb99916SMilanka Ringwald packet[pos++] = connection->operation_id; 381aeb99916SMilanka Ringwald // parameter length 382aeb99916SMilanka Ringwald packet[pos++] = (uint8_t) connection->data_len; 383aeb99916SMilanka Ringwald pos += 2; 384aeb99916SMilanka Ringwald break; 385aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_UNIT_INFO: 386aeb99916SMilanka Ringwald break; 387aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_SUBUNIT_INFO: 388aeb99916SMilanka Ringwald break; 389aeb99916SMilanka Ringwald default: 390aeb99916SMilanka Ringwald btstack_assert(false); 391aeb99916SMilanka Ringwald return; 392aeb99916SMilanka Ringwald } 393aeb99916SMilanka Ringwald break; 394aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 395aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 396aeb99916SMilanka Ringwald switch (connection->pdu_id) { 397aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES: 398aeb99916SMilanka Ringwald bytes_stored = avrcp_store_avctp_now_playing_info_fragment(connection, max_payload_size, packet + pos); 399aeb99916SMilanka Ringwald 400aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += bytes_stored + pos; 401ccfcd81eSMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos + bytes_stored); 402aeb99916SMilanka Ringwald return; 403aeb99916SMilanka Ringwald 404aeb99916SMilanka Ringwald default: 405aeb99916SMilanka Ringwald break; 406aeb99916SMilanka Ringwald } 407aeb99916SMilanka Ringwald break; 408aeb99916SMilanka Ringwald default: 409aeb99916SMilanka Ringwald btstack_assert(false); 410aeb99916SMilanka Ringwald return; 4114b338011SMilanka Ringwald } 4124b338011SMilanka Ringwald 413aeb99916SMilanka Ringwald // compare number of bytes to store with the remaining buffer size 414aeb99916SMilanka Ringwald uint16_t bytes_to_copy = btstack_min(connection->data_len - connection->data_offset, max_payload_size - pos); 415aeb99916SMilanka Ringwald 416aeb99916SMilanka Ringwald (void)memcpy(packet + pos, &connection->data[connection->data_offset], bytes_to_copy); 417aeb99916SMilanka Ringwald pos += bytes_to_copy; 418aeb99916SMilanka Ringwald connection->data_offset += bytes_to_copy; 419aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += pos; 420aeb99916SMilanka Ringwald 421aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos); 422aeb99916SMilanka Ringwald } 423aeb99916SMilanka Ringwald 4243b899686SMilanka Ringwald static void avctp_send_reject_cmd_wrong_pid(avrcp_connection_t * connection){ 425aeb99916SMilanka Ringwald l2cap_reserve_packet_buffer(); 426aeb99916SMilanka Ringwald uint8_t * packet = l2cap_get_outgoing_buffer(); 427aeb99916SMilanka Ringwald 428aeb99916SMilanka Ringwald // AVCTP header 429aeb99916SMilanka Ringwald // transport header : transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 4303b899686SMilanka Ringwald packet[0] = (connection->transaction_id << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_RESPONSE_FRAME << 1) | 1; 431393f5724SMilanka Ringwald big_endian_store_16(packet, 1, connection->message_body[0]); 432aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, 3); 433aeb99916SMilanka Ringwald } 434aeb99916SMilanka Ringwald 435c3c3461eSMilanka Ringwald static void avrcp_target_custom_command_data_init(avrcp_connection_t * connection, 436aeb99916SMilanka Ringwald avrcp_command_opcode_t opcode, avrcp_command_type_t command_type, 437aeb99916SMilanka Ringwald avrcp_subunit_type_t subunit_type, avrcp_subunit_id_t subunit_id, 438aeb99916SMilanka Ringwald avrcp_pdu_id_t pdu_id, uint32_t company_id){ 439aeb99916SMilanka Ringwald 440aeb99916SMilanka Ringwald connection->command_opcode = opcode; 4416649facbSMatthias Ringwald connection->command_type = command_type; 44212b7c8ecSMilanka Ringwald connection->subunit_type = subunit_type; 44312b7c8ecSMilanka Ringwald connection->subunit_id = subunit_id; 444aeb99916SMilanka Ringwald connection->company_id = company_id << 16; 445aeb99916SMilanka Ringwald connection->pdu_id = pdu_id; 446aeb99916SMilanka Ringwald connection->data = NULL; 447aeb99916SMilanka Ringwald connection->data_offset = 0; 448aeb99916SMilanka Ringwald connection->data_len = 0; 4491e852c68SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 4506649facbSMatthias Ringwald } 4516649facbSMatthias Ringwald 452aeb99916SMilanka 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){ 453aeb99916SMilanka Ringwald connection->command_opcode = AVRCP_CMD_OPCODE_VENDOR_DEPENDENT; 454aeb99916SMilanka Ringwald connection->subunit_type = AVRCP_SUBUNIT_TYPE_PANEL; 455aeb99916SMilanka Ringwald connection->subunit_id = AVRCP_SUBUNIT_ID; 456aeb99916SMilanka Ringwald connection->company_id = BT_SIG_COMPANY_ID; 457aeb99916SMilanka Ringwald 458aeb99916SMilanka Ringwald connection->command_type = command_type; 459aeb99916SMilanka Ringwald connection->pdu_id = pdu_id; 460393f5724SMilanka Ringwald connection->data = connection->message_body; 461aeb99916SMilanka Ringwald connection->data_offset = 0; 4621e852c68SMilanka Ringwald connection->data_len = 0; 463aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 464aeb99916SMilanka Ringwald } 465aeb99916SMilanka Ringwald 466aeb99916SMilanka 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){ 467aeb99916SMilanka Ringwald connection->command_opcode = AVRCP_CMD_OPCODE_PASS_THROUGH; 468aeb99916SMilanka Ringwald connection->subunit_type = AVRCP_SUBUNIT_TYPE_PANEL; 469aeb99916SMilanka Ringwald connection->subunit_id = AVRCP_SUBUNIT_ID; 470aeb99916SMilanka Ringwald 471aeb99916SMilanka Ringwald connection->command_type = command_type; 472aeb99916SMilanka Ringwald connection->company_id = 0; 473ccfcd81eSMilanka Ringwald connection->pdu_id = AVRCP_PDU_ID_UNDEFINED; 474aeb99916SMilanka Ringwald connection->operation_id = opid; 475aeb99916SMilanka Ringwald 476393f5724SMilanka Ringwald connection->data = connection->message_body; 477aeb99916SMilanka Ringwald connection->data_offset = 0; 478aeb99916SMilanka Ringwald connection->data_len = 0; 4791e852c68SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 480aeb99916SMilanka Ringwald } 481aeb99916SMilanka Ringwald 482aeb99916SMilanka Ringwald 483aeb99916SMilanka Ringwald static uint8_t avrcp_target_vendor_dependent_response_accept(avrcp_connection_t * connection, avrcp_pdu_id_t pdu_id, uint8_t status){ 484aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, pdu_id); 485aeb99916SMilanka Ringwald connection->data_len = 1; 486aeb99916SMilanka Ringwald connection->data[0] = status; 487aeb99916SMilanka Ringwald 488393f5724SMilanka Ringwald connection->target_accept_response = true; 489ef8ea8e4SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 49012b7c8ecSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 49112b7c8ecSMilanka Ringwald return ERROR_CODE_SUCCESS; 49212b7c8ecSMilanka Ringwald } 49312b7c8ecSMilanka Ringwald 494aeb99916SMilanka 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){ 495aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_REJECTED, pdu_id); 496aeb99916SMilanka Ringwald connection->data_len = 1; 497aeb99916SMilanka Ringwald connection->data[0] = status; 498aeb99916SMilanka Ringwald 499d1207cd8SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 500d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 501d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 502d1207cd8SMilanka Ringwald } 503d1207cd8SMilanka Ringwald 504aeb99916SMilanka 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){ 505aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_NOT_IMPLEMENTED, pdu_id); 506aeb99916SMilanka Ringwald connection->data_len = 1; 507aeb99916SMilanka Ringwald connection->data[0] = event_id; 508d1386928SMilanka Ringwald 509d1386928SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 510d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 511d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 512e0bbf3edSMilanka Ringwald } 513e0bbf3edSMilanka Ringwald 514d8133c09SMatthias 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){ 515aeb99916SMilanka Ringwald btstack_assert(value_len + 1 < AVRCP_MAX_COMMAND_PARAMETER_LENGTH); 516aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_INTERIM, pdu_id); 517aeb99916SMilanka Ringwald connection->data_len = 1 + value_len; 518aeb99916SMilanka Ringwald connection->data[0] = event_id; 519aeb99916SMilanka Ringwald 520c1ab6cc1SMatthias Ringwald if (value && (value_len > 0)){ 521aeb99916SMilanka Ringwald (void)memcpy(connection->data + 1, value, value_len); 522d1207cd8SMilanka Ringwald } 523aeb99916SMilanka Ringwald 524d1207cd8SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 525d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 526d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 527d1207cd8SMilanka Ringwald } 528d1207cd8SMilanka Ringwald 529aeb99916SMilanka 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){ 530aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_INTERIM, pdu_id); 5310a2feef4SMilanka Ringwald 532aeb99916SMilanka Ringwald connection->data_len = 5; 533aeb99916SMilanka Ringwald connection->data[0] = event_id; 534393f5724SMilanka Ringwald big_endian_store_16(connection->data, 1, connection->target_addressed_player_id); 535393f5724SMilanka Ringwald big_endian_store_16(connection->data, 3, connection->target_uid_counter); 5366568eb54SMilanka Ringwald 5376568eb54SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 5386568eb54SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 5396568eb54SMilanka Ringwald return ERROR_CODE_SUCCESS; 5406568eb54SMilanka Ringwald } 5416568eb54SMilanka Ringwald 542aeb99916SMilanka 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){ 5431945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 544831d3fd5SMilanka Ringwald if (!connection){ 545831d3fd5SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 546831d3fd5SMilanka Ringwald } 547aeb99916SMilanka Ringwald avrcp_target_pass_through_command_data_init(connection, ctype, opid); 548831d3fd5SMilanka Ringwald 549831d3fd5SMilanka Ringwald if (operands_length == 1){ 550aeb99916SMilanka Ringwald connection->data_len = 1; 551393f5724SMilanka Ringwald connection->message_body[0] = operand; 552831d3fd5SMilanka Ringwald } 553831d3fd5SMilanka Ringwald 554831d3fd5SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 555831d3fd5SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 556831d3fd5SMilanka Ringwald return ERROR_CODE_SUCCESS; 557831d3fd5SMilanka Ringwald } 558831d3fd5SMilanka Ringwald 559831d3fd5SMilanka Ringwald uint8_t avrcp_target_operation_rejected(uint16_t avrcp_cid, avrcp_operation_id_t opid, uint8_t operands_length, uint8_t operand){ 560831d3fd5SMilanka Ringwald return avrcp_target_pass_through_response(avrcp_cid, AVRCP_CTYPE_RESPONSE_REJECTED, opid, operands_length, operand); 561831d3fd5SMilanka Ringwald } 562831d3fd5SMilanka Ringwald 563831d3fd5SMilanka Ringwald uint8_t avrcp_target_operation_accepted(uint16_t avrcp_cid, avrcp_operation_id_t opid, uint8_t operands_length, uint8_t operand){ 564831d3fd5SMilanka Ringwald return avrcp_target_pass_through_response(avrcp_cid, AVRCP_CTYPE_RESPONSE_ACCEPTED, opid, operands_length, operand); 565831d3fd5SMilanka Ringwald } 566831d3fd5SMilanka Ringwald 567831d3fd5SMilanka Ringwald uint8_t avrcp_target_operation_not_implemented(uint16_t avrcp_cid, avrcp_operation_id_t opid, uint8_t operands_length, uint8_t operand){ 568831d3fd5SMilanka Ringwald return avrcp_target_pass_through_response(avrcp_cid, AVRCP_CTYPE_RESPONSE_ACCEPTED, opid, operands_length, operand); 569831d3fd5SMilanka Ringwald } 570831d3fd5SMilanka Ringwald 57161a3437bSMilanka Ringwald uint8_t avrcp_target_set_unit_info(uint16_t avrcp_cid, avrcp_subunit_type_t unit_type, uint32_t company_id){ 5721945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 5734b338011SMilanka Ringwald if (!connection){ 57461a3437bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 5754b338011SMilanka Ringwald } 576393f5724SMilanka Ringwald connection->target_unit_type = unit_type; 577d1207cd8SMilanka Ringwald connection->company_id = company_id; 57861a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 579d1207cd8SMilanka Ringwald } 580d1207cd8SMilanka Ringwald 58161a3437bSMilanka 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){ 5821945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 583d1207cd8SMilanka Ringwald if (!connection){ 58461a3437bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 585d1207cd8SMilanka Ringwald } 586393f5724SMilanka Ringwald connection->target_subunit_info_type = subunit_type; 587393f5724SMilanka Ringwald connection->target_subunit_info_data = subunit_info_data; 588393f5724SMilanka Ringwald connection->target_subunit_info_data_size = subunit_info_data_size; 58961a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 590d1207cd8SMilanka Ringwald } 591d1207cd8SMilanka Ringwald 592d1207cd8SMilanka Ringwald static uint8_t avrcp_target_unit_info(avrcp_connection_t * connection){ 593aeb99916SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED){ 594aeb99916SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 595aeb99916SMilanka Ringwald } 596aeb99916SMilanka Ringwald 5977bccc69fSMilanka Ringwald avrcp_target_custom_command_data_init(connection, 598aeb99916SMilanka Ringwald AVRCP_CMD_OPCODE_UNIT_INFO, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, 5997bccc69fSMilanka Ringwald AVRCP_SUBUNIT_TYPE_UNIT, AVRCP_SUBUNIT_ID_IGNORE, AVRCP_PDU_ID_UNDEFINED, 6007bccc69fSMilanka Ringwald connection->company_id); 6014b338011SMilanka Ringwald 6024b338011SMilanka Ringwald uint8_t unit = 0; 603393f5724SMilanka Ringwald connection->data = connection->message_body; 604aeb99916SMilanka Ringwald connection->data_len = 5; 605aeb99916SMilanka Ringwald connection->data[0] = 0x07; 606393f5724SMilanka Ringwald connection->data[1] = (connection->target_unit_type << 4) | unit; 6074b338011SMilanka Ringwald // company id is 3 bytes long 608aeb99916SMilanka Ringwald big_endian_store_24(connection->data, 2, connection->company_id); 609d1386928SMilanka Ringwald 610d3db1226SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 611d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 612d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 6134b338011SMilanka Ringwald } 6144b338011SMilanka Ringwald 615d1207cd8SMilanka Ringwald static uint8_t avrcp_target_subunit_info(avrcp_connection_t * connection, uint8_t offset){ 6164b338011SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED) return ERROR_CODE_COMMAND_DISALLOWED; 617fda344fcSMatthias Ringwald if (offset >= 32) return AVRCP_STATUS_INVALID_PARAMETER; 6184b338011SMilanka Ringwald 6197bccc69fSMilanka Ringwald avrcp_target_custom_command_data_init(connection, AVRCP_CMD_OPCODE_SUBUNIT_INFO, 6207bccc69fSMilanka Ringwald AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, 6217bccc69fSMilanka Ringwald AVRCP_SUBUNIT_TYPE_UNIT, AVRCP_SUBUNIT_ID_IGNORE, AVRCP_PDU_ID_UNDEFINED, 6227bccc69fSMilanka Ringwald connection->company_id); 6234b338011SMilanka Ringwald 6244b338011SMilanka Ringwald uint8_t page = offset / 4; 6254b338011SMilanka Ringwald uint8_t extension_code = 7; 626393f5724SMilanka Ringwald connection->data = connection->message_body; 627aeb99916SMilanka Ringwald connection->data_len = 5; 628aeb99916SMilanka Ringwald connection->data[0] = (page << 4) | extension_code; 629d1207cd8SMilanka Ringwald 630f0af2234SMatthias Ringwald // mark non-existent entries with 0xff 631393f5724SMilanka Ringwald memset(&connection->message_body[1], 0xFF, 4); 632393f5724SMilanka Ringwald if ((connection->data != NULL) && (offset < connection->target_subunit_info_data_size)){ 633393f5724SMilanka Ringwald uint8_t bytes_to_copy = btstack_min(connection->target_subunit_info_data_size - offset, 4); 634393f5724SMilanka Ringwald memcpy(&connection->data[1], &connection->target_subunit_info_data[offset], bytes_to_copy); 635fda344fcSMatthias Ringwald } 636d1386928SMilanka Ringwald 637d3db1226SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 638d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 639d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 6404b338011SMilanka Ringwald } 6414b338011SMilanka Ringwald 642aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_supported_events(avrcp_connection_t * connection){ 643aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_CAPABILITIES); 64493259c07SMilanka Ringwald 64589042326SMilanka Ringwald uint8_t event_id; 646aeb99916SMilanka Ringwald uint8_t num_events = 0; 64789042326SMilanka Ringwald for (event_id = (uint8_t) AVRCP_NOTIFICATION_EVENT_FIRST_INDEX; event_id < (uint8_t) AVRCP_NOTIFICATION_EVENT_LAST_INDEX; event_id++){ 648393f5724SMilanka Ringwald if ((connection->notifications_supported_by_target & (1 << event_id)) == 0){ 64989042326SMilanka Ringwald continue; 65089042326SMilanka Ringwald } 65189042326SMilanka Ringwald num_events++; 65289042326SMilanka Ringwald } 65389042326SMilanka Ringwald 654aeb99916SMilanka Ringwald connection->data[0] = AVRCP_CAPABILITY_ID_EVENT; 655aeb99916SMilanka Ringwald connection->data[1] = num_events; 656aeb99916SMilanka Ringwald connection->data_len = 2 + num_events; 657e0bbf3edSMilanka Ringwald 658aeb99916SMilanka Ringwald // fill the data later directly to the L2CAP outgoing buffer 659ae5447c3SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 660d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 661d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 662e0bbf3edSMilanka Ringwald } 663e0bbf3edSMilanka Ringwald 664aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_supported_companies(avrcp_connection_t * connection){ 665aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_CAPABILITIES); 666aeb99916SMilanka Ringwald 667aeb99916SMilanka Ringwald connection->data[0] = AVRCP_CAPABILITY_ID_COMPANY; 668ef07fa3cSMilanka Ringwald if (connection->target_supported_companies_num == 0){ 669ef07fa3cSMilanka Ringwald connection->target_supported_companies_num = 1; 670ef07fa3cSMilanka Ringwald connection->target_supported_companies = default_companies; 671ef07fa3cSMilanka Ringwald } 672ef07fa3cSMilanka Ringwald 673aeb99916SMilanka Ringwald connection->data[1] = connection->target_supported_companies_num; 674ef07fa3cSMilanka Ringwald connection->data_len = 2 + connection->data[1] * 3; 675aeb99916SMilanka Ringwald 676aeb99916SMilanka Ringwald // fill the data later directly to the L2CAP outgoing buffer and 677aeb99916SMilanka Ringwald // use Bluetooth SIG as default company 678ae5447c3SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 67989042326SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 68089042326SMilanka Ringwald return ERROR_CODE_SUCCESS; 68189042326SMilanka Ringwald } 68289042326SMilanka Ringwald 6830346f11dSMilanka Ringwald uint8_t avrcp_target_support_event(uint16_t avrcp_cid, avrcp_notification_event_id_t event_id){ 6840346f11dSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 6850346f11dSMilanka Ringwald if (!connection){ 6860346f11dSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 68789042326SMilanka Ringwald } 68889042326SMilanka Ringwald 6890346f11dSMilanka Ringwald if ((event_id < (uint8_t)AVRCP_NOTIFICATION_EVENT_FIRST_INDEX) || (event_id > (uint8_t)AVRCP_NOTIFICATION_EVENT_LAST_INDEX)){ 6900346f11dSMilanka Ringwald return ERROR_CODE_UNSUPPORTED_FEATURE_OR_PARAMETER_VALUE; 6910346f11dSMilanka Ringwald } 6920346f11dSMilanka Ringwald 693393f5724SMilanka Ringwald connection->notifications_supported_by_target |= (1 << (uint8_t)event_id); 6940346f11dSMilanka Ringwald return ERROR_CODE_SUCCESS; 6950346f11dSMilanka Ringwald } 6960346f11dSMilanka Ringwald 6970346f11dSMilanka Ringwald uint8_t avrcp_target_support_companies(uint16_t avrcp_cid, uint8_t num_companies, const uint32_t *companies){ 6980346f11dSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 6990346f11dSMilanka Ringwald if (!connection){ 7000346f11dSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7010346f11dSMilanka Ringwald } 7020346f11dSMilanka Ringwald 7030346f11dSMilanka Ringwald connection->target_supported_companies_num = num_companies; 7040346f11dSMilanka Ringwald connection->target_supported_companies = companies; 7050346f11dSMilanka Ringwald return ERROR_CODE_SUCCESS; 7065e3d4d2bSMilanka Ringwald } 7075e3d4d2bSMilanka Ringwald 708aeb99916SMilanka Ringwald // TODO Review (use flags) 709d1207cd8SMilanka 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){ 710aeb99916SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 711aeb99916SMilanka Ringwald if (!connection){ 712aeb99916SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 713aeb99916SMilanka Ringwald } 714aeb99916SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED){ 715aeb99916SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 7165e3d4d2bSMilanka Ringwald } 7175e3d4d2bSMilanka Ringwald 718aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_PLAY_STATUS); 719aeb99916SMilanka Ringwald connection->data_len = 9; 720aeb99916SMilanka Ringwald big_endian_store_32(connection->data, 0, song_length_ms); 721aeb99916SMilanka Ringwald big_endian_store_32(connection->data, 4, song_position_ms); 722aeb99916SMilanka Ringwald connection->data[8] = play_status; 7235e3d4d2bSMilanka Ringwald 7245e3d4d2bSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 7255e3d4d2bSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7265e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7275e3d4d2bSMilanka Ringwald } 7285e3d4d2bSMilanka Ringwald 729d1207cd8SMilanka Ringwald static uint8_t avrcp_target_store_media_attr(avrcp_connection_t * connection, avrcp_media_attribute_id_t attr_id, const char * value){ 7305e3d4d2bSMilanka Ringwald int index = attr_id - 1; 731d1207cd8SMilanka Ringwald if (!value) return AVRCP_STATUS_INVALID_PARAMETER; 732*ab2445a0SMatthias Ringwald uint16_t value_len = (uint16_t)strlen(value); 733*ab2445a0SMatthias Ringwald btstack_assert(value_len <= 255); 734393f5724SMilanka Ringwald connection->target_now_playing_info[index].value = (uint8_t*)value; 735*ab2445a0SMatthias Ringwald connection->target_now_playing_info[index].len = value_len; 7365e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7375e3d4d2bSMilanka Ringwald } 7385e3d4d2bSMilanka Ringwald 739d1207cd8SMilanka Ringwald uint8_t avrcp_target_set_playback_status(uint16_t avrcp_cid, avrcp_playback_status_t playback_status){ 7401945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7415e3d4d2bSMilanka Ringwald if (!connection){ 7425e3d4d2bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7435e3d4d2bSMilanka Ringwald } 744393f5724SMilanka Ringwald if (connection->target_playback_status == playback_status){ 74561a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 74661a3437bSMilanka Ringwald } 747d1207cd8SMilanka Ringwald 748393f5724SMilanka Ringwald connection->target_playback_status = playback_status; 7494b5213b0SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED)) { 750393f5724SMilanka Ringwald connection->target_playback_status_changed = true; 751d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7524b5213b0SMilanka Ringwald } 7535e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7545e3d4d2bSMilanka Ringwald } 7555e3d4d2bSMilanka Ringwald 756276c795aSMilanka Ringwald static void avrcp_target_register_track_changed(avrcp_connection_t * connection, const uint8_t * track_id){ 7577d1b72e5SMilanka Ringwald if (track_id == NULL){ 7587d1b72e5SMilanka Ringwald memset(connection->target_track_id, 0xFF, 8); 7597d1b72e5SMilanka Ringwald connection->target_track_selected = false; 7607d1b72e5SMilanka Ringwald } else { 7617d1b72e5SMilanka Ringwald (void)memcpy(connection->target_track_id, track_id, 8); 7627d1b72e5SMilanka Ringwald connection->target_track_selected = true; 7637d1b72e5SMilanka Ringwald } 7647d1b72e5SMilanka Ringwald 7657d1b72e5SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED)) { 7667d1b72e5SMilanka Ringwald connection->target_track_changed = true; 7677d1b72e5SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7687d1b72e5SMilanka Ringwald } 7697d1b72e5SMilanka Ringwald } 7707d1b72e5SMilanka Ringwald 7717d1b72e5SMilanka Ringwald uint8_t avrcp_target_track_changed(uint16_t avrcp_cid, uint8_t * track_id){ 7727d1b72e5SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7737d1b72e5SMilanka Ringwald if (!connection){ 7747d1b72e5SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7757d1b72e5SMilanka Ringwald } 7767d1b72e5SMilanka Ringwald avrcp_target_register_track_changed(connection, track_id); 7777d1b72e5SMilanka Ringwald return ERROR_CODE_SUCCESS; 7787d1b72e5SMilanka Ringwald } 7797d1b72e5SMilanka Ringwald 78061a3437bSMilanka Ringwald uint8_t avrcp_target_set_now_playing_info(uint16_t avrcp_cid, const avrcp_track_t * current_track, uint16_t total_tracks){ 7811945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7825e3d4d2bSMilanka Ringwald if (!connection){ 78361a3437bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7845e3d4d2bSMilanka Ringwald } 785d1207cd8SMilanka Ringwald if (!current_track){ 78661a3437bSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 787d1207cd8SMilanka Ringwald } 78861a3437bSMilanka Ringwald 789393f5724SMilanka Ringwald (void)memcpy(connection->target_track_id, current_track->track_id, 8); 790393f5724SMilanka Ringwald connection->target_song_length_ms = current_track->song_length_ms; 791393f5724SMilanka Ringwald connection->target_track_nr = current_track->track_nr; 792393f5724SMilanka Ringwald connection->target_total_tracks = total_tracks; 793d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_TITLE, current_track->title); 794d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_ARTIST, current_track->artist); 795d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_ALBUM, current_track->album); 796d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_GENRE, current_track->genre); 797aeb99916SMilanka Ringwald 7987d1b72e5SMilanka Ringwald avrcp_target_register_track_changed(connection, current_track->track_id); 79961a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 8005e3d4d2bSMilanka Ringwald } 8015e3d4d2bSMilanka Ringwald 80212b7c8ecSMilanka Ringwald 803d1207cd8SMilanka Ringwald uint8_t avrcp_target_playing_content_changed(uint16_t avrcp_cid){ 8041945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8055e3d4d2bSMilanka Ringwald if (!connection){ 8065e3d4d2bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8075e3d4d2bSMilanka Ringwald } 80812b7c8ecSMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED)) { 809393f5724SMilanka Ringwald connection->target_playing_content_changed = true; 810d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 81112b7c8ecSMilanka Ringwald } 8125e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 8135e3d4d2bSMilanka Ringwald } 814e0bbf3edSMilanka Ringwald 8156568eb54SMilanka Ringwald uint8_t avrcp_target_addressed_player_changed(uint16_t avrcp_cid, uint16_t player_id, uint16_t uid_counter){ 8161945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8176568eb54SMilanka Ringwald if (!connection){ 8186568eb54SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8196568eb54SMilanka Ringwald } 82061a3437bSMilanka Ringwald 821393f5724SMilanka Ringwald if (connection->target_addressed_player_id == player_id){ 82261a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 82361a3437bSMilanka Ringwald } 824db818650SMilanka Ringwald 825393f5724SMilanka Ringwald connection->target_uid_counter = uid_counter; 826393f5724SMilanka Ringwald connection->target_addressed_player_id = player_id; 827db818650SMilanka Ringwald 828db818650SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED)) { 829393f5724SMilanka Ringwald connection->target_addressed_player_changed = true; 830db818650SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 8316568eb54SMilanka Ringwald } 8326568eb54SMilanka Ringwald return ERROR_CODE_SUCCESS; 8336568eb54SMilanka Ringwald } 8346568eb54SMilanka Ringwald 835d1207cd8SMilanka Ringwald uint8_t avrcp_target_battery_status_changed(uint16_t avrcp_cid, avrcp_battery_status_t battery_status){ 8361945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 837d1207cd8SMilanka Ringwald if (!connection){ 838d1207cd8SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 839d1207cd8SMilanka Ringwald } 840393f5724SMilanka Ringwald if (connection->target_battery_status == battery_status){ 84161a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 84261a3437bSMilanka Ringwald } 843db818650SMilanka Ringwald 844393f5724SMilanka Ringwald connection->target_battery_status = battery_status; 845db818650SMilanka Ringwald 846db818650SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED)) { 847393f5724SMilanka Ringwald connection->target_battery_status_changed = true; 848d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 84912b7c8ecSMilanka Ringwald } 850d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 851d1207cd8SMilanka Ringwald } 852d1207cd8SMilanka Ringwald 853b772a0d7SMilanka Ringwald uint8_t avrcp_target_adjust_absolute_volume(uint16_t avrcp_cid, uint8_t absolute_volume){ 854b772a0d7SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 855b772a0d7SMilanka Ringwald if (!connection){ 856b772a0d7SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 857b772a0d7SMilanka Ringwald } 858b772a0d7SMilanka Ringwald 859393f5724SMilanka Ringwald connection->target_absolute_volume = absolute_volume; 860b772a0d7SMilanka Ringwald return ERROR_CODE_SUCCESS; 861b772a0d7SMilanka Ringwald } 862b772a0d7SMilanka Ringwald 863984bbc3eSMilanka Ringwald uint8_t avrcp_target_volume_changed(uint16_t avrcp_cid, uint8_t absolute_volume){ 8641945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 865d1207cd8SMilanka Ringwald if (!connection){ 866d1207cd8SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 867d1207cd8SMilanka Ringwald } 868393f5724SMilanka Ringwald if (connection->target_absolute_volume == absolute_volume){ 86961a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 87061a3437bSMilanka Ringwald } 87161a3437bSMilanka Ringwald 872393f5724SMilanka Ringwald connection->target_absolute_volume = absolute_volume; 87361a3437bSMilanka Ringwald 8743e5c3c5bSMatthias Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED )) { 875393f5724SMilanka Ringwald connection->target_notify_absolute_volume_changed = true; 876d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 87712b7c8ecSMilanka Ringwald } 878d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 879d1207cd8SMilanka Ringwald } 880d1207cd8SMilanka Ringwald 8810ec79bd0SMilanka Ringwald static void avrcp_target_set_transaction_label_for_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification, uint8_t transaction_label){ 882fe934656SMatthias Ringwald if (notification > AVRCP_NOTIFICATION_EVENT_MAX_VALUE) return; 883393f5724SMilanka Ringwald connection->target_notifications_transaction_label[notification] = transaction_label; 8840ec79bd0SMilanka Ringwald } 8850ec79bd0SMilanka Ringwald 8860ec79bd0SMilanka Ringwald static uint8_t avrcp_target_get_transaction_label_for_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification){ 887fe934656SMatthias Ringwald if (notification > AVRCP_NOTIFICATION_EVENT_MAX_VALUE) return 0; 888393f5724SMilanka Ringwald return connection->target_notifications_transaction_label[notification]; 8890ec79bd0SMilanka Ringwald } 890d1207cd8SMilanka Ringwald 8910b578d06SMilanka Ringwald static bool avcrp_operation_id_is_valid(avrcp_operation_id_t operation_id){ 8920b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_1) return true; 8930b578d06SMilanka Ringwald 8940b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_0) return false; 8950b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_2) return true; 8960b578d06SMilanka Ringwald 8970b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_CHANNEL_UP) return false; 8980b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_3) return true; 8990b578d06SMilanka Ringwald 9000b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_CHANNEL_UP) return false; 9010b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_3) return true; 9020b578d06SMilanka Ringwald 9030b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_SKIP) return false; 9040b578d06SMilanka Ringwald if (operation_id == AVRCP_OPERATION_ID_SKIP) return true; 9050b578d06SMilanka Ringwald 9060b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_POWER) return false; 9070b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_4) return true; 9080b578d06SMilanka Ringwald 9090b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_ANGLE) return false; 9100b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_5) return true; 9110b578d06SMilanka Ringwald 9120b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_F1) return false; 9130b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_6) return true; 9140b578d06SMilanka Ringwald 9150b578d06SMilanka Ringwald return false; 9160b578d06SMilanka Ringwald } 9170b578d06SMilanka Ringwald 918aeb99916SMilanka Ringwald 919aeb99916SMilanka Ringwald #ifdef ENABLE_AVCTP_FRAGMENTATION 920aeb99916SMilanka Ringwald static void avctp_reassemble_message(avrcp_connection_t * connection, avctp_packet_type_t packet_type, uint8_t *packet, uint16_t size){ 921aeb99916SMilanka Ringwald // after header (transaction label and packet type) 922aeb99916SMilanka Ringwald uint16_t pos; 923aeb99916SMilanka Ringwald uint16_t bytes_to_store; 924aeb99916SMilanka Ringwald 925aeb99916SMilanka Ringwald switch (packet_type){ 926aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 927aeb99916SMilanka Ringwald if (size < 2) return; 928aeb99916SMilanka Ringwald 929aeb99916SMilanka Ringwald // store header 930aeb99916SMilanka Ringwald pos = 0; 931aeb99916SMilanka Ringwald connection->avctp_reassembly_buffer[pos] = packet[pos]; 932aeb99916SMilanka Ringwald pos++; 933aeb99916SMilanka Ringwald connection->avctp_reassembly_size = pos; 934aeb99916SMilanka Ringwald 935aeb99916SMilanka Ringwald // NOTE: num packets not needed for reassembly, ignoring it does not pose security risk -> no need to store it 936aeb99916SMilanka Ringwald pos++; 937aeb99916SMilanka Ringwald 938aeb99916SMilanka 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 939aeb99916SMilanka Ringwald 940aeb99916SMilanka Ringwald bytes_to_store = btstack_min(size - pos, sizeof(connection->avctp_reassembly_buffer) - connection->avctp_reassembly_size); 941aeb99916SMilanka Ringwald memcpy(&connection->avctp_reassembly_buffer[connection->avctp_reassembly_size], &packet[pos], bytes_to_store); 942aeb99916SMilanka Ringwald connection->avctp_reassembly_size += bytes_to_store; 943aeb99916SMilanka Ringwald break; 944aeb99916SMilanka Ringwald 945aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 946aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 947aeb99916SMilanka Ringwald if (size < 1) return; 948aeb99916SMilanka Ringwald 949aeb99916SMilanka Ringwald // store remaining data, ignore header 950aeb99916SMilanka Ringwald pos = 1; 951aeb99916SMilanka Ringwald bytes_to_store = btstack_min(size - pos, sizeof(connection->avctp_reassembly_buffer) - connection->avctp_reassembly_size); 952aeb99916SMilanka Ringwald memcpy(&connection->avctp_reassembly_buffer[connection->avctp_reassembly_size], &packet[pos], bytes_to_store); 953aeb99916SMilanka Ringwald connection->avctp_reassembly_size += bytes_to_store; 954aeb99916SMilanka Ringwald break; 955aeb99916SMilanka Ringwald 956aeb99916SMilanka Ringwald default: 957aeb99916SMilanka Ringwald return; 958aeb99916SMilanka Ringwald } 959aeb99916SMilanka Ringwald } 960aeb99916SMilanka Ringwald #endif 961aeb99916SMilanka Ringwald 96201dc6e35SMilanka Ringwald static void avrcp_handle_l2cap_data_packet_for_signaling_connection(avrcp_connection_t * connection, uint8_t * packet, uint16_t size){ 963aeb99916SMilanka Ringwald uint8_t avctp_header = packet[0]; 964aeb99916SMilanka Ringwald connection->transaction_id = avctp_header >> 4; 965aeb99916SMilanka Ringwald 966aeb99916SMilanka Ringwald avctp_packet_type_t avctp_packet_type = (avctp_packet_type_t) ((avctp_header & 0x0F) >> 2); 967aeb99916SMilanka Ringwald switch (avctp_packet_type){ 968aeb99916SMilanka Ringwald case AVCTP_SINGLE_PACKET: 969aeb99916SMilanka Ringwald break; 970aeb99916SMilanka Ringwald 971aeb99916SMilanka Ringwald #ifdef ENABLE_AVCTP_FRAGMENTATION 972aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 973aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 974aeb99916SMilanka Ringwald avctp_reassemble_message(connection, avctp_packet_type, packet, size); 975aeb99916SMilanka Ringwald return; 976aeb99916SMilanka Ringwald 977aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 978aeb99916SMilanka Ringwald avctp_reassemble_message(connection, avctp_packet_type, packet, size); 979aeb99916SMilanka Ringwald 980aeb99916SMilanka Ringwald packet = connection->avctp_reassembly_buffer; 981aeb99916SMilanka Ringwald size = connection->avctp_reassembly_size; 982aeb99916SMilanka Ringwald break; 983aeb99916SMilanka Ringwald #endif 984aeb99916SMilanka Ringwald default: 985aeb99916SMilanka Ringwald return; 986aeb99916SMilanka Ringwald } 987697b823eSMatthias Ringwald 988697b823eSMatthias Ringwald if (size < 6u) return; 9894b338011SMilanka Ringwald 990aeb99916SMilanka Ringwald uint16_t pid = big_endian_read_16(packet, 1); 991be4cc80aSMilanka Ringwald 99267049435SMilanka Ringwald if (pid != BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL){ 9933b899686SMilanka Ringwald log_info("Invalid pid 0x%02x, expected 0x%02x", pid, BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL); 994393f5724SMilanka Ringwald connection->target_reject_transport_header = true; 995393f5724SMilanka Ringwald connection->target_invalid_pid = pid; 9964e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 99767049435SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 99867049435SMilanka Ringwald return; 99967049435SMilanka Ringwald } 100067049435SMilanka Ringwald 1001aeb99916SMilanka Ringwald // avrcp_subunit_type_t subunit_type = (avrcp_subunit_type_t) (packet[4] >> 3); 1002aeb99916SMilanka Ringwald // avrcp_subunit_id_t subunit_id = (avrcp_subunit_id_t) (packet[4] & 0x07); 10034b338011SMilanka Ringwald 1004dfbbc26bSMatthias Ringwald avrcp_command_opcode_t opcode = (avrcp_command_opcode_t) avrcp_cmd_opcode(packet,size); 1005e0bbf3edSMilanka Ringwald 1006e8ad6d13SMatthias Ringwald int pos = 6; 1007dfbbc26bSMatthias Ringwald uint16_t length; 1008dfbbc26bSMatthias Ringwald avrcp_pdu_id_t pdu_id; 1009aeb99916SMilanka Ringwald // connection->data_len = 0; 10102c197cf1SMilanka Ringwald uint8_t offset; 10110b578d06SMilanka Ringwald uint8_t operand; 1012367e3c55SMilanka Ringwald uint16_t event_mask; 101338410698SMilanka Ringwald avrcp_operation_id_t operation_id; 1014e0bbf3edSMilanka Ringwald 1015e0bbf3edSMilanka Ringwald switch (opcode){ 10164b338011SMilanka Ringwald case AVRCP_CMD_OPCODE_UNIT_INFO: 1017d1207cd8SMilanka Ringwald avrcp_target_unit_info(connection); 10184b338011SMilanka Ringwald break; 10192c197cf1SMilanka Ringwald case AVRCP_CMD_OPCODE_SUBUNIT_INFO: 102099702381SMatthias Ringwald if ((size - pos) < 3) return; 10212c197cf1SMilanka Ringwald // page: packet[pos] >> 4, 10222c197cf1SMilanka Ringwald offset = 4 * (packet[pos]>>4); 10232c197cf1SMilanka Ringwald // extension code (fixed 7) = packet[pos] & 0x0F 10242c197cf1SMilanka Ringwald // 4 bytes paga data, all 0xFF 1025d1207cd8SMilanka Ringwald avrcp_target_subunit_info(connection, offset); 1026e0bbf3edSMilanka Ringwald break; 10272c197cf1SMilanka Ringwald 10280b578d06SMilanka Ringwald case AVRCP_CMD_OPCODE_PASS_THROUGH: 1029810b0d84SMatthias Ringwald if (size < 8) return; 1030810b0d84SMatthias Ringwald log_info("AVRCP_OPERATION_ID 0x%02x, operands length %d", packet[6], packet[7]); 103138410698SMilanka Ringwald operation_id = (avrcp_operation_id_t) (packet[6] & 0x7f); 10320b578d06SMilanka Ringwald operand = 0; 1033810b0d84SMatthias Ringwald if ((packet[7] >= 1) && (size >= 9)){ 1034810b0d84SMatthias Ringwald operand = packet[8]; 1035810b0d84SMatthias Ringwald } 1036d1207cd8SMilanka Ringwald 10370b578d06SMilanka Ringwald if (avcrp_operation_id_is_valid(operation_id)){ 10380b578d06SMilanka Ringwald bool button_pressed = (packet[6] & 0x80) == 0; 10390b578d06SMilanka Ringwald 1040810b0d84SMatthias Ringwald avrcp_target_operation_accepted(connection->avrcp_cid, (avrcp_operation_id_t) packet[6], packet[7], operand); 1041319131f8SMatthias Ringwald avrcp_target_emit_operation(avrcp_target_context.avrcp_callback, connection->avrcp_cid, 1042927fd23fSMatthias Ringwald operation_id, button_pressed, packet[7], operand); 10430b578d06SMilanka Ringwald } else { 1044810b0d84SMatthias Ringwald avrcp_target_operation_not_implemented(connection->avrcp_cid, (avrcp_operation_id_t) packet[6], packet[7], operand); 1045d1207cd8SMilanka Ringwald } 1046d1207cd8SMilanka Ringwald break; 10470b578d06SMilanka Ringwald 1048d1207cd8SMilanka Ringwald 1049e0bbf3edSMilanka Ringwald case AVRCP_CMD_OPCODE_VENDOR_DEPENDENT: 105099702381SMatthias Ringwald 105199702381SMatthias Ringwald if (size < 13) return; 105299702381SMatthias Ringwald 105399702381SMatthias Ringwald // pos = 6 - company id 1054393f5724SMilanka Ringwald (void)memcpy(connection->message_body, &packet[pos], 3); 1055aeb99916SMilanka Ringwald // connection->data_len = 3; 105699702381SMatthias Ringwald pos += 3; 105799702381SMatthias Ringwald // pos = 9 105899702381SMatthias Ringwald pdu_id = (avrcp_pdu_id_t) packet[pos++]; 105999702381SMatthias Ringwald // 1 - reserved 106099702381SMatthias Ringwald pos++; 106199702381SMatthias Ringwald // 2-3 param length, 106299702381SMatthias Ringwald length = big_endian_read_16(packet, pos); 106399702381SMatthias Ringwald pos += 2; 106499702381SMatthias Ringwald // pos = 13 1065e0bbf3edSMilanka Ringwald switch (pdu_id){ 10666568eb54SMilanka Ringwald case AVRCP_PDU_ID_SET_ADDRESSED_PLAYER:{ 106799702381SMatthias Ringwald if ((pos + 2) > size) return; 1068d1cf25b1SMatthias Ringwald bool ok = length == 4; 1069d1cf25b1SMatthias Ringwald if (avrcp_target_context.set_addressed_player_callback != NULL){ 107099702381SMatthias Ringwald uint16_t player_id = big_endian_read_16(packet, pos); 1071d1cf25b1SMatthias Ringwald ok = avrcp_target_context.set_addressed_player_callback(player_id); 10726568eb54SMilanka Ringwald } 1073d1cf25b1SMatthias Ringwald if (ok){ 1074aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, AVRCP_STATUS_SUCCESS); 1075d1cf25b1SMatthias Ringwald } else { 1076aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PLAYER_ID); 1077d1cf25b1SMatthias Ringwald } 10786568eb54SMilanka Ringwald break; 10796568eb54SMilanka Ringwald } 1080e0bbf3edSMilanka Ringwald case AVRCP_PDU_ID_GET_CAPABILITIES:{ 108199702381SMatthias Ringwald avrcp_capability_id_t capability_id = (avrcp_capability_id_t) packet[pos]; 1082e0bbf3edSMilanka Ringwald switch (capability_id){ 1083e0bbf3edSMilanka Ringwald case AVRCP_CAPABILITY_ID_EVENT: 1084aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_supported_events(connection); 1085e0bbf3edSMilanka Ringwald break; 1086e0bbf3edSMilanka Ringwald case AVRCP_CAPABILITY_ID_COMPANY: 1087aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_supported_companies(connection); 1088e0bbf3edSMilanka Ringwald break; 1089e0bbf3edSMilanka Ringwald default: 1090aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 1091e0bbf3edSMilanka Ringwald break; 1092e0bbf3edSMilanka Ringwald } 1093e0bbf3edSMilanka Ringwald break; 1094e0bbf3edSMilanka Ringwald } 1095c045af99SMilanka Ringwald case AVRCP_PDU_ID_GET_PLAY_STATUS: 1096c045af99SMilanka Ringwald avrcp_target_emit_respond_vendor_dependent_query(avrcp_target_context.avrcp_callback, connection->avrcp_cid, AVRCP_SUBEVENT_PLAY_STATUS_QUERY); 1097c045af99SMilanka Ringwald break; 10981bf7a74fSMilanka Ringwald case AVRCP_PDU_ID_REQUEST_ABORT_CONTINUING_RESPONSE: 109999702381SMatthias Ringwald if ((pos + 1) > size) return; 1100393f5724SMilanka Ringwald connection->target_abort_continue_response = true; 11014e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 11021bf7a74fSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 11031bf7a74fSMilanka Ringwald break; 1104d1207cd8SMilanka Ringwald case AVRCP_PDU_ID_REQUEST_CONTINUING_RESPONSE: 110599702381SMatthias Ringwald if ((pos + 1) > size) return; 110699702381SMatthias Ringwald if (packet[pos] != AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES){ 1107aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 1108d1207cd8SMilanka Ringwald return; 1109d1207cd8SMilanka Ringwald } 11104b47bcb9SMilanka Ringwald connection->target_continue_response = true; 1111393f5724SMilanka Ringwald connection->target_now_playing_info_response = true; 11124e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 1113d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1114d1207cd8SMilanka Ringwald break; 11155e3d4d2bSMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES:{ 111699702381SMatthias Ringwald if ((pos + 9) > size) return; 11175e3d4d2bSMilanka Ringwald uint8_t play_identifier[8]; 11185e3d4d2bSMilanka Ringwald memset(play_identifier, 0, 8); 111999702381SMatthias Ringwald if (memcmp(&packet[pos], play_identifier, 8) != 0) { 1120aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 11215e3d4d2bSMilanka Ringwald return; 11225e3d4d2bSMilanka Ringwald } 1123d1207cd8SMilanka Ringwald pos += 8; 112499702381SMatthias Ringwald uint8_t attribute_count = packet[pos++]; 1125d1207cd8SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_NONE; 1126d1207cd8SMilanka Ringwald if (!attribute_count){ 1127aeb99916SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_TITLE; 1128393f5724SMilanka Ringwald connection->target_now_playing_info_attr_bitmap = 0xFE; 1129d1207cd8SMilanka Ringwald } else { 1130d1207cd8SMilanka Ringwald int i; 1131aeb99916SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_TITLE; 1132393f5724SMilanka Ringwald connection->target_now_playing_info_attr_bitmap = 0; 1133aeb99916SMilanka Ringwald if ((pos + attribute_count * 4) > size) return; 1134d1207cd8SMilanka Ringwald for (i=0; i < attribute_count; i++){ 1135aeb99916SMilanka Ringwald uint32_t attr_id = big_endian_read_32(packet, pos); 1136393f5724SMilanka Ringwald connection->target_now_playing_info_attr_bitmap |= (1 << attr_id); 1137aeb99916SMilanka Ringwald pos += 4; 1138d1207cd8SMilanka Ringwald } 1139d1207cd8SMilanka Ringwald } 1140393f5724SMilanka Ringwald log_info("target_now_playing_info_attr_bitmap 0x%02x", connection->target_now_playing_info_attr_bitmap); 1141393f5724SMilanka Ringwald connection->target_now_playing_info_response = true; 11424e0673dbSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 1143aeb99916SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1144d1207cd8SMilanka Ringwald break; 1145d1207cd8SMilanka Ringwald } 1146d1207cd8SMilanka Ringwald case AVRCP_PDU_ID_REGISTER_NOTIFICATION:{ 114799702381SMatthias Ringwald if ((pos + 1) > size) return; 114899702381SMatthias Ringwald avrcp_notification_event_id_t event_id = (avrcp_notification_event_id_t) packet[pos]; 1149367e3c55SMilanka Ringwald 1150022b77fcSMilanka Ringwald avrcp_target_set_transaction_label_for_notification(connection, event_id, connection->transaction_id); 1151d1207cd8SMilanka Ringwald 1152367e3c55SMilanka Ringwald if (event_id < AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED || 1153367e3c55SMilanka Ringwald event_id > AVRCP_NOTIFICATION_EVENT_MAX_VALUE){ 1154aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 1155367e3c55SMilanka Ringwald return; 1156367e3c55SMilanka Ringwald } 1157367e3c55SMilanka Ringwald 1158367e3c55SMilanka Ringwald switch (event_id){ 1159367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_AVAILABLE_PLAYERS_CHANGED: 1160367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYER_APPLICATION_SETTING_CHANGED: 1161367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_UIDS_CHANGED: 11624b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_REACHED_END: 11634b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_REACHED_START: 11644b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYBACK_POS_CHANGED: 11654b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_SYSTEM_STATUS_CHANGED: 11664b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_MAX_VALUE: 1167aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_not_implemented(connection, pdu_id, event_id); 1168367e3c55SMilanka Ringwald return; 1169367e3c55SMilanka Ringwald default: 1170367e3c55SMilanka Ringwald break; 1171367e3c55SMilanka Ringwald } 1172367e3c55SMilanka Ringwald 1173367e3c55SMilanka Ringwald event_mask = (1 << event_id); 1174367e3c55SMilanka Ringwald connection->notifications_enabled |= event_mask; 1175367e3c55SMilanka Ringwald 1176d1207cd8SMilanka Ringwald switch (event_id){ 1177d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED: 1178393f5724SMilanka Ringwald if (connection->target_track_selected){ 1179d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, AVRCP_NOTIFICATION_TRACK_SELECTED, 8); 1180d1207cd8SMilanka Ringwald } else { 1181d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, AVRCP_NOTIFICATION_TRACK_NOT_SELECTED, 8); 1182d1207cd8SMilanka Ringwald } 1183d1207cd8SMilanka Ringwald break; 1184d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED: 1185393f5724SMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->target_playback_status, 1); 1186d1207cd8SMilanka Ringwald break; 1187d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED: 1188d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, NULL, 0); 1189d1207cd8SMilanka Ringwald break; 1190d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED: 1191393f5724SMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->target_absolute_volume, 1); 1192d1207cd8SMilanka Ringwald break; 1193d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED: 1194393f5724SMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->target_battery_status, 1); 1195d1207cd8SMilanka Ringwald break; 11966568eb54SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED: 1197aeb99916SMilanka Ringwald avrcp_target_response_addressed_player_changed_interim(connection, pdu_id, event_id); 11986568eb54SMilanka Ringwald return; 1199d1207cd8SMilanka Ringwald default: 1200367e3c55SMilanka Ringwald btstack_assert(false); 1201d1207cd8SMilanka Ringwald return; 1202d1207cd8SMilanka Ringwald } 12035e3d4d2bSMilanka Ringwald break; 12045e3d4d2bSMilanka Ringwald } 120512b7c8ecSMilanka Ringwald case AVRCP_PDU_ID_SET_ABSOLUTE_VOLUME: { 1206984bbc3eSMilanka Ringwald if ( (length != 1) || ((pos + 1) > size)){ 1207aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 120812b7c8ecSMilanka Ringwald break; 120912b7c8ecSMilanka Ringwald } 121012b7c8ecSMilanka Ringwald 121199702381SMatthias Ringwald uint8_t absolute_volume = packet[pos]; 12120a2feef4SMilanka Ringwald if (absolute_volume < 0x80){ 1213393f5724SMilanka Ringwald connection->target_absolute_volume = absolute_volume; 12140a2feef4SMilanka Ringwald } 1215393f5724SMilanka Ringwald avrcp_target_emit_volume_changed(avrcp_target_context.avrcp_callback, connection->avrcp_cid, connection->target_absolute_volume); 1216393f5724SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, connection->target_absolute_volume); 121712b7c8ecSMilanka Ringwald break; 121812b7c8ecSMilanka Ringwald } 1219e0bbf3edSMilanka Ringwald default: 12201bf7a74fSMilanka Ringwald log_info("AVRCP target: unhandled pdu id 0x%02x", pdu_id); 1221aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 1222e0bbf3edSMilanka Ringwald break; 1223e0bbf3edSMilanka Ringwald } 12244b338011SMilanka Ringwald break; 12254b338011SMilanka Ringwald default: 12261bf7a74fSMilanka Ringwald log_info("AVRCP target: opcode 0x%02x not implemented", avrcp_cmd_opcode(packet,size)); 12274b338011SMilanka Ringwald break; 12284b338011SMilanka Ringwald } 122901dc6e35SMilanka Ringwald } 123001dc6e35SMilanka Ringwald 1231aeb99916SMilanka 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){ 1232aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_CHANGED_STABLE, AVRCP_PDU_ID_REGISTER_NOTIFICATION); 1233022b77fcSMilanka Ringwald connection->transaction_id = avrcp_target_get_transaction_label_for_notification(connection, notification_id); 1234d1207cd8SMilanka Ringwald 1235aeb99916SMilanka Ringwald connection->data_len = 1 + value_len; 1236aeb99916SMilanka Ringwald connection->data[0] = notification_id; 1237276c795aSMilanka Ringwald if (value != NULL){ 1238aeb99916SMilanka Ringwald (void)memcpy(connection->data + 1, value, value_len); 1239aeb99916SMilanka Ringwald } 1240df842b64SMatthias Ringwald } 1241bf67b2dbSMatthias Ringwald 1242aeb99916SMilanka Ringwald static void avrcp_target_notification_addressed_player_changed_init(avrcp_connection_t * connection){ 1243aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_CHANGED_STABLE, AVRCP_PDU_ID_REGISTER_NOTIFICATION); 1244aeb99916SMilanka Ringwald connection->transaction_id = avrcp_target_get_transaction_label_for_notification(connection, AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED); 1245aeb99916SMilanka Ringwald 1246aeb99916SMilanka Ringwald connection->data_len = 5; 1247aeb99916SMilanka Ringwald connection->data[0] = AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED; 1248393f5724SMilanka Ringwald big_endian_store_16(connection->data, 1, connection->target_addressed_player_id); 1249393f5724SMilanka Ringwald big_endian_store_16(connection->data, 3, connection->target_uid_counter); 1250d1207cd8SMilanka Ringwald } 1251d1207cd8SMilanka Ringwald 1252aeb99916SMilanka Ringwald 12530ec79bd0SMilanka Ringwald static void avrcp_target_reset_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification_id){ 1254aeb99916SMilanka Ringwald if (notification_id < AVRCP_NOTIFICATION_EVENT_FIRST_INDEX || notification_id > AVRCP_NOTIFICATION_EVENT_LAST_INDEX){ 125512b7c8ecSMilanka Ringwald return; 125612b7c8ecSMilanka Ringwald } 125712b7c8ecSMilanka Ringwald connection->notifications_enabled &= ~(1 << notification_id); 1258393f5724SMilanka Ringwald connection->target_notifications_transaction_label[notification_id] = 0; 1259aeb99916SMilanka Ringwald } 1260aeb99916SMilanka Ringwald 1261aeb99916SMilanka Ringwald static void avrcp_request_next_avctp_segment(avrcp_connection_t * connection){ 1262aeb99916SMilanka Ringwald // AVCTP 1263aeb99916SMilanka Ringwald switch (connection->avctp_packet_type){ 1264aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 1265ccfcd81eSMilanka Ringwald case AVCTP_SINGLE_PACKET: 1266aeb99916SMilanka Ringwald connection->state = AVCTP_CONNECTION_OPENED; 1267aeb99916SMilanka Ringwald break; 1268aeb99916SMilanka Ringwald default: 1269ffe8f3acSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 1270aeb99916SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1271aeb99916SMilanka Ringwald break; 1272aeb99916SMilanka Ringwald } 127312b7c8ecSMilanka Ringwald } 127412b7c8ecSMilanka Ringwald 1275654724deSMilanka Ringwald static void avrcp_target_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 127601dc6e35SMilanka Ringwald avrcp_connection_t * connection; 1277ffe8f3acSMilanka Ringwald avrcp_notification_event_id_t notification_id = AVRCP_NOTIFICATION_EVENT_NONE; 1278aeb99916SMilanka Ringwald 127901dc6e35SMilanka Ringwald switch (packet_type){ 128001dc6e35SMilanka Ringwald case L2CAP_DATA_PACKET: 12811945fe3eSMilanka Ringwald connection = avrcp_get_connection_for_l2cap_signaling_cid_for_role(AVRCP_TARGET, channel); 128201dc6e35SMilanka Ringwald avrcp_handle_l2cap_data_packet_for_signaling_connection(connection, packet, size); 1283ffe8f3acSMilanka Ringwald return; 1284ffe8f3acSMilanka Ringwald 128501dc6e35SMilanka Ringwald case HCI_EVENT_PACKET: 1286ffe8f3acSMilanka Ringwald if (hci_event_packet_get_type(packet) != L2CAP_EVENT_CAN_SEND_NOW){ 1287ffe8f3acSMilanka Ringwald return; 1288ffe8f3acSMilanka Ringwald } 1289ffe8f3acSMilanka Ringwald 12901945fe3eSMilanka Ringwald connection = avrcp_get_connection_for_l2cap_signaling_cid_for_role(AVRCP_TARGET, channel); 1291ffe8f3acSMilanka Ringwald if (connection == NULL){ 1292ffe8f3acSMilanka Ringwald return; 1293ffe8f3acSMilanka Ringwald } 1294ffe8f3acSMilanka Ringwald 12954e0673dbSMilanka Ringwald if (connection->state == AVCTP_W2_SEND_RESPONSE){ 1296d3db1226SMilanka Ringwald // start AVCTP 1297393f5724SMilanka Ringwald if (connection->target_reject_transport_header){ 1298393f5724SMilanka Ringwald connection->target_reject_transport_header = false; 1299ffe8f3acSMilanka Ringwald avctp_send_reject_cmd_wrong_pid(connection); 1300ffe8f3acSMilanka Ringwald connection->state = AVCTP_CONNECTION_OPENED; 1301ffe8f3acSMilanka Ringwald return; 1302ffe8f3acSMilanka Ringwald } 1303d3db1226SMilanka Ringwald // end AVCTP 1304d1207cd8SMilanka Ringwald 1305d3db1226SMilanka Ringwald // start AVRCP 1306393f5724SMilanka Ringwald if (connection->target_abort_continue_response){ 1307393f5724SMilanka Ringwald connection->target_abort_continue_response = false; 1308aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, AVRCP_PDU_ID_REQUEST_ABORT_CONTINUING_RESPONSE); 1309ffe8f3acSMilanka Ringwald break; 13101bf7a74fSMilanka Ringwald } 13111bf7a74fSMilanka Ringwald 1312393f5724SMilanka Ringwald if (connection->target_now_playing_info_response){ 1313393f5724SMilanka Ringwald connection->target_now_playing_info_response = false; 13144b47bcb9SMilanka Ringwald if (connection->target_continue_response){ 13154e0673dbSMilanka Ringwald connection->target_continue_response = false; 13161e852c68SMilanka Ringwald if (connection->data_len == 0){ 13171e852c68SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, connection->pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 13181e852c68SMilanka Ringwald return; 13191e852c68SMilanka Ringwald } 13204b47bcb9SMilanka Ringwald } else { 1321e2ceb2dfSMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES); 1322aeb99916SMilanka Ringwald connection->data_len = avrcp_now_playing_info_value_len_with_headers(connection); 13234b47bcb9SMilanka Ringwald } 1324ffe8f3acSMilanka Ringwald break; 13255e3d4d2bSMilanka Ringwald } 1326d1207cd8SMilanka Ringwald 13274e0673dbSMilanka Ringwald // data already prepared 13284e0673dbSMilanka Ringwald break; 13294e0673dbSMilanka Ringwald } 13304e0673dbSMilanka Ringwald 13314e0673dbSMilanka Ringwald // Notifications 13324e0673dbSMilanka Ringwald 1333393f5724SMilanka Ringwald if (connection->target_track_changed){ 1334393f5724SMilanka Ringwald connection->target_track_changed = false; 1335ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED; 1336393f5724SMilanka Ringwald avrcp_target_notification_init(connection, notification_id, connection->target_track_id, 8); 1337ffe8f3acSMilanka Ringwald break; 1338d1207cd8SMilanka Ringwald } 1339d1207cd8SMilanka Ringwald 1340393f5724SMilanka Ringwald if (connection->target_playback_status_changed){ 1341393f5724SMilanka Ringwald connection->target_playback_status_changed = false; 1342ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED; 1343393f5724SMilanka Ringwald uint8_t playback_status = (uint8_t) connection->target_playback_status; 1344ffe8f3acSMilanka Ringwald avrcp_target_notification_init(connection, notification_id, &playback_status, 1); 1345ffe8f3acSMilanka Ringwald break; 1346d1207cd8SMilanka Ringwald } 1347d1207cd8SMilanka Ringwald 1348393f5724SMilanka Ringwald if (connection->target_playing_content_changed){ 1349393f5724SMilanka Ringwald connection->target_playing_content_changed = false; 1350ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED; 1351276c795aSMilanka Ringwald avrcp_target_notification_init(connection, notification_id, NULL, 0); 1352ffe8f3acSMilanka Ringwald break; 1353d1207cd8SMilanka Ringwald } 1354d1207cd8SMilanka Ringwald 1355393f5724SMilanka Ringwald if (connection->target_battery_status_changed){ 1356393f5724SMilanka Ringwald connection->target_battery_status_changed = false; 1357ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED; 1358393f5724SMilanka Ringwald avrcp_target_notification_init(connection, notification_id, (uint8_t *)&connection->target_battery_status, 1); 1359ffe8f3acSMilanka Ringwald break; 1360d1207cd8SMilanka Ringwald } 1361d1207cd8SMilanka Ringwald 1362393f5724SMilanka Ringwald if (connection->target_notify_absolute_volume_changed){ 1363393f5724SMilanka Ringwald connection->target_notify_absolute_volume_changed = false; 1364ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED; 1365393f5724SMilanka Ringwald avrcp_target_notification_init(connection, notification_id, &connection->target_absolute_volume, 1); 1366ffe8f3acSMilanka Ringwald break; 1367d1207cd8SMilanka Ringwald } 1368d1207cd8SMilanka Ringwald 1369393f5724SMilanka Ringwald if (connection->target_addressed_player_changed){ 1370393f5724SMilanka Ringwald connection->target_addressed_player_changed = false; 1371ffe8f3acSMilanka Ringwald notification_id = AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED; 1372aeb99916SMilanka Ringwald avrcp_target_notification_addressed_player_changed_init(connection); 1373ffe8f3acSMilanka Ringwald break; 1374ffe8f3acSMilanka Ringwald } 1375ffe8f3acSMilanka Ringwald 137679774d78SMilanka Ringwald // nothing to send, exit 137779774d78SMilanka Ringwald return; 1378ffe8f3acSMilanka Ringwald 1379ffe8f3acSMilanka Ringwald default: 1380f1028954SMilanka Ringwald return; 1381f1028954SMilanka Ringwald } 1382f1028954SMilanka Ringwald 1383aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 1384ffe8f3acSMilanka Ringwald avrcp_target_reset_notification(connection, notification_id); 13853b899686SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 138601dc6e35SMilanka Ringwald } 138701dc6e35SMilanka Ringwald 138801dc6e35SMilanka Ringwald void avrcp_target_init(void){ 138901dc6e35SMilanka Ringwald avrcp_target_context.role = AVRCP_TARGET; 1390654724deSMilanka Ringwald avrcp_target_context.packet_handler = avrcp_target_packet_handler; 13910ec79bd0SMilanka Ringwald avrcp_register_target_packet_handler(&avrcp_target_packet_handler); 139201dc6e35SMilanka Ringwald } 139301dc6e35SMilanka Ringwald 1394680af5dcSMatthias Ringwald void avrcp_target_deinit(void){ 1395680af5dcSMatthias Ringwald memset(&avrcp_target_context, 0, sizeof(avrcp_context_t)); 1396680af5dcSMatthias Ringwald } 1397680af5dcSMatthias Ringwald 139801dc6e35SMilanka Ringwald void avrcp_target_register_packet_handler(btstack_packet_handler_t callback){ 1399be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 140001dc6e35SMilanka Ringwald avrcp_target_context.avrcp_callback = callback; 140101dc6e35SMilanka Ringwald } 140201dc6e35SMilanka Ringwald 1403d1cf25b1SMatthias Ringwald void avrcp_target_register_set_addressed_player_handler(bool (*callback)(uint16_t player_id)){ 140495f00c7eSMilanka Ringwald btstack_assert(callback != NULL); 1405d1cf25b1SMatthias Ringwald avrcp_target_context.set_addressed_player_callback = callback; 1406d1cf25b1SMatthias Ringwald } 1407d1cf25b1SMatthias Ringwald 140801dc6e35SMilanka Ringwald 1409