Bluetooth: Fix issue with ADV_IND reports and auto-connection handling
When adding remote devices to the kernel using the Add Device management command, these devices are explicitly allowed to connect. This kind of incoming connections are possible even when the controller itself is not connectable. For BR/EDR this distinction is pretty simple since there is only one type of incoming connections. With LE this is not that simple anymore since there are ADV_IND and ADV_DIRECT_IND advertising events. The ADV_DIRECT_IND advertising events are send for incoming (slave initiated) connections only. And this is the only thing the kernel should allow when adding devices using action 0x01. This meaning of incoming connections is coming from BR/EDR and needs to be mapped to LE the same way. Supporting the auto-connection of devices using ADV_IND advertising events is an important feature as well. However it does not map to incoming connections. So introduce a new action 0x02 that allows the kernel to connect to devices using ADV_DIRECT_IND and in addition ADV_IND advertising reports. This difference is represented by the new HCI_AUTO_CONN_DIRECT value for only connecting to ADV_DIRECT_IND. For connection to ADV_IND and ADV_DIRECT_IND the old value HCI_AUTO_CONN_ALWAYS is used. Signed-off-by: Marcel Holtmann <marcel@holtmann.org> Signed-off-by: Johan Hedberg <johan.hedberg@intel.com>
This commit is contained in:
parent
cd4d567138
commit
4b9e7e7516
|
@ -458,6 +458,7 @@ struct hci_conn_params {
|
||||||
enum {
|
enum {
|
||||||
HCI_AUTO_CONN_DISABLED,
|
HCI_AUTO_CONN_DISABLED,
|
||||||
HCI_AUTO_CONN_REPORT,
|
HCI_AUTO_CONN_REPORT,
|
||||||
|
HCI_AUTO_CONN_DIRECT,
|
||||||
HCI_AUTO_CONN_ALWAYS,
|
HCI_AUTO_CONN_ALWAYS,
|
||||||
HCI_AUTO_CONN_LINK_LOSS,
|
HCI_AUTO_CONN_LINK_LOSS,
|
||||||
} auto_connect;
|
} auto_connect;
|
||||||
|
|
|
@ -3647,6 +3647,7 @@ int hci_conn_params_set(struct hci_dev *hdev, bdaddr_t *addr, u8 addr_type,
|
||||||
list_add(¶ms->action, &hdev->pend_le_reports);
|
list_add(¶ms->action, &hdev->pend_le_reports);
|
||||||
hci_update_background_scan(hdev);
|
hci_update_background_scan(hdev);
|
||||||
break;
|
break;
|
||||||
|
case HCI_AUTO_CONN_DIRECT:
|
||||||
case HCI_AUTO_CONN_ALWAYS:
|
case HCI_AUTO_CONN_ALWAYS:
|
||||||
if (!is_connected(hdev, addr, addr_type)) {
|
if (!is_connected(hdev, addr, addr_type)) {
|
||||||
list_add(¶ms->action, &hdev->pend_le_conns);
|
list_add(¶ms->action, &hdev->pend_le_conns);
|
||||||
|
|
|
@ -2259,6 +2259,7 @@ static void hci_disconn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
|
||||||
break;
|
break;
|
||||||
/* Fall through */
|
/* Fall through */
|
||||||
|
|
||||||
|
case HCI_AUTO_CONN_DIRECT:
|
||||||
case HCI_AUTO_CONN_ALWAYS:
|
case HCI_AUTO_CONN_ALWAYS:
|
||||||
list_del_init(¶ms->action);
|
list_del_init(¶ms->action);
|
||||||
list_add(¶ms->action, &hdev->pend_le_conns);
|
list_add(¶ms->action, &hdev->pend_le_conns);
|
||||||
|
@ -4251,6 +4252,7 @@ static void check_pending_le_conn(struct hci_dev *hdev, bdaddr_t *addr,
|
||||||
u8 addr_type, u8 adv_type)
|
u8 addr_type, u8 adv_type)
|
||||||
{
|
{
|
||||||
struct hci_conn *conn;
|
struct hci_conn *conn;
|
||||||
|
struct hci_conn_params *params;
|
||||||
|
|
||||||
/* If the event is not connectable don't proceed further */
|
/* If the event is not connectable don't proceed further */
|
||||||
if (adv_type != LE_ADV_IND && adv_type != LE_ADV_DIRECT_IND)
|
if (adv_type != LE_ADV_IND && adv_type != LE_ADV_DIRECT_IND)
|
||||||
|
@ -4269,9 +4271,32 @@ static void check_pending_le_conn(struct hci_dev *hdev, bdaddr_t *addr,
|
||||||
/* If we're not connectable only connect devices that we have in
|
/* If we're not connectable only connect devices that we have in
|
||||||
* our pend_le_conns list.
|
* our pend_le_conns list.
|
||||||
*/
|
*/
|
||||||
if (!hci_pend_le_action_lookup(&hdev->pend_le_conns, addr, addr_type))
|
params = hci_pend_le_action_lookup(&hdev->pend_le_conns,
|
||||||
|
addr, addr_type);
|
||||||
|
if (!params)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
switch (params->auto_connect) {
|
||||||
|
case HCI_AUTO_CONN_DIRECT:
|
||||||
|
/* Only devices advertising with ADV_DIRECT_IND are
|
||||||
|
* triggering a connection attempt. This is allowing
|
||||||
|
* incoming connections from slave devices.
|
||||||
|
*/
|
||||||
|
if (adv_type != LE_ADV_DIRECT_IND)
|
||||||
|
return;
|
||||||
|
break;
|
||||||
|
case HCI_AUTO_CONN_ALWAYS:
|
||||||
|
/* Devices advertising with ADV_IND or ADV_DIRECT_IND
|
||||||
|
* are triggering a connection attempt. This means
|
||||||
|
* that incoming connectioms from slave device are
|
||||||
|
* accepted and also outgoing connections to slave
|
||||||
|
* devices are established when found.
|
||||||
|
*/
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
conn = hci_connect_le(hdev, addr, addr_type, BT_SECURITY_LOW,
|
conn = hci_connect_le(hdev, addr, addr_type, BT_SECURITY_LOW,
|
||||||
HCI_LE_AUTOCONN_TIMEOUT, HCI_ROLE_MASTER);
|
HCI_LE_AUTOCONN_TIMEOUT, HCI_ROLE_MASTER);
|
||||||
if (!IS_ERR(conn))
|
if (!IS_ERR(conn))
|
||||||
|
|
|
@ -5271,7 +5271,7 @@ static int add_device(struct sock *sk, struct hci_dev *hdev,
|
||||||
MGMT_STATUS_INVALID_PARAMS,
|
MGMT_STATUS_INVALID_PARAMS,
|
||||||
&cp->addr, sizeof(cp->addr));
|
&cp->addr, sizeof(cp->addr));
|
||||||
|
|
||||||
if (cp->action != 0x00 && cp->action != 0x01)
|
if (cp->action != 0x00 && cp->action != 0x01 && cp->action != 0x02)
|
||||||
return cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
|
return cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
|
||||||
MGMT_STATUS_INVALID_PARAMS,
|
MGMT_STATUS_INVALID_PARAMS,
|
||||||
&cp->addr, sizeof(cp->addr));
|
&cp->addr, sizeof(cp->addr));
|
||||||
|
@ -5281,7 +5281,7 @@ static int add_device(struct sock *sk, struct hci_dev *hdev,
|
||||||
if (cp->addr.type == BDADDR_BREDR) {
|
if (cp->addr.type == BDADDR_BREDR) {
|
||||||
bool update_scan;
|
bool update_scan;
|
||||||
|
|
||||||
/* Only "connect" action supported for now */
|
/* Only incoming connections action is supported for now */
|
||||||
if (cp->action != 0x01) {
|
if (cp->action != 0x01) {
|
||||||
err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
|
err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
|
||||||
MGMT_STATUS_INVALID_PARAMS,
|
MGMT_STATUS_INVALID_PARAMS,
|
||||||
|
@ -5307,8 +5307,10 @@ static int add_device(struct sock *sk, struct hci_dev *hdev,
|
||||||
else
|
else
|
||||||
addr_type = ADDR_LE_DEV_RANDOM;
|
addr_type = ADDR_LE_DEV_RANDOM;
|
||||||
|
|
||||||
if (cp->action)
|
if (cp->action == 0x02)
|
||||||
auto_conn = HCI_AUTO_CONN_ALWAYS;
|
auto_conn = HCI_AUTO_CONN_ALWAYS;
|
||||||
|
else if (cp->action == 0x01)
|
||||||
|
auto_conn = HCI_AUTO_CONN_DIRECT;
|
||||||
else
|
else
|
||||||
auto_conn = HCI_AUTO_CONN_REPORT;
|
auto_conn = HCI_AUTO_CONN_REPORT;
|
||||||
|
|
||||||
|
@ -5870,6 +5872,7 @@ static void restart_le_actions(struct hci_dev *hdev)
|
||||||
list_del_init(&p->action);
|
list_del_init(&p->action);
|
||||||
|
|
||||||
switch (p->auto_connect) {
|
switch (p->auto_connect) {
|
||||||
|
case HCI_AUTO_CONN_DIRECT:
|
||||||
case HCI_AUTO_CONN_ALWAYS:
|
case HCI_AUTO_CONN_ALWAYS:
|
||||||
list_add(&p->action, &hdev->pend_le_conns);
|
list_add(&p->action, &hdev->pend_le_conns);
|
||||||
break;
|
break;
|
||||||
|
|
Loading…
Reference in New Issue