/******************************************************************************
*
* Copyright 2016 The Android Open Source Project
* Copyright 2002-2012 Broadcom Corporation
*
* 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.
*
******************************************************************************/
/******************************************************************************
*
* this file contains the connection interface functions
*
******************************************************************************/
#include
#include
#include
#include
#include "bta/include/bta_sec_api.h"
#include "internal_include/bt_target.h"
#include "osi/include/allocator.h"
#include "stack/hid/hidd_int.h"
#include "stack/include/bt_hdr.h"
#include "stack/include/bt_psm_types.h"
#include "stack/include/l2cdefs.h"
#include "stack/include/stack_metrics_logging.h"
#include "types/raw_address.h"
using namespace bluetooth;
static void hidd_l2cif_connect_ind(const RawAddress& bd_addr, uint16_t cid,
uint16_t psm, uint8_t id);
static void hidd_l2cif_connect_cfm(uint16_t cid, uint16_t result);
static void hidd_l2cif_config_ind(uint16_t cid, tL2CAP_CFG_INFO* p_cfg);
static void hidd_l2cif_config_cfm(uint16_t cid, uint16_t result,
tL2CAP_CFG_INFO* p_cfg);
static void hidd_l2cif_disconnect_ind(uint16_t cid, bool ack_needed);
static void hidd_l2cif_disconnect(uint16_t cid);
static void hidd_l2cif_data_ind(uint16_t cid, BT_HDR* p_msg);
static void hidd_l2cif_cong_ind(uint16_t cid, bool congested);
static void hidd_on_l2cap_error(uint16_t lcid, uint16_t result);
static const tL2CAP_APPL_INFO dev_reg_info = {hidd_l2cif_connect_ind,
hidd_l2cif_connect_cfm,
hidd_l2cif_config_ind,
hidd_l2cif_config_cfm,
hidd_l2cif_disconnect_ind,
NULL,
hidd_l2cif_data_ind,
hidd_l2cif_cong_ind,
NULL,
hidd_on_l2cap_error,
NULL,
NULL,
NULL,
NULL};
/*******************************************************************************
*
* Function hidd_check_config_done
*
* Description Checks if connection is configured and callback can be fired
*
* Returns void
*
******************************************************************************/
static void hidd_check_config_done() {
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->conn_state == HID_CONN_STATE_CONFIG) {
p_hcon->conn_state = HID_CONN_STATE_CONNECTED;
hd_cb.device.state = HIDD_DEV_CONNECTED;
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_OPEN, 0, NULL);
// send outstanding data on intr
if (hd_cb.pending_data) {
if (L2CA_DataWrite(p_hcon->intr_cid, hd_cb.pending_data) !=
L2CAP_DW_SUCCESS) {
log::warn("Unable to write L2CAP data cid:{} len:{}", p_hcon->intr_cid,
hd_cb.pending_data->len);
}
hd_cb.pending_data = NULL;
}
}
}
/*******************************************************************************
*
* Function hidd_l2cif_connect_ind
*
* Description Handles incoming L2CAP connection (we act as server)
*
* Returns void
*
******************************************************************************/
static void hidd_l2cif_connect_ind(const RawAddress& bd_addr, uint16_t cid,
uint16_t psm, uint8_t id) {
tHID_DEV_DEV_CTB* p_dev;
bool accept = TRUE; // accept by default
log::verbose("psm={:04x} cid={:04x}", psm, cid);
p_dev = &hd_cb.device;
if (!hd_cb.allow_incoming) {
log::warn("incoming connections not allowed, rejecting");
if (!L2CA_DisconnectReq(cid)) {
log::warn("Unable to disconnect L2CAP peer:{} cid:{}", p_dev->addr, cid);
}
return;
}
tHID_CONN* p_hcon = &hd_cb.device.conn;
switch (psm) {
case HID_PSM_INTERRUPT:
if (p_hcon->ctrl_cid == 0) {
accept = FALSE;
log::warn("incoming INTR without CTRL, rejecting");
}
if (p_hcon->conn_state != HID_CONN_STATE_CONNECTING_INTR) {
accept = FALSE;
log::warn("incoming INTR in invalid state ({}), rejecting",
p_hcon->conn_state);
}
break;
case HID_PSM_CONTROL:
if (p_hcon->conn_state != HID_CONN_STATE_UNUSED) {
accept = FALSE;
log::warn("incoming CTRL in invalid state ({}), rejecting",
p_hcon->conn_state);
}
break;
default:
accept = FALSE;
log::error("received invalid PSM, rejecting");
break;
}
if (!accept) {
if (!L2CA_DisconnectReq(cid)) {
log::warn("Unable to disconnect L2CAP cid:{}", cid);
}
return;
}
// for CTRL we need to go through security and we reply in callback from there
if (psm == HID_PSM_CONTROL) {
// We are ready to accept connection from this device, since we aren't
// connected to anything and are in the correct state.
p_dev->in_use = TRUE;
p_dev->addr = bd_addr;
p_dev->state = HIDD_DEV_NO_CONN;
p_hcon->conn_flags = 0;
p_hcon->ctrl_cid = cid;
p_hcon->disc_reason = HID_SUCCESS;
p_hcon->conn_state = HID_CONN_STATE_CONNECTING_INTR;
return;
}
// for INTR we go directly to config state
p_hcon->conn_state = HID_CONN_STATE_CONFIG;
p_hcon->intr_cid = cid;
}
static void hidd_on_l2cap_error(uint16_t lcid, uint16_t result) {
log::warn("connection of config failed, now disconnect");
hidd_conn_disconnect();
// NOTE that the client doesn't care about error code
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE,
HID_L2CAP_CONN_FAIL | (uint32_t)result, NULL);
}
/*******************************************************************************
*
* Function hidd_l2cif_connect_cfm
*
* Description Handles L2CAP connection response (we act as client)
*
* Returns void
*
******************************************************************************/
static void hidd_l2cif_connect_cfm(uint16_t cid, uint16_t result) {
tHID_CONN* p_hcon = &hd_cb.device.conn;
log::verbose("cid={:04x} result={}", cid, result);
if (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid) {
log::warn("unknown cid");
return;
}
if (!(p_hcon->conn_flags & HID_CONN_FLAGS_IS_ORIG) ||
((cid == p_hcon->ctrl_cid) &&
(p_hcon->conn_state != HID_CONN_STATE_CONNECTING_CTRL)) ||
((cid == p_hcon->intr_cid) &&
(p_hcon->conn_state != HID_CONN_STATE_CONNECTING_INTR))) {
log::warn("unexpected");
return;
}
if (result != L2CAP_CONN_OK) {
log::error("invoked with non OK status");
return;
}
/* CTRL connect conf */
if (cid == p_hcon->ctrl_cid) {
p_hcon->disc_reason = HID_SUCCESS;
p_hcon->conn_state = HID_CONN_STATE_CONFIG;
} else {
p_hcon->conn_state = HID_CONN_STATE_CONFIG;
}
return;
}
/*******************************************************************************
*
* Function hidd_l2cif_config_ind
*
* Description Handles incoming L2CAP configuration request
*
* Returns void
*
******************************************************************************/
static void hidd_l2cif_config_ind(uint16_t cid, tL2CAP_CFG_INFO* p_cfg) {
log::verbose("cid={:04x}", cid);
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid) {
log::warn("unknown cid");
return;
}
if ((!p_cfg->mtu_present) || (p_cfg->mtu > HID_DEV_MTU_SIZE))
p_hcon->rem_mtu_size = HID_DEV_MTU_SIZE;
else
p_hcon->rem_mtu_size = p_cfg->mtu;
}
/*******************************************************************************
*
* Function hidd_l2cif_config_cfm
*
* Description Handles incoming L2CAP configuration response
*
* Returns void
*
******************************************************************************/
static void hidd_l2cif_config_cfm(uint16_t cid, uint16_t initiator,
tL2CAP_CFG_INFO* p_cfg) {
hidd_l2cif_config_ind(cid, p_cfg);
log::verbose("cid={:04x}", cid);
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid) {
log::warn("unknown cid");
return;
}
// update flags
if (cid == p_hcon->ctrl_cid) {
if (p_hcon->conn_flags & HID_CONN_FLAGS_IS_ORIG) {
p_hcon->disc_reason = HID_L2CAP_CONN_FAIL;
if ((p_hcon->intr_cid = L2CA_ConnectReqWithSecurity(
HID_PSM_INTERRUPT, hd_cb.device.addr,
BTA_SEC_AUTHENTICATE | BTA_SEC_ENCRYPT)) == 0) {
hidd_conn_disconnect();
p_hcon->conn_state = HID_CONN_STATE_UNUSED;
log::warn("could not start L2CAP connection for INTR");
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE,
HID_ERR_L2CAP_FAILED, NULL);
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_L2CAP_NOT_STARTED_INCOMING,
1);
return;
} else {
p_hcon->conn_state = HID_CONN_STATE_CONNECTING_INTR;
}
}
}
hidd_check_config_done();
}
/*******************************************************************************
*
* Function hidd_l2cif_disconnect_ind
*
* Description Handler incoming L2CAP disconnection request
*
* Returns void
*
******************************************************************************/
static void hidd_l2cif_disconnect_ind(uint16_t cid, bool ack_needed) {
log::verbose("cid={:04x} ack_needed={}", cid, ack_needed);
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
(p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
log::warn("unknown cid");
return;
}
p_hcon->conn_state = HID_CONN_STATE_DISCONNECTING;
if (cid == p_hcon->ctrl_cid)
p_hcon->ctrl_cid = 0;
else
p_hcon->intr_cid = 0;
if ((p_hcon->ctrl_cid == 0) && (p_hcon->intr_cid == 0)) {
log::verbose("INTR and CTRL disconnected");
// clean any outstanding data on intr
if (hd_cb.pending_data) {
osi_free(hd_cb.pending_data);
hd_cb.pending_data = NULL;
}
hd_cb.device.state = HIDD_DEV_NO_CONN;
p_hcon->conn_state = HID_CONN_STATE_UNUSED;
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE, p_hcon->disc_reason,
NULL);
}
}
static void hidd_l2cif_disconnect(uint16_t cid) {
if (!L2CA_DisconnectReq(cid)) {
log::warn("Unable to disconnect L2CAP cid:{}", cid);
}
log::verbose("cid={:04x}", cid);
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
(p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
log::warn("unknown cid");
return;
}
if (cid == p_hcon->ctrl_cid) {
p_hcon->ctrl_cid = 0;
} else {
p_hcon->intr_cid = 0;
// now disconnect CTRL
if (!L2CA_DisconnectReq(p_hcon->ctrl_cid)) {
log::warn("Unable to disconnect L2CAP cid:{}", p_hcon->ctrl_cid);
}
p_hcon->ctrl_cid = 0;
}
if ((p_hcon->ctrl_cid == 0) && (p_hcon->intr_cid == 0)) {
log::verbose("INTR and CTRL disconnected");
hd_cb.device.state = HIDD_DEV_NO_CONN;
p_hcon->conn_state = HID_CONN_STATE_UNUSED;
if (hd_cb.pending_vc_unplug) {
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_VC_UNPLUG,
p_hcon->disc_reason, NULL);
hd_cb.pending_vc_unplug = FALSE;
} else {
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE,
p_hcon->disc_reason, NULL);
}
}
}
/*******************************************************************************
*
* Function hidd_l2cif_cong_ind
*
* Description Handles L2CAP congestion status event
*
* Returns void
*
******************************************************************************/
static void hidd_l2cif_cong_ind(uint16_t cid, bool congested) {
log::verbose("cid={:04x} congested={}", cid, congested);
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
(p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
log::warn("unknown cid");
return;
}
if (congested) {
p_hcon->conn_flags |= HID_CONN_FLAGS_CONGESTED;
} else {
p_hcon->conn_flags &= ~HID_CONN_FLAGS_CONGESTED;
}
}
/*******************************************************************************
*
* Function hidd_l2cif_data_ind
*
* Description Handler incoming data on L2CAP channel
*
* Returns void
*
******************************************************************************/
static void hidd_l2cif_data_ind(uint16_t cid, BT_HDR* p_msg) {
uint8_t* p_data = (uint8_t*)(p_msg + 1) + p_msg->offset;
uint8_t msg_type, param;
bool err = FALSE;
log::verbose("cid={:04x}", cid);
if (p_msg->len < 1) {
log::error("Invalid data length, ignore");
osi_free(p_msg);
return;
}
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
(p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
log::warn("unknown cid");
osi_free(p_msg);
return;
}
msg_type = HID_GET_TRANS_FROM_HDR(*p_data);
param = HID_GET_PARAM_FROM_HDR(*p_data);
if (msg_type == HID_TRANS_DATA && cid == p_hcon->intr_cid) {
// skip HID header
p_msg->offset++;
p_msg->len--;
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_INTR_DATA, 0, p_msg);
return;
}
switch (msg_type) {
case HID_TRANS_GET_REPORT:
// at this stage we don't know if Report Id shall be included in request
// so we pass complete packet in callback and let other code analyze this
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_GET_REPORT,
!!(param & HID_PAR_GET_REP_BUFSIZE_FOLLOWS), p_msg);
break;
case HID_TRANS_SET_REPORT:
// as above
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_SET_REPORT, 0, p_msg);
break;
case HID_TRANS_GET_IDLE:
hidd_conn_send_data(HID_CHANNEL_CTRL, HID_TRANS_DATA,
HID_PAR_REP_TYPE_OTHER, hd_cb.device.idle_time, 0,
NULL);
osi_free(p_msg);
break;
case HID_TRANS_SET_IDLE:
if (p_msg->len != 2) {
log::error("invalid len ({}) set idle request received", p_msg->len);
err = TRUE;
} else {
hd_cb.device.idle_time = p_data[1];
log::verbose("idle_time = {}", hd_cb.device.idle_time);
if (hd_cb.device.idle_time) {
log::warn("idle_time of {} ms not supported by HID Device",
hd_cb.device.idle_time * 4);
err = TRUE;
}
}
if (!err) {
hidd_conn_send_data(0, HID_TRANS_HANDSHAKE,
HID_PAR_HANDSHAKE_RSP_SUCCESS, 0, 0, NULL);
} else {
hidd_conn_send_data(0, HID_TRANS_HANDSHAKE,
HID_PAR_HANDSHAKE_RSP_ERR_INVALID_PARAM, 0, 0,
NULL);
}
osi_free(p_msg);
break;
case HID_TRANS_GET_PROTOCOL:
hidd_conn_send_data(HID_CHANNEL_CTRL, HID_TRANS_DATA,
HID_PAR_REP_TYPE_OTHER, !hd_cb.device.boot_mode, 0,
NULL);
osi_free(p_msg);
break;
case HID_TRANS_SET_PROTOCOL:
hd_cb.device.boot_mode = !(param & HID_PAR_PROTOCOL_MASK);
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_SET_PROTOCOL,
param & HID_PAR_PROTOCOL_MASK, NULL);
hidd_conn_send_data(0, HID_TRANS_HANDSHAKE, HID_PAR_HANDSHAKE_RSP_SUCCESS,
0, 0, NULL);
osi_free(p_msg);
break;
case HID_TRANS_CONTROL:
switch (param) {
case HID_PAR_CONTROL_SUSPEND:
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_SUSPEND, 0, NULL);
break;
case HID_PAR_CONTROL_EXIT_SUSPEND:
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_EXIT_SUSPEND, 0,
NULL);
break;
case HID_PAR_CONTROL_VIRTUAL_CABLE_UNPLUG:
hidd_conn_disconnect();
// set flag so we can notify properly when disconnected
hd_cb.pending_vc_unplug = TRUE;
break;
}
osi_free(p_msg);
break;
case HID_TRANS_DATA:
default:
log::warn("got unsupported msg ({})", msg_type);
hidd_conn_send_data(0, HID_TRANS_HANDSHAKE,
HID_PAR_HANDSHAKE_RSP_ERR_UNSUPPORTED_REQ, 0, 0,
NULL);
osi_free(p_msg);
break;
}
}
/*******************************************************************************
*
* Function hidd_conn_reg
*
* Description Registers L2CAP channels
*
* Returns void
*
******************************************************************************/
tHID_STATUS hidd_conn_reg(void) {
log::verbose("");
memset(&hd_cb.l2cap_cfg, 0, sizeof(tL2CAP_CFG_INFO));
hd_cb.l2cap_cfg.mtu_present = TRUE;
hd_cb.l2cap_cfg.mtu = HID_DEV_MTU_SIZE;
memset(&hd_cb.l2cap_intr_cfg, 0, sizeof(tL2CAP_CFG_INFO));
hd_cb.l2cap_intr_cfg.mtu_present = TRUE;
hd_cb.l2cap_intr_cfg.mtu = HID_DEV_MTU_SIZE;
if (!L2CA_RegisterWithSecurity(
HID_PSM_CONTROL, dev_reg_info, false /* enable_snoop */, nullptr,
HID_DEV_MTU_SIZE, 0, BTA_SEC_AUTHENTICATE | BTA_SEC_ENCRYPT)) {
log::error("HID Control (device) registration failed");
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_L2CAP_FAILED_CONTROL,
1);
return (HID_ERR_L2CAP_FAILED);
}
if (!L2CA_RegisterWithSecurity(
HID_PSM_INTERRUPT, dev_reg_info, false /* enable_snoop */, nullptr,
HID_DEV_MTU_SIZE, 0, BTA_SEC_AUTHENTICATE | BTA_SEC_ENCRYPT)) {
L2CA_Deregister(HID_PSM_CONTROL);
log::error("HID Interrupt (device) registration failed");
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_L2CAP_FAILED_INTERRUPT,
1);
return (HID_ERR_L2CAP_FAILED);
}
return (HID_SUCCESS);
}
/*******************************************************************************
*
* Function hidd_conn_dereg
*
* Description Deregisters L2CAP channels
*
* Returns void
*
******************************************************************************/
void hidd_conn_dereg(void) {
log::verbose("");
L2CA_Deregister(HID_PSM_CONTROL);
L2CA_Deregister(HID_PSM_INTERRUPT);
}
/*******************************************************************************
*
* Function hidd_conn_initiate
*
* Description Initiates HID connection to plugged device
*
* Returns HID_SUCCESS
*
******************************************************************************/
tHID_STATUS hidd_conn_initiate(void) {
tHID_DEV_DEV_CTB* p_dev = &hd_cb.device;
log::verbose("");
if (!p_dev->in_use) {
log::warn("no virtual cable established");
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_NOT_REGISTERED_AT_INITIATE,
1);
return (HID_ERR_NOT_REGISTERED);
}
if (p_dev->conn.conn_state != HID_CONN_STATE_UNUSED) {
log::warn("connection already in progress");
log_counter_metrics(
android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_CONN_IN_PROCESS,
1);
return (HID_ERR_CONN_IN_PROCESS);
}
p_dev->conn.ctrl_cid = 0;
p_dev->conn.intr_cid = 0;
p_dev->conn.disc_reason = HID_L2CAP_CONN_FAIL;
p_dev->conn.conn_flags = HID_CONN_FLAGS_IS_ORIG;
/* Check if L2CAP started the connection process */
if ((p_dev->conn.ctrl_cid = L2CA_ConnectReqWithSecurity(
HID_PSM_CONTROL, p_dev->addr,
BTA_SEC_AUTHENTICATE | BTA_SEC_ENCRYPT)) == 0) {
log::warn("could not start L2CAP connection");
hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE, HID_ERR_L2CAP_FAILED,
NULL);
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_L2CAP_FAILED_INITIATE,
1);
} else {
p_dev->conn.conn_state = HID_CONN_STATE_CONNECTING_CTRL;
}
return (HID_SUCCESS);
}
/*******************************************************************************
*
* Function hidd_conn_disconnect
*
* Description Disconnects existing HID connection
*
* Returns HID_SUCCESS
*
******************************************************************************/
tHID_STATUS hidd_conn_disconnect(void) {
log::verbose("");
// clean any outstanding data on intr
if (hd_cb.pending_data) {
osi_free(hd_cb.pending_data);
hd_cb.pending_data = NULL;
}
tHID_CONN* p_hcon = &hd_cb.device.conn;
if ((p_hcon->ctrl_cid != 0) || (p_hcon->intr_cid != 0)) {
p_hcon->conn_state = HID_CONN_STATE_DISCONNECTING;
/* Set l2cap idle timeout to 0 (so ACL link is disconnected
* immediately after last channel is closed) */
if (!L2CA_SetIdleTimeoutByBdAddr(hd_cb.device.addr, 0,
BT_TRANSPORT_BR_EDR)) {
log::warn("Unable to set L2CAP idle timeout peer:{} transport:{}",
hd_cb.device.addr, BT_TRANSPORT_BR_EDR);
}
if (p_hcon->intr_cid) {
hidd_l2cif_disconnect(p_hcon->intr_cid);
} else if (p_hcon->ctrl_cid) {
hidd_l2cif_disconnect(p_hcon->ctrl_cid);
}
} else {
log::warn("already disconnected");
p_hcon->conn_state = HID_CONN_STATE_UNUSED;
}
return (HID_SUCCESS);
}
/*******************************************************************************
*
* Function hidd_conn_send_data
*
* Description Sends data to host
*
* Returns tHID_STATUS
*
******************************************************************************/
tHID_STATUS hidd_conn_send_data(uint8_t channel, uint8_t msg_type,
uint8_t param, uint8_t data, uint16_t len,
uint8_t* p_data) {
BT_HDR* p_buf;
uint8_t* p_out;
uint16_t cid;
uint16_t buf_size;
log::verbose("channel({}), msg_type({}), len({})", channel, msg_type, len);
tHID_CONN* p_hcon = &hd_cb.device.conn;
if (p_hcon->conn_flags & HID_CONN_FLAGS_CONGESTED) {
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_CONGESTED_AT_FLAG_CHECK,
1);
return HID_ERR_CONGESTED;
}
switch (msg_type) {
case HID_TRANS_HANDSHAKE:
case HID_TRANS_CONTROL:
cid = p_hcon->ctrl_cid;
buf_size = HID_CONTROL_BUF_SIZE;
break;
case HID_TRANS_DATA:
if (channel == HID_CHANNEL_CTRL) {
cid = p_hcon->ctrl_cid;
buf_size = HID_CONTROL_BUF_SIZE;
} else {
cid = p_hcon->intr_cid;
buf_size = HID_INTERRUPT_BUF_SIZE;
}
break;
default:
log_counter_metrics(
android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_INVALID_PARAM,
1);
return (HID_ERR_INVALID_PARAM);
}
p_buf = (BT_HDR*)osi_malloc(buf_size);
if (p_buf == NULL) {
log_counter_metrics(
android::bluetooth::CodePathCounterKeyEnum::HIDD_ERR_NO_RESOURCES, 1);
return (HID_ERR_NO_RESOURCES);
}
p_buf->offset = L2CAP_MIN_OFFSET;
p_out = (uint8_t*)(p_buf + 1) + p_buf->offset;
*p_out = HID_BUILD_HDR(msg_type, param);
p_out++;
p_buf->len = 1; // start with header only
// add report id prefix only if non-zero (which is reserved)
if (msg_type == HID_TRANS_DATA && (data || param == HID_PAR_REP_TYPE_OTHER)) {
*p_out = data; // report_id
p_out++;
p_buf->len++;
}
if (len > 0 && p_data != NULL) {
memcpy(p_out, p_data, len);
p_buf->len += len;
}
// check if connected
if (hd_cb.device.state != HIDD_DEV_CONNECTED) {
// for DATA on intr we hold transfer and try to reconnect
if (msg_type == HID_TRANS_DATA && cid == p_hcon->intr_cid) {
// drop previous data, we do not queue it for now
if (hd_cb.pending_data) {
osi_free(hd_cb.pending_data);
}
hd_cb.pending_data = p_buf;
if (hd_cb.device.conn.conn_state == HID_CONN_STATE_UNUSED) {
hidd_conn_initiate();
}
return HID_SUCCESS;
}
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_NO_CONNECTION_AT_SEND_DATA,
1);
return HID_ERR_NO_CONNECTION;
}
log::verbose("report sent");
if (!L2CA_DataWrite(cid, p_buf)) {
log_counter_metrics(android::bluetooth::CodePathCounterKeyEnum::
HIDD_ERR_CONGESTED_AT_DATA_WRITE,
1);
return (HID_ERR_CONGESTED);
}
return (HID_SUCCESS);
}