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 54*aeb99916SMilanka 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 62*aeb99916SMilanka Ringwald static uint32_t default_companies[] = { 63*aeb99916SMilanka Ringwald 0x581900 //BT SIG registered CompanyID 64*aeb99916SMilanka Ringwald }; 65*aeb99916SMilanka 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 121*aeb99916SMilanka 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){ 122*aeb99916SMilanka Ringwald btstack_assert(attr_id > AVRCP_MEDIA_ATTR_ALL); 123*aeb99916SMilanka Ringwald btstack_assert(attr_id < AVRCP_MEDIA_ATTR_RESERVED); 124*aeb99916SMilanka Ringwald uint16_t pos = 0; 125*aeb99916SMilanka Ringwald big_endian_store_32(buffer, pos, attr_id); 126*aeb99916SMilanka Ringwald big_endian_store_16(buffer, pos + 4, RFC2978_CHARSET_MIB_UTF8); 127*aeb99916SMilanka Ringwald big_endian_store_16(buffer, pos + 6, attr_value_size); 1285e845fceSMatthias Ringwald return 8; 1295e845fceSMatthias Ringwald } 1305e845fceSMatthias Ringwald 131*aeb99916SMilanka Ringwald static uint16_t avrcp_now_playing_info_attr_id_value_len(avrcp_connection_t * connection, avrcp_media_attribute_id_t attr_id){ 132*aeb99916SMilanka Ringwald char buffer[AVRCP_MAX_ATTRIBUTE_SIZE]; 133*aeb99916SMilanka Ringwald uint16_t str_len; 1345e3d4d2bSMilanka Ringwald switch (attr_id) { 1352f511ffbSMilanka Ringwald case AVRCP_MEDIA_ATTR_ALL: 136d1207cd8SMilanka Ringwald case AVRCP_MEDIA_ATTR_NONE: 137*aeb99916SMilanka Ringwald return 0; 1385e3d4d2bSMilanka Ringwald case AVRCP_MEDIA_ATTR_TRACK: 139*aeb99916SMilanka Ringwald str_len = sprintf(buffer, "%0" PRIu32, connection->track_nr); 1405e3d4d2bSMilanka Ringwald break; 141864d08b0SMilanka Ringwald case AVRCP_MEDIA_ATTR_TOTAL_NUM_ITEMS: 142*aeb99916SMilanka Ringwald str_len = sprintf(buffer, "%0" PRIu32, connection->total_tracks); 1435e3d4d2bSMilanka Ringwald break; 1442f511ffbSMilanka Ringwald case AVRCP_MEDIA_ATTR_SONG_LENGTH_MS: 145*aeb99916SMilanka Ringwald str_len = sprintf(buffer, "%0" PRIu32, connection->song_length_ms); 1465e3d4d2bSMilanka Ringwald break; 1475e3d4d2bSMilanka Ringwald default: 148*aeb99916SMilanka Ringwald str_len = connection->now_playing_info[(uint16_t)attr_id - 1].len; 1495e3d4d2bSMilanka Ringwald break; 1505e3d4d2bSMilanka Ringwald } 151*aeb99916SMilanka Ringwald return str_len; 1525e3d4d2bSMilanka Ringwald } 1535e3d4d2bSMilanka Ringwald 154*aeb99916SMilanka Ringwald static uint16_t avrcp_now_playing_info_value_len_with_headers(avrcp_connection_t * connection){ 155*aeb99916SMilanka Ringwald uint16_t playing_info_len = 0; 156d1207cd8SMilanka Ringwald 157*aeb99916SMilanka Ringwald uint8_t i; 158*aeb99916SMilanka Ringwald for ( i = (uint8_t)AVRCP_MEDIA_ATTR_ALL + 1; i < (uint8_t) AVRCP_MEDIA_ATTR_RESERVED; i++){ 159*aeb99916SMilanka Ringwald avrcp_media_attribute_id_t attr_id = (avrcp_media_attribute_id_t) i; 1605e3d4d2bSMilanka Ringwald 161*aeb99916SMilanka Ringwald if ((connection->now_playing_info_attr_bitmap & (1 << attr_id)) == 0) { 162*aeb99916SMilanka Ringwald continue; 163*aeb99916SMilanka Ringwald } 164*aeb99916SMilanka Ringwald 165*aeb99916SMilanka Ringwald switch (attr_id) { 166*aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_ALL: 167*aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_NONE: 168*aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_DEFAULT_COVER_ART: 169*aeb99916SMilanka Ringwald break; 170*aeb99916SMilanka Ringwald default: 171*aeb99916SMilanka Ringwald playing_info_len += AVRCP_ATTR_HEADER_LEN + avrcp_now_playing_info_attr_id_value_len(connection, attr_id); 172*aeb99916SMilanka Ringwald break; 173*aeb99916SMilanka Ringwald } 174*aeb99916SMilanka Ringwald } 175*aeb99916SMilanka Ringwald return playing_info_len; 176*aeb99916SMilanka Ringwald } 177*aeb99916SMilanka Ringwald 178*aeb99916SMilanka Ringwald static uint8_t * avrcp_get_attribute_value_from_u32(avrcp_connection_t * connection, uint32_t value, uint16_t * num_bytes_to_copy){ 179*aeb99916SMilanka Ringwald *num_bytes_to_copy = 0; 180*aeb99916SMilanka Ringwald 181*aeb99916SMilanka Ringwald if (connection->attribute_value_len == 0){ 182*aeb99916SMilanka Ringwald connection->attribute_value_len = sprintf((char *)connection->attribute_value, "%0" PRIu32, value); 183*aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 184*aeb99916SMilanka Ringwald } 185*aeb99916SMilanka Ringwald *num_bytes_to_copy = connection->attribute_value_len - connection->attribute_value_offset; 186*aeb99916SMilanka Ringwald return connection->attribute_value + connection->attribute_value_offset; 187*aeb99916SMilanka Ringwald } 188*aeb99916SMilanka Ringwald 189*aeb99916SMilanka 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){ 190*aeb99916SMilanka Ringwald switch (attr_id){ 191*aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_TRACK: 192*aeb99916SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->track_nr, num_bytes_to_copy); 193*aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_TOTAL_NUM_ITEMS: 194*aeb99916SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->total_tracks, num_bytes_to_copy); 195*aeb99916SMilanka Ringwald case AVRCP_MEDIA_ATTR_SONG_LENGTH_MS: 196*aeb99916SMilanka Ringwald return avrcp_get_attribute_value_from_u32(connection, connection->song_length_ms, num_bytes_to_copy); 197*aeb99916SMilanka Ringwald default: 198*aeb99916SMilanka Ringwald break; 199*aeb99916SMilanka Ringwald } 200*aeb99916SMilanka Ringwald int attr_index = attr_id - 1; 201*aeb99916SMilanka Ringwald if (connection->attribute_value_len == 0){ 202*aeb99916SMilanka Ringwald connection->attribute_value_len = avrcp_now_playing_info_attr_id_value_len(connection, attr_id); 203*aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 204*aeb99916SMilanka Ringwald } 205*aeb99916SMilanka Ringwald *num_bytes_to_copy = connection->now_playing_info[attr_index].len - connection->attribute_value_offset; 206*aeb99916SMilanka Ringwald return (uint8_t *) (connection->now_playing_info[attr_index].value + connection->attribute_value_offset); 207*aeb99916SMilanka Ringwald } 208*aeb99916SMilanka Ringwald 209*aeb99916SMilanka Ringwald // TODO Review 210*aeb99916SMilanka Ringwald static uint16_t avrcp_store_avctp_now_playing_info_fragment(avrcp_connection_t * connection, uint16_t packet_size, uint8_t * packet){ 211*aeb99916SMilanka Ringwald uint16_t num_free_bytes = packet_size; 212*aeb99916SMilanka Ringwald 213*aeb99916SMilanka Ringwald uint16_t bytes_stored = 0; 214*aeb99916SMilanka Ringwald 215*aeb99916SMilanka Ringwald while ((num_free_bytes > 0) && (connection->next_attr_id <= AVRCP_MEDIA_ATTR_SONG_LENGTH_MS)){ 216*aeb99916SMilanka Ringwald if ((connection->now_playing_info_attr_bitmap & (1 << (uint8_t)connection->next_attr_id)) == 0) { 217*aeb99916SMilanka Ringwald continue; 218*aeb99916SMilanka Ringwald } 219*aeb99916SMilanka Ringwald 220*aeb99916SMilanka Ringwald // store header 221*aeb99916SMilanka Ringwald if (connection->attribute_value_offset == 0){ 222*aeb99916SMilanka Ringwald // pack the whole attribute value header 223*aeb99916SMilanka Ringwald if (connection->parser_attribute_header_pos == 0) { 224*aeb99916SMilanka Ringwald avrcp_target_pack_single_element_header(connection->parser_attribute_header, connection->next_attr_id, 225*aeb99916SMilanka Ringwald connection->attribute_value_len); 226*aeb99916SMilanka Ringwald connection->parser_attribute_header_pos = 0; 227*aeb99916SMilanka Ringwald } 228*aeb99916SMilanka Ringwald 229*aeb99916SMilanka Ringwald if (connection->parser_attribute_header_pos < AVRCP_ATTRIBUTE_HEADER_LEN){ 230*aeb99916SMilanka Ringwald uint16_t num_header_bytes_to_store = btstack_min(num_free_bytes, AVRCP_ATTRIBUTE_HEADER_LEN - connection->parser_attribute_header_pos); 231*aeb99916SMilanka Ringwald memcpy(packet + bytes_stored, connection->parser_attribute_header + connection->parser_attribute_header_pos, num_header_bytes_to_store); 232*aeb99916SMilanka Ringwald connection->parser_attribute_header_pos += num_header_bytes_to_store; 233*aeb99916SMilanka Ringwald bytes_stored += num_header_bytes_to_store; 234*aeb99916SMilanka Ringwald num_free_bytes -= num_header_bytes_to_store; 235*aeb99916SMilanka Ringwald connection->data_offset += num_header_bytes_to_store; 236*aeb99916SMilanka Ringwald 237*aeb99916SMilanka Ringwald if (num_free_bytes == 0){ 238*aeb99916SMilanka Ringwald continue; 239*aeb99916SMilanka Ringwald } 240*aeb99916SMilanka Ringwald } 241*aeb99916SMilanka Ringwald } 242*aeb99916SMilanka Ringwald 243*aeb99916SMilanka Ringwald // store value 244*aeb99916SMilanka Ringwald uint16_t num_bytes_to_copy; 245*aeb99916SMilanka Ringwald uint8_t * attr_value_with_offset = avrcp_get_next_value_fragment_for_attribute_id(connection, 246*aeb99916SMilanka Ringwald connection->next_attr_id, 247*aeb99916SMilanka Ringwald &num_bytes_to_copy); 248*aeb99916SMilanka Ringwald 249*aeb99916SMilanka Ringwald uint16_t num_attr_value_bytes_to_store = btstack_min(num_free_bytes, connection->attribute_value_len - connection->attribute_value_offset); 250*aeb99916SMilanka Ringwald memcpy(packet + bytes_stored, attr_value_with_offset, num_attr_value_bytes_to_store); 251*aeb99916SMilanka Ringwald 252*aeb99916SMilanka Ringwald bytes_stored += num_attr_value_bytes_to_store; 253*aeb99916SMilanka Ringwald num_free_bytes -= num_attr_value_bytes_to_store; 254*aeb99916SMilanka Ringwald connection->attribute_value_offset += num_attr_value_bytes_to_store; 255*aeb99916SMilanka Ringwald connection->data_offset += num_attr_value_bytes_to_store; 256*aeb99916SMilanka Ringwald 257*aeb99916SMilanka Ringwald if (connection->attribute_value_offset == connection->attribute_value_len){ 258*aeb99916SMilanka Ringwald // C++ compatible version of connection->next_attr_id++ 259*aeb99916SMilanka Ringwald connection->next_attr_id = (avrcp_media_attribute_id_t) (((int) connection->next_attr_id) + 1); 260*aeb99916SMilanka Ringwald connection->attribute_value_offset = 0; 261*aeb99916SMilanka Ringwald connection->attribute_value_len = 0; 262*aeb99916SMilanka Ringwald connection->parser_attribute_header_pos = 0; 263*aeb99916SMilanka Ringwald } 264*aeb99916SMilanka Ringwald } 265*aeb99916SMilanka Ringwald return bytes_stored; 266*aeb99916SMilanka Ringwald } 267*aeb99916SMilanka Ringwald 268*aeb99916SMilanka Ringwald static void avrcp_send_response_with_avctp_fragmentation(avrcp_connection_t * connection){ 2695e3d4d2bSMilanka Ringwald l2cap_reserve_packet_buffer(); 2705e3d4d2bSMilanka Ringwald uint8_t * packet = l2cap_get_outgoing_buffer(); 2715e3d4d2bSMilanka Ringwald 2725e3d4d2bSMilanka Ringwald // transport header 2735e3d4d2bSMilanka Ringwald // Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 274d1207cd8SMilanka Ringwald 275*aeb99916SMilanka Ringwald uint16_t max_payload_size; 276*aeb99916SMilanka Ringwald connection->avctp_packet_type = avctp_get_packet_type(connection, &max_payload_size); 277*aeb99916SMilanka Ringwald connection->packet_type = avrcp_get_packet_type(connection); 278d1207cd8SMilanka Ringwald 279*aeb99916SMilanka Ringwald // AVCTP header 280*aeb99916SMilanka Ringwald // transport header : transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 281*aeb99916SMilanka Ringwald uint16_t pos = 0; 282*aeb99916SMilanka Ringwald packet[pos++] = (connection->transaction_id << 4) | (connection->avctp_packet_type << 2) | (AVRCP_RESPONSE_FRAME << 1) | 0; 283*aeb99916SMilanka Ringwald 284*aeb99916SMilanka Ringwald uint8_t param_len = connection->data_len; 285*aeb99916SMilanka Ringwald 286*aeb99916SMilanka Ringwald if (connection->avctp_packet_type == AVCTP_START_PACKET){ 287*aeb99916SMilanka Ringwald uint8_t max_frame_size = btstack_min(l2cap_get_remote_mtu_for_local_cid(connection->l2cap_signaling_cid), AVRCP_MAX_AV_C_MESSAGE_FRAME_SIZE); 288*aeb99916SMilanka Ringwald // first packet: max_payload_size 289*aeb99916SMilanka Ringwald // rest packets 290*aeb99916SMilanka Ringwald uint16_t num_payload_bytes = param_len - max_payload_size; 291*aeb99916SMilanka Ringwald uint16_t frame_size_for_continue_packet = max_frame_size - avctp_get_num_bytes_for_header(AVCTP_CONTINUE_PACKET); 292*aeb99916SMilanka Ringwald uint16_t num_avctp_packets = (num_payload_bytes + frame_size_for_continue_packet - 1)/frame_size_for_continue_packet + 1; 293*aeb99916SMilanka Ringwald packet[pos++] = num_avctp_packets; 294*aeb99916SMilanka Ringwald } 295*aeb99916SMilanka Ringwald 296*aeb99916SMilanka Ringwald uint16_t bytes_stored = 0; 297*aeb99916SMilanka Ringwald uint8_t i; 298*aeb99916SMilanka Ringwald 299*aeb99916SMilanka Ringwald switch (connection->avctp_packet_type) { 300*aeb99916SMilanka Ringwald case AVCTP_SINGLE_PACKET: 301*aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 3025e3d4d2bSMilanka Ringwald // Profile IDentifier (PID) 3035e3d4d2bSMilanka Ringwald packet[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8; 3045e3d4d2bSMilanka Ringwald packet[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF; 305*aeb99916SMilanka Ringwald 306*aeb99916SMilanka Ringwald // AVRCP message 3075e3d4d2bSMilanka Ringwald // command_type 3085e3d4d2bSMilanka Ringwald packet[pos++] = connection->command_type; 3095e3d4d2bSMilanka Ringwald // subunit_type | subunit ID 3105e3d4d2bSMilanka Ringwald packet[pos++] = (connection->subunit_type << 3) | connection->subunit_id; 3115e3d4d2bSMilanka Ringwald // opcode 3125e3d4d2bSMilanka Ringwald packet[pos++] = (uint8_t) connection->command_opcode; 3130ec79bd0SMilanka Ringwald 314*aeb99916SMilanka Ringwald switch (connection->command_opcode) { 315*aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_VENDOR_DEPENDENT: 316*aeb99916SMilanka Ringwald big_endian_store_24(packet, pos, connection->company_id); 317*aeb99916SMilanka Ringwald pos += 3; 318*aeb99916SMilanka Ringwald packet[pos++] = connection->pdu_id; 319*aeb99916SMilanka Ringwald // AVRCP packet type 3200ec79bd0SMilanka Ringwald 321*aeb99916SMilanka Ringwald packet[pos++] = (uint8_t)connection->packet_type; 322*aeb99916SMilanka Ringwald // parameter length 323*aeb99916SMilanka Ringwald big_endian_store_16(packet, pos, param_len); 324*aeb99916SMilanka Ringwald pos += 2; 325*aeb99916SMilanka Ringwald 326*aeb99916SMilanka Ringwald switch (connection->pdu_id) { 327*aeb99916SMilanka Ringwald // message is small enough to fit the single packet, no need for extra check 328*aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_CAPABILITIES: 329*aeb99916SMilanka Ringwald packet[pos++] = connection->data[0]; 330*aeb99916SMilanka Ringwald connection->data_offset++; 331*aeb99916SMilanka Ringwald 332*aeb99916SMilanka Ringwald packet[pos++] = connection->data[1]; 333*aeb99916SMilanka Ringwald connection->data_offset++; 334*aeb99916SMilanka Ringwald 335*aeb99916SMilanka Ringwald switch ((avrcp_capability_id_t) connection->data[0]) { 336*aeb99916SMilanka Ringwald case AVRCP_CAPABILITY_ID_EVENT: 337*aeb99916SMilanka Ringwald for (i = (uint8_t) AVRCP_NOTIFICATION_EVENT_FIRST_INDEX; 338*aeb99916SMilanka Ringwald i < (uint8_t) AVRCP_NOTIFICATION_EVENT_LAST_INDEX; i++) { 339*aeb99916SMilanka Ringwald if ((connection->target_supported_notifications & (1 << i)) == 0) { 340*aeb99916SMilanka Ringwald continue; 341*aeb99916SMilanka Ringwald } 342*aeb99916SMilanka Ringwald packet[pos++] = i; 343*aeb99916SMilanka Ringwald connection->data_offset++; 344*aeb99916SMilanka Ringwald } 345*aeb99916SMilanka Ringwald break; 346*aeb99916SMilanka Ringwald case AVRCP_CAPABILITY_ID_COMPANY: 347*aeb99916SMilanka Ringwald // use Bluetooth SIG as default company 348*aeb99916SMilanka Ringwald if (connection->target_supported_companies_num == 0) { 349*aeb99916SMilanka Ringwald little_endian_store_24(packet, pos, default_companies[0]); 350*aeb99916SMilanka Ringwald pos += 3; 351*aeb99916SMilanka Ringwald connection->data_offset += 3; 352*aeb99916SMilanka Ringwald } else { 353*aeb99916SMilanka Ringwald for (i = 0; i < connection->data[1]; i++) { 354*aeb99916SMilanka Ringwald little_endian_store_24(packet, pos, 355*aeb99916SMilanka Ringwald connection->target_supported_companies[i]); 356*aeb99916SMilanka Ringwald pos += 3; 357*aeb99916SMilanka Ringwald connection->data_offset += 3; 358*aeb99916SMilanka Ringwald } 359*aeb99916SMilanka Ringwald } 360*aeb99916SMilanka Ringwald break; 361*aeb99916SMilanka Ringwald default: 362*aeb99916SMilanka Ringwald btstack_assert(false); 363*aeb99916SMilanka Ringwald return; 364*aeb99916SMilanka Ringwald } 365*aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos); 366*aeb99916SMilanka Ringwald return; 367*aeb99916SMilanka Ringwald 368*aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES: 369*aeb99916SMilanka Ringwald packet[pos++] = connection->num_attributes; 370*aeb99916SMilanka Ringwald bytes_stored = avrcp_store_avctp_now_playing_info_fragment(connection, max_payload_size, packet + pos); 371*aeb99916SMilanka Ringwald 372*aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += bytes_stored + pos; 373*aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, bytes_stored); 374*aeb99916SMilanka Ringwald return; 375*aeb99916SMilanka Ringwald 376*aeb99916SMilanka Ringwald default: 377*aeb99916SMilanka Ringwald break; 378*aeb99916SMilanka Ringwald } 379*aeb99916SMilanka Ringwald break; 380*aeb99916SMilanka Ringwald 381*aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_PASS_THROUGH: 382*aeb99916SMilanka Ringwald packet[pos++] = connection->operation_id; 383*aeb99916SMilanka Ringwald // parameter length 384*aeb99916SMilanka Ringwald packet[pos++] = (uint8_t) connection->data_len; 385*aeb99916SMilanka Ringwald pos += 2; 386*aeb99916SMilanka Ringwald break; 387*aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_UNIT_INFO: 388*aeb99916SMilanka Ringwald break; 389*aeb99916SMilanka Ringwald case AVRCP_CMD_OPCODE_SUBUNIT_INFO: 390*aeb99916SMilanka Ringwald break; 391*aeb99916SMilanka Ringwald default: 392*aeb99916SMilanka Ringwald btstack_assert(false); 393*aeb99916SMilanka Ringwald return; 394*aeb99916SMilanka Ringwald } 395*aeb99916SMilanka Ringwald break; 396*aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 397*aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 398*aeb99916SMilanka Ringwald switch (connection->pdu_id) { 399*aeb99916SMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES: 400*aeb99916SMilanka Ringwald bytes_stored = avrcp_store_avctp_now_playing_info_fragment(connection, max_payload_size, packet + pos); 401*aeb99916SMilanka Ringwald 402*aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += bytes_stored + pos; 403*aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, bytes_stored); 404*aeb99916SMilanka Ringwald return; 405*aeb99916SMilanka Ringwald 406*aeb99916SMilanka Ringwald default: 407*aeb99916SMilanka Ringwald break; 408*aeb99916SMilanka Ringwald } 409*aeb99916SMilanka Ringwald break; 410*aeb99916SMilanka Ringwald default: 411*aeb99916SMilanka Ringwald btstack_assert(false); 412*aeb99916SMilanka Ringwald return; 4134b338011SMilanka Ringwald } 4144b338011SMilanka Ringwald 415*aeb99916SMilanka Ringwald // compare number of bytes to store with the remaining buffer size 416*aeb99916SMilanka Ringwald uint16_t bytes_to_copy = btstack_min(connection->data_len - connection->data_offset, max_payload_size - pos); 417*aeb99916SMilanka Ringwald 418*aeb99916SMilanka Ringwald (void)memcpy(packet + pos, &connection->data[connection->data_offset], bytes_to_copy); 419*aeb99916SMilanka Ringwald pos += bytes_to_copy; 420*aeb99916SMilanka Ringwald connection->data_offset += bytes_to_copy; 421*aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent += pos; 422*aeb99916SMilanka Ringwald 423*aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, pos); 424*aeb99916SMilanka Ringwald } 425*aeb99916SMilanka Ringwald 426*aeb99916SMilanka Ringwald static void avrcp_send_reject_cmd_wrong_pid(avrcp_connection_t * connection){ 427*aeb99916SMilanka Ringwald l2cap_reserve_packet_buffer(); 428*aeb99916SMilanka Ringwald uint8_t * packet = l2cap_get_outgoing_buffer(); 429*aeb99916SMilanka Ringwald 430*aeb99916SMilanka Ringwald // AVCTP header 431*aeb99916SMilanka Ringwald // transport header : transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier) 432*aeb99916SMilanka Ringwald packet[0] = (connection->transaction_id << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_RESPONSE_FRAME << 1) | 0; 433*aeb99916SMilanka Ringwald big_endian_store_16(packet, 1, connection->invalid_pid); 434*aeb99916SMilanka Ringwald l2cap_send_prepared(connection->l2cap_signaling_cid, 3); 435*aeb99916SMilanka Ringwald } 436*aeb99916SMilanka Ringwald 437*aeb99916SMilanka Ringwald static void avrcp_target_custome_command_data_init(avrcp_connection_t * connection, 438*aeb99916SMilanka Ringwald avrcp_command_opcode_t opcode, avrcp_command_type_t command_type, 439*aeb99916SMilanka Ringwald avrcp_subunit_type_t subunit_type, avrcp_subunit_id_t subunit_id, 440*aeb99916SMilanka Ringwald avrcp_pdu_id_t pdu_id, uint32_t company_id){ 441*aeb99916SMilanka Ringwald 442*aeb99916SMilanka Ringwald connection->command_opcode = opcode; 4436649facbSMatthias Ringwald connection->command_type = command_type; 44412b7c8ecSMilanka Ringwald connection->subunit_type = subunit_type; 44512b7c8ecSMilanka Ringwald connection->subunit_id = subunit_id; 446*aeb99916SMilanka Ringwald connection->company_id = company_id << 16; 447*aeb99916SMilanka Ringwald connection->pdu_id = pdu_id; 448*aeb99916SMilanka Ringwald connection->data = NULL; 449*aeb99916SMilanka Ringwald connection->data_offset = 0; 450*aeb99916SMilanka Ringwald connection->data_len = 0; 4516649facbSMatthias Ringwald } 4526649facbSMatthias Ringwald 453*aeb99916SMilanka 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){ 454*aeb99916SMilanka Ringwald connection->command_opcode = AVRCP_CMD_OPCODE_VENDOR_DEPENDENT; 455*aeb99916SMilanka Ringwald connection->subunit_type = AVRCP_SUBUNIT_TYPE_PANEL; 456*aeb99916SMilanka Ringwald connection->subunit_id = AVRCP_SUBUNIT_ID; 457*aeb99916SMilanka Ringwald connection->company_id = BT_SIG_COMPANY_ID; 458*aeb99916SMilanka Ringwald 459*aeb99916SMilanka Ringwald connection->command_type = command_type; 460*aeb99916SMilanka Ringwald connection->pdu_id = pdu_id; 461*aeb99916SMilanka Ringwald connection->data = connection->cmd_operands; 462*aeb99916SMilanka Ringwald connection->data_offset = 0; 463*aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 464*aeb99916SMilanka Ringwald connection->state = AVCTP_W2_SEND_AVCTP_FRAGMENTED_MESSAGE; 465*aeb99916SMilanka Ringwald } 466*aeb99916SMilanka Ringwald 467*aeb99916SMilanka 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){ 468*aeb99916SMilanka Ringwald connection->command_opcode = AVRCP_CMD_OPCODE_PASS_THROUGH; 469*aeb99916SMilanka Ringwald connection->subunit_type = AVRCP_SUBUNIT_TYPE_PANEL; 470*aeb99916SMilanka Ringwald connection->subunit_id = AVRCP_SUBUNIT_ID; 471*aeb99916SMilanka Ringwald 472*aeb99916SMilanka Ringwald connection->command_type = command_type; 473*aeb99916SMilanka Ringwald connection->company_id = 0; 474*aeb99916SMilanka Ringwald connection->pdu_id = 0; 475*aeb99916SMilanka Ringwald connection->operation_id = opid; 476*aeb99916SMilanka Ringwald 477*aeb99916SMilanka Ringwald connection->data = connection->cmd_operands; 478*aeb99916SMilanka Ringwald connection->data_offset = 0; 479*aeb99916SMilanka Ringwald connection->data_len = 0; 480*aeb99916SMilanka Ringwald } 481*aeb99916SMilanka Ringwald 482*aeb99916SMilanka Ringwald 483*aeb99916SMilanka Ringwald static uint8_t avrcp_target_vendor_dependent_response_accept(avrcp_connection_t * connection, avrcp_pdu_id_t pdu_id, uint8_t status){ 484*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, pdu_id); 485*aeb99916SMilanka Ringwald connection->data_len = 1; 486*aeb99916SMilanka Ringwald connection->data[0] = status; 487*aeb99916SMilanka Ringwald 4880ec79bd0SMilanka Ringwald connection->accept_response = 1; 48912b7c8ecSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 49012b7c8ecSMilanka Ringwald return ERROR_CODE_SUCCESS; 49112b7c8ecSMilanka Ringwald } 49212b7c8ecSMilanka Ringwald 493*aeb99916SMilanka 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){ 494*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_REJECTED, pdu_id); 495*aeb99916SMilanka Ringwald connection->data_len = 1; 496*aeb99916SMilanka Ringwald connection->data[0] = status; 497*aeb99916SMilanka Ringwald 498d1207cd8SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 499d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 500d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 501d1207cd8SMilanka Ringwald } 502d1207cd8SMilanka Ringwald 503*aeb99916SMilanka 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){ 504*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_NOT_IMPLEMENTED, pdu_id); 505*aeb99916SMilanka Ringwald connection->data_len = 1; 506*aeb99916SMilanka Ringwald connection->data[0] = event_id; 507d1386928SMilanka Ringwald 508d1386928SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 509d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 510d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 511e0bbf3edSMilanka Ringwald } 512e0bbf3edSMilanka Ringwald 513d8133c09SMatthias 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){ 514*aeb99916SMilanka Ringwald btstack_assert(value_len + 1 < AVRCP_MAX_COMMAND_PARAMETER_LENGTH); 515d1207cd8SMilanka Ringwald 516*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_INTERIM, pdu_id); 517*aeb99916SMilanka Ringwald connection->data_len = 1 + value_len; 518*aeb99916SMilanka Ringwald connection->data[0] = event_id; 519*aeb99916SMilanka Ringwald 520c1ab6cc1SMatthias Ringwald if (value && (value_len > 0)){ 521*aeb99916SMilanka Ringwald (void)memcpy(connection->data + 1, value, value_len); 522d1207cd8SMilanka Ringwald } 523*aeb99916SMilanka 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 529*aeb99916SMilanka 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){ 530*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_INTERIM, pdu_id); 5310a2feef4SMilanka Ringwald 532*aeb99916SMilanka Ringwald connection->data_len = 5; 533*aeb99916SMilanka Ringwald connection->data[0] = event_id; 534*aeb99916SMilanka Ringwald big_endian_store_16(connection->data, 1, connection->addressed_player_id); 535*aeb99916SMilanka Ringwald big_endian_store_16(connection->data, 3, connection->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 542*aeb99916SMilanka 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 } 547*aeb99916SMilanka Ringwald avrcp_target_pass_through_command_data_init(connection, ctype, opid); 548831d3fd5SMilanka Ringwald 549831d3fd5SMilanka Ringwald if (operands_length == 1){ 550*aeb99916SMilanka Ringwald connection->data_len = 1; 551*aeb99916SMilanka Ringwald connection->cmd_operands[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 } 576d1207cd8SMilanka Ringwald connection->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 } 586d1207cd8SMilanka Ringwald connection->subunit_info_type = subunit_type; 587d1207cd8SMilanka Ringwald connection->subunit_info_data = subunit_info_data; 588d1207cd8SMilanka Ringwald connection->subunit_info_data_size = subunit_info_data_size; 58961a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 590d1207cd8SMilanka Ringwald } 591d1207cd8SMilanka Ringwald 592*aeb99916SMilanka Ringwald // TODO Review 593d1207cd8SMilanka Ringwald static uint8_t avrcp_target_unit_info(avrcp_connection_t * connection){ 594*aeb99916SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED){ 595*aeb99916SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 596*aeb99916SMilanka Ringwald } 597*aeb99916SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 598*aeb99916SMilanka Ringwald 599*aeb99916SMilanka Ringwald avrcp_target_custome_command_data_init(connection, 600*aeb99916SMilanka Ringwald AVRCP_CMD_OPCODE_UNIT_INFO, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, 601*aeb99916SMilanka Ringwald AVRCP_SUBUNIT_TYPE_UNIT, AVRCP_SUBUNIT_ID_IGNORE, 0, connection->company_id); 6024b338011SMilanka Ringwald 6034b338011SMilanka Ringwald uint8_t unit = 0; 604*aeb99916SMilanka Ringwald connection->data_len = 5; 605*aeb99916SMilanka Ringwald connection->data[0] = 0x07; 606*aeb99916SMilanka Ringwald connection->data[1] = (connection->unit_type << 4) | unit; 6074b338011SMilanka Ringwald // company id is 3 bytes long 608*aeb99916SMilanka Ringwald big_endian_store_24(connection->data, 2, connection->company_id); 609d1386928SMilanka Ringwald 610d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 611d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 6124b338011SMilanka Ringwald } 6134b338011SMilanka Ringwald 614*aeb99916SMilanka Ringwald // TODO Review 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 619*aeb99916SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 620*aeb99916SMilanka Ringwald 621*aeb99916SMilanka Ringwald avrcp_target_custome_command_data_init(connection, AVRCP_CMD_OPCODE_SUBUNIT_INFO, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, 622*aeb99916SMilanka Ringwald AVRCP_SUBUNIT_TYPE_UNIT, AVRCP_SUBUNIT_ID_IGNORE, 0, connection->company_id); 6234b338011SMilanka Ringwald 6244b338011SMilanka Ringwald uint8_t page = offset / 4; 6254b338011SMilanka Ringwald uint8_t extension_code = 7; 626*aeb99916SMilanka Ringwald connection->data_len = 5; 627*aeb99916SMilanka Ringwald connection->data[0] = (page << 4) | extension_code; 628d1207cd8SMilanka Ringwald 629f0af2234SMatthias Ringwald // mark non-existent entries with 0xff 630*aeb99916SMilanka Ringwald memset(&connection->cmd_operands[1], 0xFF, 4); 631*aeb99916SMilanka Ringwald if ((connection->data != NULL) && (offset < connection->subunit_info_data_size)){ 632fda344fcSMatthias Ringwald uint8_t bytes_to_copy = btstack_min(connection->subunit_info_data_size - offset, 4); 633*aeb99916SMilanka Ringwald memcpy(&connection->data[1], &connection->subunit_info_data[offset], bytes_to_copy); 634fda344fcSMatthias Ringwald } 635d1386928SMilanka Ringwald 636d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 637d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 6384b338011SMilanka Ringwald } 6394b338011SMilanka Ringwald 640*aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_supported_events(avrcp_connection_t * connection){ 641*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_CAPABILITIES); 642*aeb99916SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 64393259c07SMilanka Ringwald 64489042326SMilanka Ringwald uint8_t event_id; 645*aeb99916SMilanka Ringwald uint8_t num_events = 0; 64689042326SMilanka Ringwald for (event_id = (uint8_t) AVRCP_NOTIFICATION_EVENT_FIRST_INDEX; event_id < (uint8_t) AVRCP_NOTIFICATION_EVENT_LAST_INDEX; event_id++){ 64789042326SMilanka Ringwald if ((connection->target_supported_notifications & (1<<event_id)) == 0){ 64889042326SMilanka Ringwald continue; 64989042326SMilanka Ringwald } 65089042326SMilanka Ringwald num_events++; 65189042326SMilanka Ringwald } 65289042326SMilanka Ringwald 653*aeb99916SMilanka Ringwald connection->data[0] = AVRCP_CAPABILITY_ID_EVENT; 654*aeb99916SMilanka Ringwald connection->data[1] = num_events; 655*aeb99916SMilanka Ringwald connection->data_len = 2 + num_events; 656e0bbf3edSMilanka Ringwald 657*aeb99916SMilanka Ringwald // fill the data later directly to the L2CAP outgoing buffer 658d1386928SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 659d1386928SMilanka Ringwald return ERROR_CODE_SUCCESS; 660e0bbf3edSMilanka Ringwald } 661e0bbf3edSMilanka Ringwald 662*aeb99916SMilanka Ringwald static uint8_t avrcp_target_response_vendor_dependent_supported_companies(avrcp_connection_t * connection){ 663*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_CAPABILITIES); 66489042326SMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 665*aeb99916SMilanka Ringwald 666*aeb99916SMilanka Ringwald connection->data[0] = AVRCP_CAPABILITY_ID_COMPANY; 667*aeb99916SMilanka Ringwald connection->data[1] = connection->target_supported_companies_num; 668*aeb99916SMilanka Ringwald connection->data_len = 2 + connection->target_supported_companies_num * 3; 669*aeb99916SMilanka Ringwald 670*aeb99916SMilanka Ringwald // fill the data later directly to the L2CAP outgoing buffer and 671*aeb99916SMilanka Ringwald // use Bluetooth SIG as default company 67289042326SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 67389042326SMilanka Ringwald return ERROR_CODE_SUCCESS; 67489042326SMilanka Ringwald } 67589042326SMilanka Ringwald 6760346f11dSMilanka Ringwald uint8_t avrcp_target_support_event(uint16_t avrcp_cid, avrcp_notification_event_id_t event_id){ 6770346f11dSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 6780346f11dSMilanka Ringwald if (!connection){ 6790346f11dSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 68089042326SMilanka Ringwald } 68189042326SMilanka Ringwald 6820346f11dSMilanka Ringwald if ((event_id < (uint8_t)AVRCP_NOTIFICATION_EVENT_FIRST_INDEX) || (event_id > (uint8_t)AVRCP_NOTIFICATION_EVENT_LAST_INDEX)){ 6830346f11dSMilanka Ringwald return ERROR_CODE_UNSUPPORTED_FEATURE_OR_PARAMETER_VALUE; 6840346f11dSMilanka Ringwald } 6850346f11dSMilanka Ringwald 6860346f11dSMilanka Ringwald connection->target_supported_notifications |= (1 << (uint8_t)event_id); 6870346f11dSMilanka Ringwald return ERROR_CODE_SUCCESS; 6880346f11dSMilanka Ringwald } 6890346f11dSMilanka Ringwald 6900346f11dSMilanka Ringwald uint8_t avrcp_target_support_companies(uint16_t avrcp_cid, uint8_t num_companies, const uint32_t *companies){ 6910346f11dSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 6920346f11dSMilanka Ringwald if (!connection){ 6930346f11dSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 6940346f11dSMilanka Ringwald } 6950346f11dSMilanka Ringwald 6960346f11dSMilanka Ringwald connection->target_supported_companies_num = num_companies; 6970346f11dSMilanka Ringwald connection->target_supported_companies = companies; 6980346f11dSMilanka Ringwald return ERROR_CODE_SUCCESS; 6995e3d4d2bSMilanka Ringwald } 7005e3d4d2bSMilanka Ringwald 701*aeb99916SMilanka Ringwald // TODO Review (use flags) 702d1207cd8SMilanka 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){ 703*aeb99916SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 704*aeb99916SMilanka Ringwald if (!connection){ 705*aeb99916SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 706*aeb99916SMilanka Ringwald } 707*aeb99916SMilanka Ringwald if (connection->state != AVCTP_CONNECTION_OPENED){ 708*aeb99916SMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 7095e3d4d2bSMilanka Ringwald } 7105e3d4d2bSMilanka Ringwald 711*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_IMPLEMENTED_STABLE, AVRCP_PDU_ID_GET_PLAY_STATUS); 712*aeb99916SMilanka Ringwald connection->data_len = 9; 713*aeb99916SMilanka Ringwald big_endian_store_32(connection->data, 0, song_length_ms); 714*aeb99916SMilanka Ringwald big_endian_store_32(connection->data, 4, song_position_ms); 715*aeb99916SMilanka Ringwald connection->data[8] = play_status; 7165e3d4d2bSMilanka Ringwald 7175e3d4d2bSMilanka Ringwald connection->state = AVCTP_W2_SEND_RESPONSE; 7185e3d4d2bSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7195e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7205e3d4d2bSMilanka Ringwald } 7215e3d4d2bSMilanka Ringwald 722d1207cd8SMilanka Ringwald static uint8_t avrcp_target_store_media_attr(avrcp_connection_t * connection, avrcp_media_attribute_id_t attr_id, const char * value){ 7235e3d4d2bSMilanka Ringwald int index = attr_id - 1; 724d1207cd8SMilanka Ringwald if (!value) return AVRCP_STATUS_INVALID_PARAMETER; 7255e3d4d2bSMilanka Ringwald connection->now_playing_info[index].value = (uint8_t*)value; 7265e3d4d2bSMilanka Ringwald connection->now_playing_info[index].len = strlen(value); 7275e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7285e3d4d2bSMilanka Ringwald } 7295e3d4d2bSMilanka Ringwald 730d1207cd8SMilanka Ringwald uint8_t avrcp_target_set_playback_status(uint16_t avrcp_cid, avrcp_playback_status_t playback_status){ 7311945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7325e3d4d2bSMilanka Ringwald if (!connection){ 7335e3d4d2bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7345e3d4d2bSMilanka Ringwald } 73561a3437bSMilanka Ringwald if (connection->playback_status == playback_status){ 73661a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 73761a3437bSMilanka Ringwald } 738d1207cd8SMilanka Ringwald 739d1207cd8SMilanka Ringwald connection->playback_status = playback_status; 7404b5213b0SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED)) { 741d1207cd8SMilanka Ringwald connection->playback_status_changed = 1; 742d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 7434b5213b0SMilanka Ringwald } 7445e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7455e3d4d2bSMilanka Ringwald } 7465e3d4d2bSMilanka Ringwald 74761a3437bSMilanka Ringwald uint8_t avrcp_target_set_now_playing_info(uint16_t avrcp_cid, const avrcp_track_t * current_track, uint16_t total_tracks){ 7481945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7495e3d4d2bSMilanka Ringwald if (!connection){ 75061a3437bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7515e3d4d2bSMilanka Ringwald } 752d1207cd8SMilanka Ringwald if (!current_track){ 753d1207cd8SMilanka Ringwald connection->track_selected = 0; 754d1207cd8SMilanka Ringwald connection->playback_status = AVRCP_PLAYBACK_STATUS_ERROR; 75561a3437bSMilanka Ringwald return ERROR_CODE_COMMAND_DISALLOWED; 756d1207cd8SMilanka Ringwald } 75761a3437bSMilanka Ringwald 7586535961aSMatthias Ringwald (void)memcpy(connection->track_id, current_track->track_id, 8); 759d1207cd8SMilanka Ringwald connection->song_length_ms = current_track->song_length_ms; 760d1207cd8SMilanka Ringwald connection->track_nr = current_track->track_nr; 7615e3d4d2bSMilanka Ringwald connection->total_tracks = total_tracks; 762d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_TITLE, current_track->title); 763d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_ARTIST, current_track->artist); 764d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_ALBUM, current_track->album); 765d1207cd8SMilanka Ringwald avrcp_target_store_media_attr(connection, AVRCP_MEDIA_ATTR_GENRE, current_track->genre); 766*aeb99916SMilanka Ringwald 767d1207cd8SMilanka Ringwald connection->track_selected = 1; 768d1207cd8SMilanka Ringwald 769d1207cd8SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED)) { 770d1207cd8SMilanka Ringwald connection->track_changed = 1; 771d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 772d1207cd8SMilanka Ringwald } 77361a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 7745e3d4d2bSMilanka Ringwald } 7755e3d4d2bSMilanka Ringwald 77612b7c8ecSMilanka Ringwald uint8_t avrcp_target_track_changed(uint16_t avrcp_cid, uint8_t * track_id){ 7771945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 77812b7c8ecSMilanka Ringwald if (!connection){ 77912b7c8ecSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 78012b7c8ecSMilanka Ringwald } 78161a3437bSMilanka Ringwald if (!track_id){ 78261a3437bSMilanka Ringwald return ERROR_CODE_UNSUPPORTED_FEATURE_OR_PARAMETER_VALUE; 78361a3437bSMilanka Ringwald } 78412b7c8ecSMilanka Ringwald 785db818650SMilanka Ringwald (void)memcpy(connection->track_id, track_id, 8); 78612b7c8ecSMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED)) { 78712b7c8ecSMilanka Ringwald connection->track_changed = 1; 78812b7c8ecSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 78912b7c8ecSMilanka Ringwald } 79012b7c8ecSMilanka Ringwald return ERROR_CODE_SUCCESS; 79112b7c8ecSMilanka Ringwald } 79212b7c8ecSMilanka Ringwald 793d1207cd8SMilanka Ringwald uint8_t avrcp_target_playing_content_changed(uint16_t avrcp_cid){ 7941945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 7955e3d4d2bSMilanka Ringwald if (!connection){ 7965e3d4d2bSMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 7975e3d4d2bSMilanka Ringwald } 79812b7c8ecSMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED)) { 799d1207cd8SMilanka Ringwald connection->playing_content_changed = 1; 800d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 80112b7c8ecSMilanka Ringwald } 8025e3d4d2bSMilanka Ringwald return ERROR_CODE_SUCCESS; 8035e3d4d2bSMilanka Ringwald } 804e0bbf3edSMilanka Ringwald 8056568eb54SMilanka Ringwald uint8_t avrcp_target_addressed_player_changed(uint16_t avrcp_cid, uint16_t player_id, uint16_t uid_counter){ 8061945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 8076568eb54SMilanka Ringwald if (!connection){ 8086568eb54SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 8096568eb54SMilanka Ringwald } 81061a3437bSMilanka Ringwald 81161a3437bSMilanka Ringwald if (connection->addressed_player_id == player_id){ 81261a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 81361a3437bSMilanka Ringwald } 814db818650SMilanka Ringwald 8158a518194SMilanka Ringwald connection->uid_counter = uid_counter; 8168a518194SMilanka Ringwald connection->addressed_player_id = player_id; 817db818650SMilanka Ringwald 818db818650SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED)) { 819db818650SMilanka Ringwald connection->addressed_player_changed = 1; 820db818650SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 8216568eb54SMilanka Ringwald } 8226568eb54SMilanka Ringwald return ERROR_CODE_SUCCESS; 8236568eb54SMilanka Ringwald } 8246568eb54SMilanka Ringwald 825d1207cd8SMilanka Ringwald uint8_t avrcp_target_battery_status_changed(uint16_t avrcp_cid, avrcp_battery_status_t battery_status){ 8261945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 827d1207cd8SMilanka Ringwald if (!connection){ 828d1207cd8SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 829d1207cd8SMilanka Ringwald } 83061a3437bSMilanka Ringwald if (connection->battery_status == battery_status){ 83161a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 83261a3437bSMilanka Ringwald } 833db818650SMilanka Ringwald 834d1207cd8SMilanka Ringwald connection->battery_status = battery_status; 835db818650SMilanka Ringwald 836db818650SMilanka Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED)) { 837d1207cd8SMilanka Ringwald connection->battery_status_changed = 1; 838d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 83912b7c8ecSMilanka Ringwald } 840d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 841d1207cd8SMilanka Ringwald } 842d1207cd8SMilanka Ringwald 843b772a0d7SMilanka Ringwald uint8_t avrcp_target_adjust_absolute_volume(uint16_t avrcp_cid, uint8_t absolute_volume){ 844b772a0d7SMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 845b772a0d7SMilanka Ringwald if (!connection){ 846b772a0d7SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 847b772a0d7SMilanka Ringwald } 848b772a0d7SMilanka Ringwald 849b772a0d7SMilanka Ringwald connection->absolute_volume = absolute_volume; 850b772a0d7SMilanka Ringwald return ERROR_CODE_SUCCESS; 851b772a0d7SMilanka Ringwald } 852b772a0d7SMilanka Ringwald 853984bbc3eSMilanka Ringwald uint8_t avrcp_target_volume_changed(uint16_t avrcp_cid, uint8_t absolute_volume){ 8541945fe3eSMilanka Ringwald avrcp_connection_t * connection = avrcp_get_connection_for_avrcp_cid_for_role(AVRCP_TARGET, avrcp_cid); 855d1207cd8SMilanka Ringwald if (!connection){ 856d1207cd8SMilanka Ringwald return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER; 857d1207cd8SMilanka Ringwald } 858984bbc3eSMilanka Ringwald if (connection->absolute_volume == absolute_volume){ 85961a3437bSMilanka Ringwald return ERROR_CODE_SUCCESS; 86061a3437bSMilanka Ringwald } 86161a3437bSMilanka Ringwald 862984bbc3eSMilanka Ringwald connection->absolute_volume = absolute_volume; 86361a3437bSMilanka Ringwald 8643e5c3c5bSMatthias Ringwald if (connection->notifications_enabled & (1 << AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED )) { 865984bbc3eSMilanka Ringwald connection->notify_absolute_volume_changed = 1; 866d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 86712b7c8ecSMilanka Ringwald } 868d1207cd8SMilanka Ringwald return ERROR_CODE_SUCCESS; 869d1207cd8SMilanka Ringwald } 870d1207cd8SMilanka Ringwald 8710ec79bd0SMilanka Ringwald static void avrcp_target_set_transaction_label_for_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification, uint8_t transaction_label){ 872fe934656SMatthias Ringwald if (notification > AVRCP_NOTIFICATION_EVENT_MAX_VALUE) return; 8730ec79bd0SMilanka Ringwald connection->notifications_transaction_label[notification] = transaction_label; 8740ec79bd0SMilanka Ringwald } 8750ec79bd0SMilanka Ringwald 8760ec79bd0SMilanka Ringwald static uint8_t avrcp_target_get_transaction_label_for_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification){ 877fe934656SMatthias Ringwald if (notification > AVRCP_NOTIFICATION_EVENT_MAX_VALUE) return 0; 8780ec79bd0SMilanka Ringwald return connection->notifications_transaction_label[notification]; 8790ec79bd0SMilanka Ringwald } 880d1207cd8SMilanka Ringwald 8810b578d06SMilanka Ringwald static bool avcrp_operation_id_is_valid(avrcp_operation_id_t operation_id){ 8820b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_1) return true; 8830b578d06SMilanka Ringwald 8840b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_0) return false; 8850b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_2) return true; 8860b578d06SMilanka Ringwald 8870b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_CHANNEL_UP) return false; 8880b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_3) return true; 8890b578d06SMilanka Ringwald 8900b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_CHANNEL_UP) return false; 8910b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_3) return true; 8920b578d06SMilanka Ringwald 8930b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_SKIP) return false; 8940b578d06SMilanka Ringwald if (operation_id == AVRCP_OPERATION_ID_SKIP) return true; 8950b578d06SMilanka Ringwald 8960b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_POWER) return false; 8970b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_4) return true; 8980b578d06SMilanka Ringwald 8990b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_ANGLE) return false; 9000b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_5) return true; 9010b578d06SMilanka Ringwald 9020b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_F1) return false; 9030b578d06SMilanka Ringwald if (operation_id < AVRCP_OPERATION_ID_RESERVED_6) return true; 9040b578d06SMilanka Ringwald 9050b578d06SMilanka Ringwald return false; 9060b578d06SMilanka Ringwald } 9070b578d06SMilanka Ringwald 908*aeb99916SMilanka Ringwald 909*aeb99916SMilanka Ringwald #ifdef ENABLE_AVCTP_FRAGMENTATION 910*aeb99916SMilanka Ringwald static void avctp_reassemble_message(avrcp_connection_t * connection, avctp_packet_type_t packet_type, uint8_t *packet, uint16_t size){ 911*aeb99916SMilanka Ringwald // after header (transaction label and packet type) 912*aeb99916SMilanka Ringwald uint16_t pos; 913*aeb99916SMilanka Ringwald uint16_t bytes_to_store; 914*aeb99916SMilanka Ringwald 915*aeb99916SMilanka Ringwald switch (packet_type){ 916*aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 917*aeb99916SMilanka Ringwald if (size < 2) return; 918*aeb99916SMilanka Ringwald 919*aeb99916SMilanka Ringwald // store header 920*aeb99916SMilanka Ringwald pos = 0; 921*aeb99916SMilanka Ringwald connection->avctp_reassembly_buffer[pos] = packet[pos]; 922*aeb99916SMilanka Ringwald pos++; 923*aeb99916SMilanka Ringwald connection->avctp_reassembly_size = pos; 924*aeb99916SMilanka Ringwald 925*aeb99916SMilanka Ringwald // NOTE: num packets not needed for reassembly, ignoring it does not pose security risk -> no need to store it 926*aeb99916SMilanka Ringwald pos++; 927*aeb99916SMilanka Ringwald 928*aeb99916SMilanka 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 929*aeb99916SMilanka Ringwald 930*aeb99916SMilanka Ringwald bytes_to_store = btstack_min(size - pos, sizeof(connection->avctp_reassembly_buffer) - connection->avctp_reassembly_size); 931*aeb99916SMilanka Ringwald memcpy(&connection->avctp_reassembly_buffer[connection->avctp_reassembly_size], &packet[pos], bytes_to_store); 932*aeb99916SMilanka Ringwald connection->avctp_reassembly_size += bytes_to_store; 933*aeb99916SMilanka Ringwald break; 934*aeb99916SMilanka Ringwald 935*aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 936*aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 937*aeb99916SMilanka Ringwald if (size < 1) return; 938*aeb99916SMilanka Ringwald 939*aeb99916SMilanka Ringwald // store remaining data, ignore header 940*aeb99916SMilanka Ringwald pos = 1; 941*aeb99916SMilanka Ringwald bytes_to_store = btstack_min(size - pos, sizeof(connection->avctp_reassembly_buffer) - connection->avctp_reassembly_size); 942*aeb99916SMilanka Ringwald memcpy(&connection->avctp_reassembly_buffer[connection->avctp_reassembly_size], &packet[pos], bytes_to_store); 943*aeb99916SMilanka Ringwald connection->avctp_reassembly_size += bytes_to_store; 944*aeb99916SMilanka Ringwald break; 945*aeb99916SMilanka Ringwald 946*aeb99916SMilanka Ringwald default: 947*aeb99916SMilanka Ringwald return; 948*aeb99916SMilanka Ringwald } 949*aeb99916SMilanka Ringwald } 950*aeb99916SMilanka Ringwald #endif 951*aeb99916SMilanka Ringwald 95201dc6e35SMilanka Ringwald static void avrcp_handle_l2cap_data_packet_for_signaling_connection(avrcp_connection_t * connection, uint8_t * packet, uint16_t size){ 953*aeb99916SMilanka Ringwald uint8_t avctp_header = packet[0]; 954*aeb99916SMilanka Ringwald connection->transaction_id = avctp_header >> 4; 955*aeb99916SMilanka Ringwald 956*aeb99916SMilanka Ringwald avctp_packet_type_t avctp_packet_type = (avctp_packet_type_t) ((avctp_header & 0x0F) >> 2); 957*aeb99916SMilanka Ringwald switch (avctp_packet_type){ 958*aeb99916SMilanka Ringwald case AVCTP_SINGLE_PACKET: 959*aeb99916SMilanka Ringwald break; 960*aeb99916SMilanka Ringwald 961*aeb99916SMilanka Ringwald #ifdef ENABLE_AVCTP_FRAGMENTATION 962*aeb99916SMilanka Ringwald case AVCTP_START_PACKET: 963*aeb99916SMilanka Ringwald case AVCTP_CONTINUE_PACKET: 964*aeb99916SMilanka Ringwald avctp_reassemble_message(connection, avctp_packet_type, packet, size); 965*aeb99916SMilanka Ringwald return; 966*aeb99916SMilanka Ringwald 967*aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 968*aeb99916SMilanka Ringwald avctp_reassemble_message(connection, avctp_packet_type, packet, size); 969*aeb99916SMilanka Ringwald 970*aeb99916SMilanka Ringwald packet = connection->avctp_reassembly_buffer; 971*aeb99916SMilanka Ringwald size = connection->avctp_reassembly_size; 972*aeb99916SMilanka Ringwald break; 973*aeb99916SMilanka Ringwald #endif 974*aeb99916SMilanka Ringwald default: 975*aeb99916SMilanka Ringwald return; 976*aeb99916SMilanka Ringwald } 977697b823eSMatthias Ringwald 978697b823eSMatthias Ringwald if (size < 6u) return; 9794b338011SMilanka Ringwald 980*aeb99916SMilanka Ringwald uint16_t pid = big_endian_read_16(packet, 1); 981be4cc80aSMilanka Ringwald 98267049435SMilanka Ringwald if (pid != BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL){ 983355a220fSMilanka Ringwald log_info("Invalid pid 0x%02x, expected 0x%02x", connection->invalid_pid, BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL); 98467049435SMilanka Ringwald connection->reject_transport_header = 1; 985355a220fSMilanka Ringwald connection->invalid_pid = pid; 986022b77fcSMilanka Ringwald connection->transport_header = (connection->transaction_id << 4) | (AVRCP_SINGLE_PACKET << 2 ) | (AVRCP_RESPONSE_FRAME << 1) | 1; 98767049435SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 98867049435SMilanka Ringwald return; 98967049435SMilanka Ringwald } 99067049435SMilanka Ringwald 991*aeb99916SMilanka Ringwald // avrcp_subunit_type_t subunit_type = (avrcp_subunit_type_t) (packet[4] >> 3); 992*aeb99916SMilanka Ringwald // avrcp_subunit_id_t subunit_id = (avrcp_subunit_id_t) (packet[4] & 0x07); 9934b338011SMilanka Ringwald 994dfbbc26bSMatthias Ringwald avrcp_command_opcode_t opcode = (avrcp_command_opcode_t) avrcp_cmd_opcode(packet,size); 995e0bbf3edSMilanka Ringwald 996e8ad6d13SMatthias Ringwald int pos = 6; 997dfbbc26bSMatthias Ringwald uint16_t length; 998dfbbc26bSMatthias Ringwald avrcp_pdu_id_t pdu_id; 999*aeb99916SMilanka Ringwald // connection->data_len = 0; 10002c197cf1SMilanka Ringwald uint8_t offset; 10010b578d06SMilanka Ringwald uint8_t operand; 1002367e3c55SMilanka Ringwald uint16_t event_mask; 100338410698SMilanka Ringwald avrcp_operation_id_t operation_id; 1004e0bbf3edSMilanka Ringwald 1005e0bbf3edSMilanka Ringwald switch (opcode){ 10064b338011SMilanka Ringwald case AVRCP_CMD_OPCODE_UNIT_INFO: 1007d1207cd8SMilanka Ringwald avrcp_target_unit_info(connection); 10084b338011SMilanka Ringwald break; 10092c197cf1SMilanka Ringwald case AVRCP_CMD_OPCODE_SUBUNIT_INFO: 101099702381SMatthias Ringwald if ((size - pos) < 3) return; 10112c197cf1SMilanka Ringwald // page: packet[pos] >> 4, 10122c197cf1SMilanka Ringwald offset = 4 * (packet[pos]>>4); 10132c197cf1SMilanka Ringwald // extension code (fixed 7) = packet[pos] & 0x0F 10142c197cf1SMilanka Ringwald // 4 bytes paga data, all 0xFF 1015d1207cd8SMilanka Ringwald avrcp_target_subunit_info(connection, offset); 1016e0bbf3edSMilanka Ringwald break; 10172c197cf1SMilanka Ringwald 10180b578d06SMilanka Ringwald case AVRCP_CMD_OPCODE_PASS_THROUGH: 1019810b0d84SMatthias Ringwald if (size < 8) return; 1020810b0d84SMatthias Ringwald log_info("AVRCP_OPERATION_ID 0x%02x, operands length %d", packet[6], packet[7]); 102138410698SMilanka Ringwald operation_id = (avrcp_operation_id_t) (packet[6] & 0x7f); 10220b578d06SMilanka Ringwald operand = 0; 1023810b0d84SMatthias Ringwald if ((packet[7] >= 1) && (size >= 9)){ 1024810b0d84SMatthias Ringwald operand = packet[8]; 1025810b0d84SMatthias Ringwald } 1026d1207cd8SMilanka Ringwald 10270b578d06SMilanka Ringwald if (avcrp_operation_id_is_valid(operation_id)){ 10280b578d06SMilanka Ringwald bool button_pressed = (packet[6] & 0x80) == 0; 10290b578d06SMilanka Ringwald 1030810b0d84SMatthias Ringwald avrcp_target_operation_accepted(connection->avrcp_cid, (avrcp_operation_id_t) packet[6], packet[7], operand); 1031319131f8SMatthias Ringwald avrcp_target_emit_operation(avrcp_target_context.avrcp_callback, connection->avrcp_cid, 1032927fd23fSMatthias Ringwald operation_id, button_pressed, packet[7], operand); 10330b578d06SMilanka Ringwald } else { 1034810b0d84SMatthias Ringwald avrcp_target_operation_not_implemented(connection->avrcp_cid, (avrcp_operation_id_t) packet[6], packet[7], operand); 1035d1207cd8SMilanka Ringwald } 1036d1207cd8SMilanka Ringwald break; 10370b578d06SMilanka Ringwald 1038d1207cd8SMilanka Ringwald 1039e0bbf3edSMilanka Ringwald case AVRCP_CMD_OPCODE_VENDOR_DEPENDENT: 104099702381SMatthias Ringwald 104199702381SMatthias Ringwald if (size < 13) return; 104299702381SMatthias Ringwald 104399702381SMatthias Ringwald // pos = 6 - company id 104499702381SMatthias Ringwald (void)memcpy(connection->cmd_operands, &packet[pos], 3); 1045*aeb99916SMilanka Ringwald // connection->data_len = 3; 104699702381SMatthias Ringwald pos += 3; 104799702381SMatthias Ringwald // pos = 9 104899702381SMatthias Ringwald pdu_id = (avrcp_pdu_id_t) packet[pos++]; 104999702381SMatthias Ringwald // 1 - reserved 105099702381SMatthias Ringwald pos++; 105199702381SMatthias Ringwald // 2-3 param length, 105299702381SMatthias Ringwald length = big_endian_read_16(packet, pos); 105399702381SMatthias Ringwald pos += 2; 105499702381SMatthias Ringwald // pos = 13 1055e0bbf3edSMilanka Ringwald switch (pdu_id){ 10566568eb54SMilanka Ringwald case AVRCP_PDU_ID_SET_ADDRESSED_PLAYER:{ 105799702381SMatthias Ringwald if ((pos + 2) > size) return; 1058d1cf25b1SMatthias Ringwald bool ok = length == 4; 1059d1cf25b1SMatthias Ringwald if (avrcp_target_context.set_addressed_player_callback != NULL){ 106099702381SMatthias Ringwald uint16_t player_id = big_endian_read_16(packet, pos); 1061d1cf25b1SMatthias Ringwald ok = avrcp_target_context.set_addressed_player_callback(player_id); 10626568eb54SMilanka Ringwald } 1063d1cf25b1SMatthias Ringwald if (ok){ 1064*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, AVRCP_STATUS_SUCCESS); 1065d1cf25b1SMatthias Ringwald } else { 1066*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PLAYER_ID); 1067d1cf25b1SMatthias Ringwald } 10686568eb54SMilanka Ringwald break; 10696568eb54SMilanka Ringwald } 1070e0bbf3edSMilanka Ringwald case AVRCP_PDU_ID_GET_CAPABILITIES:{ 107199702381SMatthias Ringwald avrcp_capability_id_t capability_id = (avrcp_capability_id_t) packet[pos]; 1072e0bbf3edSMilanka Ringwald switch (capability_id){ 1073e0bbf3edSMilanka Ringwald case AVRCP_CAPABILITY_ID_EVENT: 1074*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_supported_events(connection); 1075e0bbf3edSMilanka Ringwald break; 1076e0bbf3edSMilanka Ringwald case AVRCP_CAPABILITY_ID_COMPANY: 1077*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_supported_companies(connection); 1078e0bbf3edSMilanka Ringwald break; 1079e0bbf3edSMilanka Ringwald default: 1080*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 1081e0bbf3edSMilanka Ringwald break; 1082e0bbf3edSMilanka Ringwald } 1083e0bbf3edSMilanka Ringwald break; 1084e0bbf3edSMilanka Ringwald } 1085c045af99SMilanka Ringwald case AVRCP_PDU_ID_GET_PLAY_STATUS: 1086c045af99SMilanka Ringwald avrcp_target_emit_respond_vendor_dependent_query(avrcp_target_context.avrcp_callback, connection->avrcp_cid, AVRCP_SUBEVENT_PLAY_STATUS_QUERY); 1087c045af99SMilanka Ringwald break; 10881bf7a74fSMilanka Ringwald case AVRCP_PDU_ID_REQUEST_ABORT_CONTINUING_RESPONSE: 108999702381SMatthias Ringwald if ((pos + 1) > size) return; 109099702381SMatthias Ringwald connection->cmd_operands[0] = packet[pos]; 10911bf7a74fSMilanka Ringwald connection->abort_continue_response = 1; 10921bf7a74fSMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 10931bf7a74fSMilanka Ringwald break; 1094d1207cd8SMilanka Ringwald case AVRCP_PDU_ID_REQUEST_CONTINUING_RESPONSE: 109599702381SMatthias Ringwald if ((pos + 1) > size) return; 109699702381SMatthias Ringwald if (packet[pos] != AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES){ 1097*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 1098d1207cd8SMilanka Ringwald return; 1099d1207cd8SMilanka Ringwald } 1100a1946e79SMilanka Ringwald connection->now_playing_info_response = true; 1101d1207cd8SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1102d1207cd8SMilanka Ringwald break; 11035e3d4d2bSMilanka Ringwald case AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES:{ 110499702381SMatthias Ringwald if ((pos + 9) > size) return; 11055e3d4d2bSMilanka Ringwald uint8_t play_identifier[8]; 11065e3d4d2bSMilanka Ringwald memset(play_identifier, 0, 8); 110799702381SMatthias Ringwald if (memcmp(&packet[pos], play_identifier, 8) != 0) { 1108*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 11095e3d4d2bSMilanka Ringwald return; 11105e3d4d2bSMilanka Ringwald } 1111d1207cd8SMilanka Ringwald pos += 8; 111299702381SMatthias Ringwald uint8_t attribute_count = packet[pos++]; 1113d1207cd8SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_NONE; 1114d1207cd8SMilanka Ringwald if (!attribute_count){ 1115*aeb99916SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_TITLE; 1116d1207cd8SMilanka Ringwald connection->now_playing_info_attr_bitmap = 0xFE; 1117d1207cd8SMilanka Ringwald } else { 1118d1207cd8SMilanka Ringwald int i; 1119*aeb99916SMilanka Ringwald connection->next_attr_id = AVRCP_MEDIA_ATTR_TITLE; 1120d1207cd8SMilanka Ringwald connection->now_playing_info_attr_bitmap = 0; 1121*aeb99916SMilanka Ringwald connection->num_attributes = 0; 1122*aeb99916SMilanka Ringwald if ((pos + attribute_count * 4) > size) return; 1123d1207cd8SMilanka Ringwald for (i=0; i < attribute_count; i++){ 1124*aeb99916SMilanka Ringwald uint32_t attr_id = big_endian_read_32(packet, pos); 1125d1207cd8SMilanka Ringwald connection->now_playing_info_attr_bitmap |= (1 << attr_id); 1126*aeb99916SMilanka Ringwald pos += 4; 1127d1207cd8SMilanka Ringwald } 1128d1207cd8SMilanka Ringwald } 1129d1207cd8SMilanka Ringwald log_info("now_playing_info_attr_bitmap 0x%02x", connection->now_playing_info_attr_bitmap); 1130*aeb99916SMilanka Ringwald connection->num_attributes = count_set_bits_uint32(connection->now_playing_info_attr_bitmap); 1131*aeb99916SMilanka Ringwald connection->now_playing_info_response = true; 1132*aeb99916SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1133d1207cd8SMilanka Ringwald break; 1134d1207cd8SMilanka Ringwald } 1135d1207cd8SMilanka Ringwald case AVRCP_PDU_ID_REGISTER_NOTIFICATION:{ 113699702381SMatthias Ringwald if ((pos + 1) > size) return; 113799702381SMatthias Ringwald avrcp_notification_event_id_t event_id = (avrcp_notification_event_id_t) packet[pos]; 1138367e3c55SMilanka Ringwald 1139022b77fcSMilanka Ringwald avrcp_target_set_transaction_label_for_notification(connection, event_id, connection->transaction_id); 1140d1207cd8SMilanka Ringwald 1141367e3c55SMilanka Ringwald if (event_id < AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED || 1142367e3c55SMilanka Ringwald event_id > AVRCP_NOTIFICATION_EVENT_MAX_VALUE){ 1143*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_PARAMETER); 1144367e3c55SMilanka Ringwald return; 1145367e3c55SMilanka Ringwald } 1146367e3c55SMilanka Ringwald 1147367e3c55SMilanka Ringwald switch (event_id){ 1148367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_AVAILABLE_PLAYERS_CHANGED: 1149367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYER_APPLICATION_SETTING_CHANGED: 1150367e3c55SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_UIDS_CHANGED: 11514b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_REACHED_END: 11524b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_REACHED_START: 11534b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYBACK_POS_CHANGED: 11544b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_SYSTEM_STATUS_CHANGED: 11554b5213b0SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_MAX_VALUE: 1156*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_not_implemented(connection, pdu_id, event_id); 1157367e3c55SMilanka Ringwald return; 1158367e3c55SMilanka Ringwald default: 1159367e3c55SMilanka Ringwald break; 1160367e3c55SMilanka Ringwald } 1161367e3c55SMilanka Ringwald 1162367e3c55SMilanka Ringwald event_mask = (1 << event_id); 1163367e3c55SMilanka Ringwald connection->notifications_enabled |= event_mask; 1164367e3c55SMilanka Ringwald 1165d1207cd8SMilanka Ringwald switch (event_id){ 1166d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED: 1167d1207cd8SMilanka Ringwald if (connection->track_selected){ 1168d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, AVRCP_NOTIFICATION_TRACK_SELECTED, 8); 1169d1207cd8SMilanka Ringwald } else { 1170d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, AVRCP_NOTIFICATION_TRACK_NOT_SELECTED, 8); 1171d1207cd8SMilanka Ringwald } 1172d1207cd8SMilanka Ringwald break; 1173d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED: 1174d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->playback_status, 1); 1175d1207cd8SMilanka Ringwald break; 1176d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED: 1177d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, NULL, 0); 1178d1207cd8SMilanka Ringwald break; 1179d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED: 1180984bbc3eSMilanka Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->absolute_volume, 1); 1181d1207cd8SMilanka Ringwald break; 1182d1207cd8SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED: 1183d8133c09SMatthias Ringwald avrcp_target_response_vendor_dependent_interim(connection, pdu_id, event_id, (const uint8_t *)&connection->battery_status, 1); 1184d1207cd8SMilanka Ringwald break; 11856568eb54SMilanka Ringwald case AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED: 1186*aeb99916SMilanka Ringwald avrcp_target_response_addressed_player_changed_interim(connection, pdu_id, event_id); 11876568eb54SMilanka Ringwald return; 1188d1207cd8SMilanka Ringwald default: 1189367e3c55SMilanka Ringwald btstack_assert(false); 1190d1207cd8SMilanka Ringwald return; 1191d1207cd8SMilanka Ringwald } 11925e3d4d2bSMilanka Ringwald break; 11935e3d4d2bSMilanka Ringwald } 119412b7c8ecSMilanka Ringwald case AVRCP_PDU_ID_SET_ABSOLUTE_VOLUME: { 1195984bbc3eSMilanka Ringwald if ( (length != 1) || ((pos + 1) > size)){ 1196*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 119712b7c8ecSMilanka Ringwald break; 119812b7c8ecSMilanka Ringwald } 119912b7c8ecSMilanka Ringwald 120099702381SMatthias Ringwald uint8_t absolute_volume = packet[pos]; 12010a2feef4SMilanka Ringwald if (absolute_volume < 0x80){ 1202984bbc3eSMilanka Ringwald connection->absolute_volume = absolute_volume; 12030a2feef4SMilanka Ringwald } 1204984bbc3eSMilanka Ringwald avrcp_target_emit_volume_changed(avrcp_target_context.avrcp_callback, connection->avrcp_cid, connection->absolute_volume); 1205*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_accept(connection, pdu_id, connection->absolute_volume); 120612b7c8ecSMilanka Ringwald break; 120712b7c8ecSMilanka Ringwald } 1208e0bbf3edSMilanka Ringwald default: 12091bf7a74fSMilanka Ringwald log_info("AVRCP target: unhandled pdu id 0x%02x", pdu_id); 1210*aeb99916SMilanka Ringwald avrcp_target_response_vendor_dependent_reject(connection, pdu_id, AVRCP_STATUS_INVALID_COMMAND); 1211e0bbf3edSMilanka Ringwald break; 1212e0bbf3edSMilanka Ringwald } 12134b338011SMilanka Ringwald break; 12144b338011SMilanka Ringwald default: 12151bf7a74fSMilanka Ringwald log_info("AVRCP target: opcode 0x%02x not implemented", avrcp_cmd_opcode(packet,size)); 12164b338011SMilanka Ringwald break; 12174b338011SMilanka Ringwald } 121801dc6e35SMilanka Ringwald } 121901dc6e35SMilanka Ringwald 1220*aeb99916SMilanka 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){ 1221df842b64SMatthias Ringwald btstack_assert((value_len == 0) || (value != NULL)); 1222df842b64SMatthias Ringwald 1223*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_CHANGED_STABLE, AVRCP_PDU_ID_REGISTER_NOTIFICATION); 1224022b77fcSMilanka Ringwald connection->transaction_id = avrcp_target_get_transaction_label_for_notification(connection, notification_id); 1225d1207cd8SMilanka Ringwald 1226*aeb99916SMilanka Ringwald connection->data_len = 1 + value_len; 1227*aeb99916SMilanka Ringwald connection->data[0] = notification_id; 1228df842b64SMatthias Ringwald if (value_len > 0){ 1229*aeb99916SMilanka Ringwald (void)memcpy(connection->data + 1, value, value_len); 1230*aeb99916SMilanka Ringwald } 1231df842b64SMatthias Ringwald } 1232bf67b2dbSMatthias Ringwald 1233*aeb99916SMilanka Ringwald static void avrcp_target_notification_addressed_player_changed_init(avrcp_connection_t * connection){ 1234*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_CHANGED_STABLE, AVRCP_PDU_ID_REGISTER_NOTIFICATION); 1235*aeb99916SMilanka Ringwald connection->transaction_id = avrcp_target_get_transaction_label_for_notification(connection, AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED); 1236*aeb99916SMilanka Ringwald 1237*aeb99916SMilanka Ringwald connection->data_len = 5; 1238*aeb99916SMilanka Ringwald connection->data[0] = AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED; 1239*aeb99916SMilanka Ringwald big_endian_store_16(connection->data, 1, connection->addressed_player_id); 1240*aeb99916SMilanka Ringwald big_endian_store_16(connection->data, 3, connection->uid_counter); 1241d1207cd8SMilanka Ringwald } 1242d1207cd8SMilanka Ringwald 1243*aeb99916SMilanka Ringwald 12440ec79bd0SMilanka Ringwald static void avrcp_target_reset_notification(avrcp_connection_t * connection, avrcp_notification_event_id_t notification_id){ 1245*aeb99916SMilanka Ringwald if (notification_id < AVRCP_NOTIFICATION_EVENT_FIRST_INDEX || notification_id > AVRCP_NOTIFICATION_EVENT_LAST_INDEX){ 124612b7c8ecSMilanka Ringwald return; 124712b7c8ecSMilanka Ringwald } 124812b7c8ecSMilanka Ringwald connection->notifications_enabled &= ~(1 << notification_id); 1249*aeb99916SMilanka Ringwald connection->notifications_transaction_label[notification_id] = 0; 1250*aeb99916SMilanka Ringwald } 1251*aeb99916SMilanka Ringwald 1252*aeb99916SMilanka Ringwald static void avrcp_request_next_avctp_segment(avrcp_connection_t * connection){ 1253*aeb99916SMilanka Ringwald // AVCTP 1254*aeb99916SMilanka Ringwald switch (connection->avctp_packet_type){ 1255*aeb99916SMilanka Ringwald case AVCTP_END_PACKET: 1256*aeb99916SMilanka Ringwald case AVRCP_SINGLE_PACKET: 1257*aeb99916SMilanka Ringwald connection->avrcp_frame_bytes_sent = 0; 1258*aeb99916SMilanka Ringwald connection->data_offset = 0; 1259*aeb99916SMilanka Ringwald connection->data_len = 0; 1260*aeb99916SMilanka Ringwald connection->state = AVCTP_CONNECTION_OPENED; 1261*aeb99916SMilanka Ringwald break; 1262*aeb99916SMilanka Ringwald default: 1263*aeb99916SMilanka Ringwald connection->state = AVCTP_W2_SEND_AVCTP_FRAGMENTED_MESSAGE; 1264*aeb99916SMilanka Ringwald avrcp_request_can_send_now(connection, connection->l2cap_signaling_cid); 1265*aeb99916SMilanka Ringwald break; 1266*aeb99916SMilanka Ringwald } 126712b7c8ecSMilanka Ringwald } 126812b7c8ecSMilanka Ringwald 1269654724deSMilanka Ringwald static void avrcp_target_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){ 127001dc6e35SMilanka Ringwald avrcp_connection_t * connection; 1271*aeb99916SMilanka Ringwald 127201dc6e35SMilanka Ringwald switch (packet_type) { 127301dc6e35SMilanka Ringwald case L2CAP_DATA_PACKET: 12741945fe3eSMilanka Ringwald connection = avrcp_get_connection_for_l2cap_signaling_cid_for_role(AVRCP_TARGET, channel); 127501dc6e35SMilanka Ringwald avrcp_handle_l2cap_data_packet_for_signaling_connection(connection, packet, size); 127601dc6e35SMilanka Ringwald break; 127701dc6e35SMilanka Ringwald case HCI_EVENT_PACKET: 127801dc6e35SMilanka Ringwald switch (hci_event_packet_get_type(packet)){ 1279d1207cd8SMilanka Ringwald case L2CAP_EVENT_CAN_SEND_NOW:{ 12801945fe3eSMilanka Ringwald connection = avrcp_get_connection_for_l2cap_signaling_cid_for_role(AVRCP_TARGET, channel); 1281d1207cd8SMilanka Ringwald 1282e3d57ee2SMilanka Ringwald if (connection->accept_response){ 12830ec79bd0SMilanka Ringwald connection->accept_response = 0; 1284*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, AVRCP_PDU_ID_REQUEST_CONTINUING_RESPONSE); 1285*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 1286*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1287182c9f07SMatthias Ringwald return; 12880ec79bd0SMilanka Ringwald } 1289fa06a5d9SMatthias Ringwald 1290182c9f07SMatthias Ringwald if (connection->abort_continue_response){ 12911bf7a74fSMilanka Ringwald connection->abort_continue_response = 0; 1292a1946e79SMilanka Ringwald connection->now_playing_info_response = false; 1293*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, AVRCP_PDU_ID_REQUEST_ABORT_CONTINUING_RESPONSE); 1294*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 1295*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1296182c9f07SMatthias Ringwald return; 12971bf7a74fSMilanka Ringwald } 12981bf7a74fSMilanka Ringwald 1299182c9f07SMatthias Ringwald if (connection->now_playing_info_response){ 1300a1946e79SMilanka Ringwald connection->now_playing_info_response = false; 1301*aeb99916SMilanka Ringwald avrcp_target_vendor_dependent_response_data_init(connection, AVRCP_CTYPE_RESPONSE_ACCEPTED, AVRCP_PDU_ID_GET_ELEMENT_ATTRIBUTES); 1302*aeb99916SMilanka Ringwald connection->data_len = avrcp_now_playing_info_value_len_with_headers(connection); 1303*aeb99916SMilanka Ringwald 1304*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 1305*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1306182c9f07SMatthias Ringwald return; 13075e3d4d2bSMilanka Ringwald } 1308d1207cd8SMilanka Ringwald 1309182c9f07SMatthias Ringwald if (connection->track_changed){ 1310d1207cd8SMilanka Ringwald connection->track_changed = 0; 1311*aeb99916SMilanka Ringwald avrcp_target_notification_init(connection, AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED, connection->track_id, 8); 1312*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 131312b7c8ecSMilanka Ringwald avrcp_target_reset_notification(connection, AVRCP_NOTIFICATION_EVENT_TRACK_CHANGED); 1314*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1315182c9f07SMatthias Ringwald return; 1316d1207cd8SMilanka Ringwald } 1317d1207cd8SMilanka Ringwald 1318182c9f07SMatthias Ringwald if (connection->playback_status_changed){ 1319d1207cd8SMilanka Ringwald connection->playback_status_changed = 0; 1320c57adf92SMilanka Ringwald uint8_t playback_status = (uint8_t) connection->playback_status; 1321*aeb99916SMilanka Ringwald avrcp_target_notification_init(connection, AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED, &playback_status, 1); 1322*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 132312b7c8ecSMilanka Ringwald avrcp_target_reset_notification(connection, AVRCP_NOTIFICATION_EVENT_PLAYBACK_STATUS_CHANGED); 1324*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1325182c9f07SMatthias Ringwald return; 1326d1207cd8SMilanka Ringwald } 1327d1207cd8SMilanka Ringwald 1328182c9f07SMatthias Ringwald if (connection->playing_content_changed){ 1329d1207cd8SMilanka Ringwald connection->playing_content_changed = 0; 1330*aeb99916SMilanka Ringwald // TODO review 1331*aeb99916SMilanka Ringwald avrcp_target_notification_init(connection, AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED, NULL, 1332*aeb99916SMilanka Ringwald avrcp_now_playing_info_value_len_with_headers(connection)); 1333*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 133412b7c8ecSMilanka Ringwald avrcp_target_reset_notification(connection, AVRCP_NOTIFICATION_EVENT_NOW_PLAYING_CONTENT_CHANGED); 1335*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1336182c9f07SMatthias Ringwald return; 1337d1207cd8SMilanka Ringwald } 1338d1207cd8SMilanka Ringwald 1339182c9f07SMatthias Ringwald if (connection->battery_status_changed){ 1340d1207cd8SMilanka Ringwald connection->battery_status_changed = 0; 1341*aeb99916SMilanka Ringwald avrcp_target_notification_init(connection, AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED, (uint8_t *)&connection->battery_status, 1); 1342*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 134312b7c8ecSMilanka Ringwald avrcp_target_reset_notification(connection, AVRCP_NOTIFICATION_EVENT_BATT_STATUS_CHANGED); 1344*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1345182c9f07SMatthias Ringwald return; 1346d1207cd8SMilanka Ringwald } 1347d1207cd8SMilanka Ringwald 1348984bbc3eSMilanka Ringwald if (connection->notify_absolute_volume_changed){ 1349984bbc3eSMilanka Ringwald connection->notify_absolute_volume_changed = 0; 1350*aeb99916SMilanka Ringwald avrcp_target_notification_init(connection, AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED, &connection->absolute_volume, 1); 1351*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 135212b7c8ecSMilanka Ringwald avrcp_target_reset_notification(connection, AVRCP_NOTIFICATION_EVENT_VOLUME_CHANGED); 1353*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1354182c9f07SMatthias Ringwald return; 1355d1207cd8SMilanka Ringwald } 1356d1207cd8SMilanka Ringwald 1357f1028954SMilanka Ringwald if (connection->addressed_player_changed){ 1358f1028954SMilanka Ringwald connection->addressed_player_changed = 0; 1359*aeb99916SMilanka Ringwald avrcp_target_notification_addressed_player_changed_init(connection); 1360*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 1361f1028954SMilanka Ringwald avrcp_target_reset_notification(connection, AVRCP_NOTIFICATION_EVENT_ADDRESSED_PLAYER_CHANGED); 1362*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1363f1028954SMilanka Ringwald return; 1364f1028954SMilanka Ringwald } 1365f1028954SMilanka Ringwald 1366182c9f07SMatthias Ringwald if (connection->reject_transport_header){ 1367355a220fSMilanka Ringwald connection->reject_transport_header = 0; 1368*aeb99916SMilanka Ringwald avrcp_send_reject_cmd_wrong_pid(connection); 1369*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1370182c9f07SMatthias Ringwald return; 1371355a220fSMilanka Ringwald } 1372355a220fSMilanka Ringwald 1373d1207cd8SMilanka Ringwald switch (connection->state){ 1374*aeb99916SMilanka Ringwald // next AVCTP segment 1375*aeb99916SMilanka Ringwald case AVCTP_W2_SEND_AVCTP_FRAGMENTED_MESSAGE: 1376d1207cd8SMilanka Ringwald case AVCTP_W2_SEND_RESPONSE: 1377*aeb99916SMilanka Ringwald avrcp_send_response_with_avctp_fragmentation(connection); 1378*aeb99916SMilanka Ringwald avrcp_request_next_avctp_segment(connection); 1379*aeb99916SMilanka Ringwald break; 1380*aeb99916SMilanka Ringwald 13810ec79bd0SMilanka Ringwald default: 13820ec79bd0SMilanka Ringwald break; 1383d1386928SMilanka Ringwald } 13840ec79bd0SMilanka Ringwald 138501dc6e35SMilanka Ringwald break; 1386d1207cd8SMilanka Ringwald } 138701dc6e35SMilanka Ringwald default: 138801dc6e35SMilanka Ringwald break; 138901dc6e35SMilanka Ringwald } 139001dc6e35SMilanka Ringwald default: 139101dc6e35SMilanka Ringwald break; 139201dc6e35SMilanka Ringwald } 139301dc6e35SMilanka Ringwald } 139401dc6e35SMilanka Ringwald 139501dc6e35SMilanka Ringwald void avrcp_target_init(void){ 139601dc6e35SMilanka Ringwald avrcp_target_context.role = AVRCP_TARGET; 1397654724deSMilanka Ringwald avrcp_target_context.packet_handler = avrcp_target_packet_handler; 13980ec79bd0SMilanka Ringwald avrcp_register_target_packet_handler(&avrcp_target_packet_handler); 139901dc6e35SMilanka Ringwald } 140001dc6e35SMilanka Ringwald 1401680af5dcSMatthias Ringwald void avrcp_target_deinit(void){ 1402680af5dcSMatthias Ringwald memset(&avrcp_target_context, 0, sizeof(avrcp_context_t)); 1403680af5dcSMatthias Ringwald } 1404680af5dcSMatthias Ringwald 140501dc6e35SMilanka Ringwald void avrcp_target_register_packet_handler(btstack_packet_handler_t callback){ 1406be4cc80aSMilanka Ringwald btstack_assert(callback != NULL); 140701dc6e35SMilanka Ringwald avrcp_target_context.avrcp_callback = callback; 140801dc6e35SMilanka Ringwald } 140901dc6e35SMilanka Ringwald 1410d1cf25b1SMatthias Ringwald void avrcp_target_register_set_addressed_player_handler(bool (*callback)(uint16_t player_id)){ 141195f00c7eSMilanka Ringwald btstack_assert(callback != NULL); 1412d1cf25b1SMatthias Ringwald avrcp_target_context.set_addressed_player_callback = callback; 1413d1cf25b1SMatthias Ringwald } 1414d1cf25b1SMatthias Ringwald 141501dc6e35SMilanka Ringwald 1416