Legacy remote devices (v 2.0-) can establish link-level encryption as part of ACL connection establishment (ie., security mode 3). The host controller indicates link-level encryption is established in this case with the Encryption_Enabled flag of the Connection Complete event. This two-part fix first sets the correct link state for this condition and, second, bypasses additional auth + encrypt for subsequent security level elevations (up to but not including BT_SECURITY_HIGH). Signed-off-by: Peter Hurley <peter@xxxxxxxxxxxxxxxxxx> --- net/bluetooth/hci_conn.c | 13 +++++++++++-- net/bluetooth/hci_event.c | 8 ++++++++ 2 files changed, 19 insertions(+), 2 deletions(-) diff --git a/net/bluetooth/hci_conn.c b/net/bluetooth/hci_conn.c index 63d9949..617b51d 100644 --- a/net/bluetooth/hci_conn.c +++ b/net/bluetooth/hci_conn.c @@ -541,9 +541,18 @@ static int hci_conn_auth(struct hci_conn *conn, __u8 sec_level, __u8 auth_type) if (conn->pending_sec_level > sec_level) sec_level = conn->pending_sec_level; - if (sec_level > conn->sec_level) + if (sec_level > conn->sec_level) { conn->pending_sec_level = sec_level; - else if (conn->link_mode & HCI_LM_AUTH) + /* Legacy security mode 3 remote devices that are already + * auth'd do not need to re-auth unless promoting to + * BT_SECURITY_HIGH */ + if (!(conn->hdev->ssp_mode > 0 && conn->ssp_mode > 0) && + (sec_level != BT_SECURITY_HIGH) && + (conn->link_mode & HCI_LM_AUTH)) { + conn->sec_level = sec_level; + return 1; + } + } else if (conn->link_mode & HCI_LM_AUTH) return 1; /* Make sure we preserve an existing MITM requirement*/ diff --git a/net/bluetooth/hci_event.c b/net/bluetooth/hci_event.c index 77930aa..9e89b65 100644 --- a/net/bluetooth/hci_event.c +++ b/net/bluetooth/hci_event.c @@ -1311,6 +1311,14 @@ static inline void hci_conn_complete_evt(struct hci_dev *hdev, struct sk_buff *s if (test_bit(HCI_ENCRYPT, &hdev->flags)) conn->link_mode |= HCI_LM_ENCRYPT; + /* Indicate the correct link state for legacy sec mode 3 + * remote devices for which the LM has already established + * link encryption */ + if (conn->type == ACL_LINK && ev->encr_mode == 0x01 && + !(hdev->ssp_mode > 0 && conn->ssp_mode > 0) && + conn->pending_sec_level != BT_SECURITY_HIGH) + conn->link_mode |= HCI_LM_AUTH | HCI_LM_ENCRYPT; + /* Get remote features */ if (conn->type == ACL_LINK) { struct hci_cp_read_remote_features cp; -- 1.7.4.1 ��.n��������+%������w��{.n�����{����^n�r������&��z�ޗ�zf���h���~����������_��+v���)ߣ�