mirror of
https://github.com/bluekitchen/btstack.git
synced 2025-03-01 10:13:29 +00:00
705 lines
31 KiB
C
705 lines
31 KiB
C
/*
|
|
* Copyright (C) 2016 BlueKitchen GmbH
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of the copyright holders nor the names of
|
|
* contributors may be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
* 4. Any redistribution, use, or modification is done solely for
|
|
* personal benefit and not for any commercial purpose or for
|
|
* monetary gain.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS
|
|
* RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
|
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* Please inquire about commercial licensing options at
|
|
* contact@bluekitchen-gmbh.com
|
|
*
|
|
*/
|
|
|
|
#define BTSTACK_FILE__ "avrcp_browsing_controller.c"
|
|
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <inttypes.h>
|
|
#include "btstack.h"
|
|
#include "classic/avrcp_browsing.h"
|
|
#include "classic/avrcp_browsing_controller.h"
|
|
|
|
|
|
static void avrcp_browsing_controller_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size);
|
|
|
|
static int avrcp_browsing_controller_send_get_folder_items_cmd(uint16_t cid, avrcp_browsing_connection_t * connection){
|
|
uint8_t command[100];
|
|
int pos = 0;
|
|
// transport header
|
|
// Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier)
|
|
command[pos++] = (connection->transaction_label << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_COMMAND_FRAME << 1) | 0;
|
|
// Profile IDentifier (PID)
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8;
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF;
|
|
command[pos++] = AVRCP_PDU_ID_GET_FOLDER_ITEMS;
|
|
|
|
uint32_t attribute_count = 0;
|
|
uint32_t attributes_to_copy = 0;
|
|
|
|
switch (connection->attr_bitmap){
|
|
case AVRCP_MEDIA_ATTR_NONE:
|
|
attribute_count = AVRCP_MEDIA_ATTR_NONE; // 0xFFFFFFFF
|
|
break;
|
|
case AVRCP_MEDIA_ATTR_ALL:
|
|
attribute_count = AVRCP_MEDIA_ATTR_ALL; // 0
|
|
break;
|
|
default:
|
|
attribute_count = count_set_bits_uint32(connection->attr_bitmap & 0xff);
|
|
attributes_to_copy = attribute_count;
|
|
break;
|
|
}
|
|
big_endian_store_16(command, pos, 9 + 1 + (attribute_count*4));
|
|
pos += 2;
|
|
|
|
command[pos++] = connection->scope;
|
|
big_endian_store_32(command, pos, connection->start_item);
|
|
pos += 4;
|
|
big_endian_store_32(command, pos, connection->end_item);
|
|
pos += 4;
|
|
command[pos++] = attribute_count;
|
|
|
|
int bit_position = 1;
|
|
while (attributes_to_copy){
|
|
if (connection->attr_bitmap & (1 << bit_position)){
|
|
big_endian_store_32(command, pos, bit_position);
|
|
pos += 4;
|
|
attributes_to_copy--;
|
|
}
|
|
bit_position++;
|
|
}
|
|
return l2cap_send(cid, command, pos);
|
|
}
|
|
|
|
|
|
static int avrcp_browsing_controller_send_get_item_attributes_cmd(uint16_t cid, avrcp_browsing_connection_t * connection){
|
|
uint8_t command[100];
|
|
int pos = 0;
|
|
// transport header
|
|
// Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier)
|
|
command[pos++] = (connection->transaction_label << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_COMMAND_FRAME << 1) | 0;
|
|
// Profile IDentifier (PID)
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8;
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF;
|
|
command[pos++] = AVRCP_PDU_ID_GET_ITEM_ATTRIBUTES;
|
|
|
|
uint32_t attribute_count;
|
|
uint32_t attributes_to_copy = 0;
|
|
|
|
switch (connection->attr_bitmap){
|
|
case AVRCP_MEDIA_ATTR_NONE:
|
|
case AVRCP_MEDIA_ATTR_ALL:
|
|
attribute_count = 0;
|
|
break;
|
|
default:
|
|
attribute_count = count_set_bits_uint32(connection->attr_bitmap & 0xff);
|
|
attributes_to_copy = attribute_count;
|
|
break;
|
|
}
|
|
|
|
big_endian_store_16(command, pos, 12 + (attribute_count*4));
|
|
pos += 2;
|
|
|
|
command[pos++] = connection->scope;
|
|
(void)memcpy(command + pos, connection->folder_uid, 8);
|
|
pos += 8;
|
|
big_endian_store_16(command, pos, connection->uid_counter);
|
|
pos += 2;
|
|
command[pos++] = attribute_count;
|
|
|
|
int bit_position = 1;
|
|
while (attributes_to_copy){
|
|
if (connection->attr_bitmap & (1 << bit_position)){
|
|
big_endian_store_32(command, pos, bit_position);
|
|
pos += 4;
|
|
attributes_to_copy--;
|
|
}
|
|
bit_position++;
|
|
}
|
|
|
|
return l2cap_send(cid, command, pos);
|
|
}
|
|
|
|
|
|
static int avrcp_browsing_controller_send_change_path_cmd(uint16_t cid, avrcp_browsing_connection_t * connection){
|
|
uint8_t command[100];
|
|
int pos = 0;
|
|
// transport header
|
|
// Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier)
|
|
command[pos++] = (connection->transaction_label << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_COMMAND_FRAME << 1) | 0;
|
|
// Profile IDentifier (PID)
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8;
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF;
|
|
command[pos++] = AVRCP_PDU_ID_CHANGE_PATH;
|
|
|
|
big_endian_store_16(command, pos, 11);
|
|
pos += 2;
|
|
pos += 2;
|
|
command[pos++] = connection->direction;
|
|
(void)memcpy(command + pos, connection->folder_uid, 8);
|
|
pos += 8;
|
|
return l2cap_send(cid, command, pos);
|
|
}
|
|
|
|
static int avrcp_browsing_controller_send_search_cmd(uint16_t cid, avrcp_browsing_connection_t * connection){
|
|
uint8_t command[100];
|
|
int pos = 0;
|
|
// transport header
|
|
// Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier)
|
|
command[pos++] = (connection->transaction_label << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_COMMAND_FRAME << 1) | 0;
|
|
// Profile IDentifier (PID)
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8;
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF;
|
|
command[pos++] = AVRCP_PDU_ID_SEARCH;
|
|
|
|
big_endian_store_16(command, pos, 4 + connection->search_str_len);
|
|
pos += 2;
|
|
|
|
big_endian_store_16(command, pos, 0x006A);
|
|
pos += 2;
|
|
big_endian_store_16(command, pos, connection->search_str_len);
|
|
pos += 2;
|
|
|
|
(void)memcpy(command + pos, connection->search_str,
|
|
connection->search_str_len);
|
|
pos += connection->search_str_len;
|
|
return l2cap_send(cid, command, pos);
|
|
}
|
|
|
|
static int avrcp_browsing_controller_send_set_browsed_player_cmd(uint16_t cid, avrcp_browsing_connection_t * connection){
|
|
uint8_t command[100];
|
|
int pos = 0;
|
|
// transport header
|
|
// Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier)
|
|
command[pos++] = (connection->transaction_label << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_COMMAND_FRAME << 1) | 0;
|
|
// Profile IDentifier (PID)
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8;
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF;
|
|
command[pos++] = AVRCP_PDU_ID_SET_BROWSED_PLAYER;
|
|
|
|
big_endian_store_16(command, pos, 2);
|
|
pos += 2;
|
|
big_endian_store_16(command, pos, connection->browsed_player_id);
|
|
pos += 2;
|
|
return l2cap_send(cid, command, pos);
|
|
}
|
|
|
|
static int avrcp_browsing_controller_send_get_total_nr_items_cmd(uint16_t cid, avrcp_browsing_connection_t * connection){
|
|
uint8_t command[7];
|
|
int pos = 0;
|
|
// transport header
|
|
// Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier)
|
|
command[pos++] = (connection->transaction_label << 4) | (AVRCP_SINGLE_PACKET << 2) | (AVRCP_COMMAND_FRAME << 1) | 0;
|
|
// Profile IDentifier (PID)
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL >> 8;
|
|
command[pos++] = BLUETOOTH_SERVICE_CLASS_AV_REMOTE_CONTROL & 0x00FF;
|
|
command[pos++] = AVRCP_PDU_ID_GET_TOTAL_NUMBER_OF_ITEMS;
|
|
|
|
big_endian_store_16(command, pos, 1);
|
|
pos += 2;
|
|
command[pos++] = connection->get_total_nr_items_scope;
|
|
return l2cap_send(cid, command, pos);
|
|
}
|
|
|
|
static void avrcp_browsing_controller_handle_can_send_now(avrcp_browsing_connection_t * connection){
|
|
switch (connection->state){
|
|
case AVCTP_CONNECTION_OPENED:
|
|
if (connection->set_browsed_player_id){
|
|
connection->state = AVCTP_W2_RECEIVE_RESPONSE;
|
|
connection->set_browsed_player_id = 0;
|
|
avrcp_browsing_controller_send_set_browsed_player_cmd(connection->l2cap_browsing_cid, connection);
|
|
break;
|
|
}
|
|
|
|
if (connection->get_total_nr_items){
|
|
connection->state = AVCTP_W2_RECEIVE_RESPONSE;
|
|
connection->get_total_nr_items = 0;
|
|
avrcp_browsing_controller_send_get_total_nr_items_cmd(connection->l2cap_browsing_cid, connection);
|
|
break;
|
|
}
|
|
|
|
if (connection->get_folder_items){
|
|
connection->state = AVCTP_W2_RECEIVE_RESPONSE;
|
|
connection->get_folder_items = 0;
|
|
avrcp_browsing_controller_send_get_folder_items_cmd(connection->l2cap_browsing_cid, connection);
|
|
break;
|
|
}
|
|
|
|
if (connection->get_item_attributes){
|
|
connection->state = AVCTP_W2_RECEIVE_RESPONSE;
|
|
connection->get_item_attributes = 0;
|
|
avrcp_browsing_controller_send_get_item_attributes_cmd(connection->l2cap_browsing_cid, connection);
|
|
break;
|
|
}
|
|
|
|
if (connection->change_path){
|
|
connection->state = AVCTP_W2_RECEIVE_RESPONSE;
|
|
connection->change_path = 0;
|
|
avrcp_browsing_controller_send_change_path_cmd(connection->l2cap_browsing_cid, connection);
|
|
break;
|
|
}
|
|
|
|
if (connection->search){
|
|
connection->state = AVCTP_W2_RECEIVE_RESPONSE;
|
|
connection->search = 0;
|
|
avrcp_browsing_controller_send_search_cmd(connection->l2cap_browsing_cid, connection);
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
static void avrcp_browsing_controller_emit_done_with_uid_counter(btstack_packet_handler_t callback, uint16_t browsing_cid, uint16_t uid_counter, uint8_t browsing_status, uint8_t bluetooth_status){
|
|
btstack_assert(callback != NULL);
|
|
|
|
uint8_t event[9];
|
|
int pos = 0;
|
|
event[pos++] = HCI_EVENT_AVRCP_META;
|
|
event[pos++] = sizeof(event) - 2;
|
|
event[pos++] = AVRCP_SUBEVENT_BROWSING_DONE;
|
|
little_endian_store_16(event, pos, browsing_cid);
|
|
pos += 2;
|
|
little_endian_store_16(event, pos, uid_counter);
|
|
pos += 2;
|
|
event[pos++] = browsing_status;
|
|
event[pos++] = bluetooth_status;
|
|
(*callback)(HCI_EVENT_PACKET, 0, event, sizeof(event));
|
|
}
|
|
|
|
static void avrcp_parser_reset(avrcp_browsing_connection_t * connection){
|
|
connection->parser_attribute_header_pos = 0;
|
|
connection->parsed_attribute_value_offset = 0;
|
|
connection->parsed_num_attributes = 0;
|
|
connection->parser_state = AVRCP_PARSER_GET_ATTRIBUTE_HEADER;
|
|
}
|
|
|
|
|
|
static void avrcp_browsing_parser_process_byte(uint8_t byte, avrcp_browsing_connection_t * connection){
|
|
uint8_t prepended_header_size = 1;
|
|
uint16_t attribute_total_value_len;
|
|
|
|
switch(connection->parser_state){
|
|
case AVRCP_PARSER_GET_ATTRIBUTE_HEADER:
|
|
connection->parser_attribute_header[connection->parser_attribute_header_pos++] = byte;
|
|
if (connection->parser_attribute_header_pos < AVRCP_BROWSING_ITEM_HEADER_LEN) break;
|
|
|
|
attribute_total_value_len = big_endian_read_16(connection->parser_attribute_header, 1);
|
|
connection->parsed_attribute_value[connection->parsed_attribute_value_offset++] = connection->parser_attribute_header[0]; // prepend with item type
|
|
connection->parsed_attribute_value_len = btstack_min(attribute_total_value_len, AVRCP_MAX_ATTRIBUTTE_SIZE - prepended_header_size); // reduce AVRCP_MAX_ATTRIBUTTE_SIZE for the size ot item type
|
|
connection->parser_state = AVRCP_PARSER_GET_ATTRIBUTE_VALUE;
|
|
break;
|
|
|
|
case AVRCP_PARSER_GET_ATTRIBUTE_VALUE:
|
|
connection->parsed_attribute_value[connection->parsed_attribute_value_offset++] = byte;
|
|
if (connection->parsed_attribute_value_offset < (connection->parsed_attribute_value_len + prepended_header_size)){
|
|
break;
|
|
}
|
|
if (connection->parsed_attribute_value_offset < big_endian_read_16(connection->parser_attribute_header, 1)){
|
|
connection->parser_state = AVRCP_PARSER_IGNORE_REST_OF_ATTRIBUTE_VALUE;
|
|
break;
|
|
}
|
|
connection->parser_state = AVRCP_PARSER_GET_ATTRIBUTE_HEADER;
|
|
(*avrcp_controller_context.browsing_avrcp_callback)(AVRCP_BROWSING_DATA_PACKET, connection->l2cap_browsing_cid, &connection->parsed_attribute_value[0], connection->parsed_attribute_value_offset);
|
|
connection->parsed_num_attributes++;
|
|
connection->parsed_attribute_value_offset = 0;
|
|
connection->parser_attribute_header_pos = 0;
|
|
|
|
if (connection->parsed_num_attributes == connection->num_items){
|
|
avrcp_parser_reset(connection);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case AVRCP_PARSER_IGNORE_REST_OF_ATTRIBUTE_VALUE:
|
|
connection->parsed_attribute_value_offset++;
|
|
if (connection->parsed_attribute_value_offset < (big_endian_read_16(connection->parser_attribute_header, 1) + prepended_header_size)){
|
|
break;
|
|
}
|
|
connection->parser_state = AVRCP_PARSER_GET_ATTRIBUTE_HEADER;
|
|
(*avrcp_controller_context.browsing_avrcp_callback)(AVRCP_BROWSING_DATA_PACKET, connection->l2cap_browsing_cid, &connection->parsed_attribute_value[0], connection->parsed_attribute_value_offset);
|
|
connection->parsed_num_attributes++;
|
|
connection->parsed_attribute_value_offset = 0;
|
|
connection->parser_attribute_header_pos = 0;
|
|
|
|
if (connection->parsed_num_attributes == connection->num_items){
|
|
avrcp_parser_reset(connection);
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void avrcp_browsing_parse_and_emit_element_attrs(uint8_t * packet, uint16_t num_bytes_to_read, avrcp_browsing_connection_t * connection){
|
|
int i;
|
|
for (i=0;i<num_bytes_to_read;i++){
|
|
avrcp_browsing_parser_process_byte(packet[i], connection);
|
|
}
|
|
}
|
|
|
|
static void avrcp_browsing_controller_emit_failed(btstack_packet_handler_t callback, uint16_t browsing_cid, uint8_t browsing_status, uint8_t bluetooth_status){
|
|
avrcp_browsing_controller_emit_done_with_uid_counter(callback, browsing_cid, 0, browsing_status, bluetooth_status);
|
|
}
|
|
|
|
|
|
static void avrcp_browsing_controller_packet_handler(uint8_t packet_type, uint16_t channel, uint8_t *packet, uint16_t size){
|
|
avrcp_browsing_connection_t * browsing_connection;
|
|
uint8_t transport_header;
|
|
int pos;
|
|
switch (packet_type) {
|
|
case L2CAP_DATA_PACKET:
|
|
browsing_connection = get_avrcp_browsing_connection_for_l2cap_cid_for_role(AVRCP_CONTROLLER, channel);
|
|
if (!browsing_connection) break;
|
|
pos = 0;
|
|
transport_header = packet[pos++];
|
|
// Transaction label | Packet_type | C/R | IPID (1 == invalid profile identifier)
|
|
browsing_connection->transaction_label = transport_header >> 4;
|
|
avrcp_packet_type_t avctp_packet_type = (transport_header & 0x0F) >> 2;
|
|
switch (avctp_packet_type){
|
|
case AVRCP_SINGLE_PACKET:
|
|
case AVRCP_START_PACKET:
|
|
pos += 2;
|
|
browsing_connection->num_packets = 1;
|
|
if (avctp_packet_type == AVRCP_START_PACKET){
|
|
browsing_connection->num_packets = packet[pos++];
|
|
}
|
|
if ((pos + 4) > size){
|
|
browsing_connection->state = AVCTP_CONNECTION_OPENED;
|
|
avrcp_browsing_controller_emit_failed(avrcp_controller_context.browsing_avrcp_callback, channel, AVRCP_BROWSING_ERROR_CODE_INVALID_COMMAND, ERROR_CODE_SUCCESS);
|
|
return;
|
|
}
|
|
browsing_connection->pdu_id = packet[pos++];
|
|
pos += 2;
|
|
browsing_connection->browsing_status = packet[pos++];
|
|
if (browsing_connection->browsing_status != AVRCP_BROWSING_ERROR_CODE_SUCCESS){
|
|
browsing_connection->state = AVCTP_CONNECTION_OPENED;
|
|
avrcp_browsing_controller_emit_failed(avrcp_controller_context.browsing_avrcp_callback, channel, browsing_connection->browsing_status, ERROR_CODE_SUCCESS);
|
|
return;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
uint32_t i;
|
|
uint8_t folder_depth;
|
|
|
|
switch(browsing_connection->pdu_id){
|
|
case AVRCP_PDU_ID_CHANGE_PATH:
|
|
break;
|
|
case AVRCP_PDU_ID_SET_ADDRESSED_PLAYER:
|
|
break;
|
|
case AVRCP_PDU_ID_GET_TOTAL_NUMBER_OF_ITEMS:
|
|
break;
|
|
case AVRCP_PDU_ID_SET_BROWSED_PLAYER:
|
|
browsing_connection->uid_counter = big_endian_read_16(packet, pos);
|
|
pos += 2;
|
|
// num_items
|
|
pos += 4;
|
|
// charset
|
|
pos += 2;
|
|
folder_depth = packet[pos++];
|
|
|
|
for (i = 0; i < folder_depth; i++){
|
|
uint16_t folder_name_length = big_endian_read_16(packet, pos);
|
|
pos += 2;
|
|
// reuse packet and add data type as a header
|
|
packet[pos-1] = AVRCP_BROWSING_MEDIA_ROOT_FOLDER;
|
|
(*avrcp_controller_context.browsing_avrcp_callback)(AVRCP_BROWSING_DATA_PACKET, channel, packet+pos-1, folder_name_length+1);
|
|
pos += folder_name_length;
|
|
}
|
|
break;
|
|
|
|
case AVRCP_PDU_ID_GET_FOLDER_ITEMS:{
|
|
switch (avctp_packet_type){
|
|
case AVRCP_SINGLE_PACKET:
|
|
case AVRCP_START_PACKET:
|
|
avrcp_parser_reset(browsing_connection);
|
|
browsing_connection->uid_counter = big_endian_read_16(packet, pos);
|
|
pos += 2;
|
|
browsing_connection->num_items = big_endian_read_16(packet, pos); //num_items
|
|
pos += 2;
|
|
avrcp_browsing_parse_and_emit_element_attrs(packet+pos, size-pos, browsing_connection);
|
|
break;
|
|
|
|
case AVRCP_CONTINUE_PACKET:
|
|
avrcp_browsing_parse_and_emit_element_attrs(packet+pos, size-pos, browsing_connection);
|
|
break;
|
|
|
|
case AVRCP_END_PACKET:
|
|
avrcp_browsing_parse_and_emit_element_attrs(packet+pos, size-pos, browsing_connection);
|
|
avrcp_parser_reset(browsing_connection);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
case AVRCP_PDU_ID_SEARCH:
|
|
browsing_connection->uid_counter = big_endian_read_16(packet, pos);
|
|
pos += 2;
|
|
break;
|
|
case AVRCP_PDU_ID_GET_ITEM_ATTRIBUTES:
|
|
packet[pos-1] = AVRCP_BROWSING_MEDIA_ELEMENT_ITEM_ATTRIBUTE;
|
|
(*avrcp_controller_context.browsing_avrcp_callback)(AVRCP_BROWSING_DATA_PACKET, channel, packet+pos-1, size - pos + 1);
|
|
break;
|
|
default:
|
|
log_info(" not parsed pdu ID 0x%02x", browsing_connection->pdu_id);
|
|
break;
|
|
}
|
|
|
|
switch (avctp_packet_type){
|
|
case AVRCP_SINGLE_PACKET:
|
|
case AVRCP_END_PACKET:
|
|
browsing_connection->state = AVCTP_CONNECTION_OPENED;
|
|
avrcp_browsing_controller_emit_done_with_uid_counter(avrcp_controller_context.browsing_avrcp_callback, channel, browsing_connection->uid_counter, browsing_connection->browsing_status, ERROR_CODE_SUCCESS);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case HCI_EVENT_PACKET:
|
|
switch (hci_event_packet_get_type(packet)){
|
|
case L2CAP_EVENT_CAN_SEND_NOW:
|
|
browsing_connection = get_avrcp_browsing_connection_for_l2cap_cid_for_role(AVRCP_CONTROLLER,channel);
|
|
avrcp_browsing_controller_handle_can_send_now(browsing_connection);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void avrcp_browsing_controller_init(void){
|
|
avrcp_controller_context.browsing_packet_handler = avrcp_browsing_controller_packet_handler;
|
|
avrcp_browsing_register_controller_packet_handler(avrcp_browsing_controller_packet_handler);
|
|
}
|
|
|
|
void avrcp_browsing_controller_register_packet_handler(btstack_packet_handler_t callback){
|
|
btstack_assert(callback != NULL);
|
|
avrcp_controller_context.browsing_avrcp_callback = callback;
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_get_item_attributes_for_scope(uint16_t avrcp_browsing_cid, uint8_t * uid, uint16_t uid_counter, uint32_t attr_bitmap, avrcp_browsing_scope_t scope){
|
|
avrcp_connection_t * avrcp_connection = get_avrcp_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
|
|
if (!avrcp_connection){
|
|
log_error("avrcp_browsing_controller_get_item_attributes: could not find a connection.");
|
|
return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
|
|
}
|
|
avrcp_browsing_connection_t * connection = avrcp_connection->browsing_connection;
|
|
if (connection->state != AVCTP_CONNECTION_OPENED){
|
|
log_error("avrcp_browsing_controller_get_item_attributes: connection in wrong state %d, expected %d.", connection->state, AVCTP_CONNECTION_OPENED);
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
|
|
connection->get_item_attributes = 1;
|
|
connection->scope = scope;
|
|
(void)memcpy(connection->folder_uid, uid, 8);
|
|
connection->uid_counter = uid_counter;
|
|
connection->attr_bitmap = attr_bitmap;
|
|
|
|
avrcp_browsing_request_can_send_now(connection, connection->l2cap_browsing_cid);
|
|
return ERROR_CODE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief Retrieve a listing of the contents of a folder.
|
|
* @param scope 0-player list, 1-virtual file system, 2-search, 3-now playing
|
|
* @param start_item
|
|
* @param end_item
|
|
* @param attribute_count
|
|
* @param attribute_list
|
|
**/
|
|
static uint8_t avrcp_browsing_controller_get_folder_items(uint16_t avrcp_browsing_cid, avrcp_browsing_scope_t scope, uint32_t start_item, uint32_t end_item, uint32_t attr_bitmap){
|
|
avrcp_connection_t * avrcp_connection = get_avrcp_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
|
|
if (!avrcp_connection){
|
|
log_error("avrcp_browsing_controller_disconnect: could not find a connection.");
|
|
return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
|
|
}
|
|
avrcp_browsing_connection_t * connection = avrcp_connection->browsing_connection;
|
|
if (connection->state != AVCTP_CONNECTION_OPENED) {
|
|
log_error("avrcp_browsing_controller_get_folder_items: connection in wrong state %d, expected %d.", connection->state, AVCTP_CONNECTION_OPENED);
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
|
|
connection->get_folder_items = 1;
|
|
connection->scope = scope;
|
|
connection->start_item = start_item;
|
|
connection->end_item = end_item;
|
|
connection->attr_bitmap = attr_bitmap;
|
|
|
|
avrcp_browsing_request_can_send_now(connection, connection->l2cap_browsing_cid);
|
|
return ERROR_CODE_SUCCESS;
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_get_media_players(uint16_t avrcp_browsing_cid, uint32_t start_item, uint32_t end_item, uint32_t attr_bitmap){
|
|
return avrcp_browsing_controller_get_folder_items(avrcp_browsing_cid, AVRCP_BROWSING_MEDIA_PLAYER_LIST, start_item, end_item, attr_bitmap);
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_browse_file_system(uint16_t avrcp_browsing_cid, uint32_t start_item, uint32_t end_item, uint32_t attr_bitmap){
|
|
// return avrcp_browsing_controller_get_folder_items(avrcp_browsing_cid, 1, 0, 0xFFFFFFFF, attr_bitmap);
|
|
return avrcp_browsing_controller_get_folder_items(avrcp_browsing_cid, AVRCP_BROWSING_MEDIA_PLAYER_VIRTUAL_FILESYSTEM, start_item, end_item, attr_bitmap);
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_browse_media(uint16_t avrcp_browsing_cid, uint32_t start_item, uint32_t end_item, uint32_t attr_bitmap){
|
|
// return avrcp_browsing_controller_get_folder_items(avrcp_browsing_cid, 2, 0, 0xFFFFFFFF, 0, NULL);
|
|
return avrcp_browsing_controller_get_folder_items(avrcp_browsing_cid, AVRCP_BROWSING_SEARCH, start_item, end_item, attr_bitmap);
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_browse_now_playing_list(uint16_t avrcp_browsing_cid, uint32_t start_item, uint32_t end_item, uint32_t attr_bitmap){
|
|
return avrcp_browsing_controller_get_folder_items(avrcp_browsing_cid, AVRCP_BROWSING_NOW_PLAYING, start_item, end_item, attr_bitmap);
|
|
}
|
|
|
|
|
|
uint8_t avrcp_browsing_controller_set_browsed_player(uint16_t avrcp_browsing_cid, uint16_t browsed_player_id){
|
|
avrcp_connection_t * avrcp_connection = get_avrcp_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
|
|
if (!avrcp_connection){
|
|
log_error("avrcp_browsing_controller_change_path: could not find a connection.");
|
|
return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
|
|
}
|
|
|
|
avrcp_browsing_connection_t * connection = avrcp_connection->browsing_connection;
|
|
if (connection->state != AVCTP_CONNECTION_OPENED){
|
|
log_error("avrcp_browsing_controller_change_path: connection in wrong state.");
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
|
|
connection->set_browsed_player_id = 1;
|
|
connection->browsed_player_id = browsed_player_id;
|
|
avrcp_browsing_request_can_send_now(connection, connection->l2cap_browsing_cid);
|
|
return ERROR_CODE_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* @brief Retrieve a listing of the contents of a folder.
|
|
* @param direction 0-folder up, 1-folder down
|
|
* @param folder_uid 8 bytes long
|
|
**/
|
|
uint8_t avrcp_browsing_controller_change_path(uint16_t avrcp_browsing_cid, uint8_t direction, uint8_t * folder_uid){
|
|
avrcp_connection_t * avrcp_connection = get_avrcp_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
|
|
if (!avrcp_connection){
|
|
log_error("avrcp_browsing_controller_change_path: could not find a connection.");
|
|
return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
|
|
}
|
|
|
|
avrcp_browsing_connection_t * connection = avrcp_connection->browsing_connection;
|
|
|
|
if ((connection == NULL) || (connection->state != AVCTP_CONNECTION_OPENED)){
|
|
log_error("avrcp_browsing_controller_change_path: connection in wrong state.");
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
|
|
if (!connection->browsed_player_id){
|
|
log_error("avrcp_browsing_controller_change_path: no browsed player set.");
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
connection->change_path = 1;
|
|
connection->direction = direction;
|
|
memset(connection->folder_uid, 0, 8);
|
|
if (folder_uid){
|
|
(void)memcpy(connection->folder_uid, folder_uid, 8);
|
|
}
|
|
|
|
avrcp_browsing_request_can_send_now(connection, connection->l2cap_browsing_cid);
|
|
return ERROR_CODE_SUCCESS;
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_go_up_one_level(uint16_t avrcp_browsing_cid){
|
|
return avrcp_browsing_controller_change_path(avrcp_browsing_cid, 0, NULL);
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_go_down_one_level(uint16_t avrcp_browsing_cid, uint8_t * folder_uid){
|
|
return avrcp_browsing_controller_change_path(avrcp_browsing_cid, 1, folder_uid);
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_search(uint16_t avrcp_browsing_cid, uint16_t search_str_len, char * search_str){
|
|
avrcp_connection_t * avrcp_connection = get_avrcp_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
|
|
if (!avrcp_connection){
|
|
log_error("avrcp_browsing_controller_change_path: could not find a connection.");
|
|
return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
|
|
}
|
|
|
|
avrcp_browsing_connection_t * connection = avrcp_connection->browsing_connection;
|
|
|
|
if ((connection == NULL) || (connection->state != AVCTP_CONNECTION_OPENED)){
|
|
log_error("avrcp_browsing_controller_change_path: connection in wrong state.");
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
|
|
if (!connection->browsed_player_id){
|
|
log_error("avrcp_browsing_controller_change_path: no browsed player set.");
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
if (!search_str || (search_str_len == 0)){
|
|
return AVRCP_BROWSING_ERROR_CODE_INVALID_COMMAND;
|
|
}
|
|
|
|
connection->search = 1;
|
|
|
|
connection->search_str_len = btstack_min(search_str_len, sizeof(connection->search_str)-1);
|
|
memset(connection->search_str, 0, sizeof(connection->search_str));
|
|
(void)memcpy(connection->search_str, search_str,
|
|
connection->search_str_len);
|
|
avrcp_browsing_request_can_send_now(connection, connection->l2cap_browsing_cid);
|
|
return ERROR_CODE_SUCCESS;
|
|
}
|
|
|
|
uint8_t avrcp_browsing_controller_get_total_nr_items_for_scope(uint16_t avrcp_browsing_cid, avrcp_browsing_scope_t scope){
|
|
avrcp_connection_t * avrcp_connection = get_avrcp_connection_for_browsing_cid_for_role(AVRCP_CONTROLLER, avrcp_browsing_cid);
|
|
if (!avrcp_connection){
|
|
log_error("avrcp_browsing_controller_change_path: could not find a connection.");
|
|
return ERROR_CODE_UNKNOWN_CONNECTION_IDENTIFIER;
|
|
}
|
|
|
|
avrcp_browsing_connection_t * connection = avrcp_connection->browsing_connection;
|
|
|
|
if ((connection == NULL) || (connection->state != AVCTP_CONNECTION_OPENED)){
|
|
log_error("avrcp_browsing_controller_change_path: connection in wrong state.");
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
|
|
if (!connection->browsed_player_id){
|
|
log_error("avrcp_browsing_controller_change_path: no browsed player set.");
|
|
return ERROR_CODE_COMMAND_DISALLOWED;
|
|
}
|
|
connection->get_total_nr_items = 1;
|
|
connection->get_total_nr_items_scope = scope;
|
|
avrcp_browsing_request_can_send_now(connection, connection->l2cap_browsing_cid);
|
|
return ERROR_CODE_SUCCESS;
|
|
}
|