70474ae844
components/bt: Add AVRCP feature about volume See merge request idf/esp-idf!5209
485 lines
14 KiB
C
485 lines
14 KiB
C
// Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#include "common/bt_target.h"
|
|
#include <string.h>
|
|
#include "esp_err.h"
|
|
#include "esp_avrc_api.h"
|
|
#include "esp_bt_main.h"
|
|
#include "btc/btc_manage.h"
|
|
#include "btc_avrc.h"
|
|
|
|
#if BTC_AV_INCLUDED
|
|
|
|
esp_err_t esp_avrc_ct_register_callback(esp_avrc_ct_cb_t callback)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (callback == NULL) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
btc_profile_cb_set(BTC_PID_AVRC_CT, callback);
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_init(void)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_CT_API_INIT_EVT;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, NULL, 0, NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_deinit(void)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_CT_API_DEINIT_EVT;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, NULL, 0, NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_send_set_player_value_cmd(uint8_t tl, uint8_t attr_id, uint8_t value_id)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (tl > ESP_AVRC_TRANS_LABEL_MAX || attr_id > ESP_AVRC_PS_MAX_ATTR - 1) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_CTRL_API_SND_SET_PLAYER_SETTING_EVT;
|
|
|
|
btc_avrc_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_args_t));
|
|
|
|
arg.ps_cmd.tl = tl;
|
|
arg.ps_cmd.attr_id = attr_id;
|
|
arg.ps_cmd.value_id = value_id;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_send_get_rn_capabilities_cmd(uint8_t tl)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (tl > ESP_AVRC_TRANS_LABEL_MAX) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_STATUS_API_SND_GET_RN_CAPS_EVT;
|
|
|
|
btc_avrc_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_args_t));
|
|
|
|
arg.get_caps_cmd.tl = tl;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_send_register_notification_cmd(uint8_t tl, uint8_t event_id, uint32_t event_parameter)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (tl > ESP_AVRC_TRANS_LABEL_MAX || event_id > ESP_AVRC_RN_MAX_EVT - 1) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
if (!btc_avrc_ct_rn_evt_supported(event_id)) {
|
|
return ESP_ERR_NOT_SUPPORTED;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_NOTIFY_API_SND_REG_NOTIFY_EVT;
|
|
|
|
btc_avrc_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_args_t));
|
|
|
|
arg.rn_cmd.tl = tl;
|
|
arg.rn_cmd.event_id = event_id;
|
|
arg.rn_cmd.event_parameter = event_parameter;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_send_set_absolute_volume_cmd(uint8_t tl, uint8_t volume)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (tl > ESP_AVRC_TRANS_LABEL_MAX) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
if (volume > BTC_AVRC_MAX_VOLUME) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
if (!btc_avrc_ct_rn_evt_supported(ESP_AVRC_RN_VOLUME_CHANGE)) {
|
|
return ESP_ERR_NOT_SUPPORTED;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_CTRL_API_SND_SET_ABSOLUTE_VOLUME_EVT;
|
|
|
|
btc_avrc_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_args_t));
|
|
|
|
arg.set_abs_vol_cmd.tl = tl;
|
|
arg.set_abs_vol_cmd.volume = volume;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_send_metadata_cmd(uint8_t tl, uint8_t attr_mask)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (tl > ESP_AVRC_TRANS_LABEL_MAX) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_STATUS_API_SND_META_EVT;
|
|
|
|
btc_avrc_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_args_t));
|
|
|
|
arg.md_cmd.tl = tl;
|
|
arg.md_cmd.attr_mask = attr_mask;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_ct_send_passthrough_cmd(uint8_t tl, uint8_t key_code, uint8_t key_state)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (tl > ESP_AVRC_TRANS_LABEL_MAX || key_state > ESP_AVRC_PT_CMD_STATE_RELEASED) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_CT;
|
|
msg.act = BTC_AVRC_CTRL_API_SND_PTCMD_EVT;
|
|
|
|
btc_avrc_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_args_t));
|
|
|
|
arg.pt_cmd.tl = tl;
|
|
arg.pt_cmd.key_code = key_code;
|
|
arg.pt_cmd.key_state = key_state;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
/*********************************************************************************************/
|
|
/** following is the API of AVRCP target role **/
|
|
/*********************************************************************************************/
|
|
|
|
esp_err_t esp_avrc_tg_register_callback(esp_avrc_tg_cb_t callback)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if (callback == NULL) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
btc_profile_cb_set(BTC_PID_AVRC_TG, callback);
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_avrc_tg_init(void)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_TG;
|
|
msg.act = BTC_AVRC_TG_API_INIT_EVT;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, NULL, 0, NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_avrc_tg_deinit(void)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_TG;
|
|
msg.act = BTC_AVRC_TG_API_DEINIT_EVT;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, NULL, 0, NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
bool esp_avrc_psth_bit_mask_operation(esp_avrc_bit_mask_op_t op, esp_avrc_psth_bit_mask_t *psth,
|
|
esp_avrc_pt_cmd_t cmd)
|
|
{
|
|
if (!psth ||
|
|
cmd > ESP_AVRC_PT_CMD_VENDOR) {
|
|
return false;
|
|
}
|
|
|
|
uint16_t *p = &psth->bits[(uint8_t)cmd >> 4];
|
|
uint16_t mask = (uint16_t)1 << ((uint8_t)cmd & 0x0F);
|
|
switch (op) {
|
|
case ESP_AVRC_BIT_MASK_OP_SET:
|
|
*p |= mask;
|
|
break;
|
|
case ESP_AVRC_BIT_MASK_OP_CLEAR:
|
|
*p &= ~mask;
|
|
break;
|
|
case ESP_AVRC_BIT_MASK_OP_TEST:
|
|
return (*p & mask);
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
esp_err_t esp_avrc_tg_get_psth_cmd_filter(esp_avrc_psth_filter_t filter, esp_avrc_psth_bit_mask_t *cmd_set)
|
|
{
|
|
if ((esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) ||
|
|
(! btc_avrc_tg_init_p())) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
if (filter >= ESP_AVRC_PSTH_FILTER_SUPPORT_MAX ||
|
|
cmd_set == NULL) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
if (filter == ESP_AVRC_PSTH_FILTER_ALLOWED_CMD) {
|
|
const uint16_t *allowed_cmd_set = btc_avrc_tg_get_allowed_command();
|
|
memcpy(cmd_set, allowed_cmd_set, sizeof(esp_avrc_psth_bit_mask_t));
|
|
} else if (filter == ESP_AVRC_PSTH_FILTER_SUPPORTED_CMD) {
|
|
const uint16_t *supported_cmd_set = btc_avrc_tg_get_supported_command();
|
|
memcpy(cmd_set, supported_cmd_set, sizeof(esp_avrc_psth_bit_mask_t));
|
|
} else {
|
|
}
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_avrc_tg_set_psth_cmd_filter(esp_avrc_psth_filter_t filter, const esp_avrc_psth_bit_mask_t *cmd_set)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
if (filter >= ESP_AVRC_PSTH_FILTER_SUPPORT_MAX ||
|
|
cmd_set == NULL) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
if (filter == ESP_AVRC_PSTH_FILTER_ALLOWED_CMD) {
|
|
return ESP_ERR_NOT_SUPPORTED;
|
|
}
|
|
if (filter == ESP_AVRC_PSTH_FILTER_SUPPORTED_CMD) {
|
|
bool allowed = btc_avrc_tg_check_supported_command(cmd_set->bits);
|
|
if (!allowed) {
|
|
return ESP_ERR_NOT_SUPPORTED;
|
|
}
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_TG;
|
|
msg.act = BTC_AVRC_TG_API_SET_PSTH_SUPPORTED_CMD_EVT;
|
|
|
|
btc_avrc_tg_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_tg_args_t));
|
|
arg.set_psth_cmd = (uint16_t *)cmd_set->bits;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_tg_args_t),
|
|
btc_avrc_tg_arg_deep_copy);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
} else {
|
|
return ESP_FAIL;
|
|
}
|
|
}
|
|
|
|
esp_err_t esp_avrc_tg_get_rn_evt_cap(esp_avrc_rn_evt_cap_t cap, esp_avrc_rn_evt_cap_mask_t *evt_set)
|
|
{
|
|
if ((esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) ||
|
|
(! btc_avrc_tg_init_p())) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
if (cap >= ESP_AVRC_RN_CAP_MAX ||
|
|
evt_set == NULL) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
if (cap == ESP_AVRC_RN_CAP_ALLOWED_EVT) {
|
|
evt_set->bits = btc_avrc_tg_get_rn_allowed_evt();
|
|
} else if (cap == ESP_AVRC_RN_CAP_SUPPORTED_EVT) {
|
|
evt_set->bits = btc_avrc_tg_get_rn_supported_evt();
|
|
} else {
|
|
}
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_avrc_tg_set_rn_evt_cap(const esp_avrc_rn_evt_cap_mask_t *evt_set)
|
|
{
|
|
if (esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
if (evt_set == NULL) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
bool allowed = btc_avrc_tg_check_rn_supported_evt(evt_set->bits);
|
|
if (!allowed) {
|
|
return ESP_ERR_NOT_SUPPORTED;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_TG;
|
|
msg.act = BTC_AVRC_TG_API_SET_RN_SUPPORTED_EVT;
|
|
|
|
btc_avrc_tg_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_tg_args_t));
|
|
|
|
arg.set_rn_evt = evt_set->bits;
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_tg_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
}
|
|
|
|
bool esp_avrc_rn_evt_bit_mask_operation(esp_avrc_bit_mask_op_t op, esp_avrc_rn_evt_cap_mask_t *events,
|
|
esp_avrc_rn_event_ids_t event_id)
|
|
{
|
|
if (!events ||
|
|
event_id >= ESP_AVRC_RN_MAX_EVT) {
|
|
return false;
|
|
}
|
|
|
|
uint16_t *p = &events->bits;
|
|
uint16_t mask = (uint16_t)1 << ((uint8_t)event_id & 0x0F);
|
|
switch (op) {
|
|
case ESP_AVRC_BIT_MASK_OP_SET:
|
|
*p |= mask;
|
|
break;
|
|
case ESP_AVRC_BIT_MASK_OP_CLEAR:
|
|
*p &= ~mask;
|
|
break;
|
|
case ESP_AVRC_BIT_MASK_OP_TEST:
|
|
return (*p & mask);
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
esp_err_t esp_avrc_tg_send_rn_rsp(esp_avrc_rn_event_ids_t event_id, esp_avrc_rn_rsp_t rsp,
|
|
esp_avrc_rn_param_t *param)
|
|
{
|
|
if ((esp_bluedroid_get_status() != ESP_BLUEDROID_STATUS_ENABLED) ||
|
|
(! btc_avrc_tg_connected_p())) {
|
|
return ESP_ERR_INVALID_STATE;
|
|
}
|
|
|
|
if ( ! btc_avrc_tg_rn_evt_supported((uint8_t)event_id)) {
|
|
return ESP_ERR_NOT_SUPPORTED;
|
|
}
|
|
|
|
btc_msg_t msg;
|
|
msg.sig = BTC_SIG_API_CALL;
|
|
msg.pid = BTC_PID_AVRC_TG;
|
|
msg.act = BTC_AVRC_TG_API_SEND_RN_RSP_EVT;
|
|
|
|
btc_avrc_tg_args_t arg;
|
|
memset(&arg, 0, sizeof(btc_avrc_tg_args_t));
|
|
|
|
arg.rn_rsp.event_id = event_id;
|
|
arg.rn_rsp.rsp = rsp;
|
|
memcpy(&arg.rn_rsp.param, param, sizeof(esp_avrc_rn_param_t));
|
|
|
|
/* Switch to BTC context */
|
|
bt_status_t stat = btc_transfer_context(&msg, &arg, sizeof(btc_avrc_tg_args_t), NULL);
|
|
return (stat == BT_STATUS_SUCCESS) ? ESP_OK : ESP_FAIL;
|
|
|
|
}
|
|
|
|
#endif /* #if BTC_AV_INCLUDED */
|