mirror of
https://github.com/hardkernel/linux.git
synced 2026-06-11 13:27:06 +09:00
Bluetooth: 6lowpan: fix BDADDR_LE vs ADDR_LE_DEV address type confusion
[ Upstream commit b454505bf57a2e4f5d49951d4deb03730a9348d9 ]
Bluetooth 6lowpan.c confuses BDADDR_LE and ADDR_LE_DEV address types,
e.g. debugfs "connect" command takes the former, and "disconnect" and
"connect" to already connected device take the latter. This is due to
using same value both for l2cap_chan_connect and hci_conn_hash_lookup_le
which take different dst_type values.
Fix address type passed to hci_conn_hash_lookup_le().
Retain the debugfs API difference between "connect" and "disconnect"
commands since it's been like this since 2015 and nobody apparently
complained.
Fixes: f5ad4ffceb ("Bluetooth: 6lowpan: Use hci_conn_hash_lookup_le() when possible")
Reviewed-by: Paul Menzel <pmenzel@molgen.mpg.de>
Signed-off-by: Pauli Virtanen <pav@iki.fi>
Signed-off-by: Luiz Augusto von Dentz <luiz.von.dentz@intel.com>
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
c24ac6cfe4
commit
2a9ff4086c
@@ -956,10 +956,11 @@ static struct l2cap_chan *bt_6lowpan_listen(void)
|
||||
}
|
||||
|
||||
static int get_l2cap_conn(char *buf, bdaddr_t *addr, u8 *addr_type,
|
||||
struct l2cap_conn **conn)
|
||||
struct l2cap_conn **conn, bool disconnect)
|
||||
{
|
||||
struct hci_conn *hcon;
|
||||
struct hci_dev *hdev;
|
||||
int le_addr_type;
|
||||
int n;
|
||||
|
||||
n = sscanf(buf, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx %hhu",
|
||||
@@ -970,13 +971,32 @@ static int get_l2cap_conn(char *buf, bdaddr_t *addr, u8 *addr_type,
|
||||
if (n < 7)
|
||||
return -EINVAL;
|
||||
|
||||
if (disconnect) {
|
||||
/* The "disconnect" debugfs command has used different address
|
||||
* type constants than "connect" since 2015. Let's retain that
|
||||
* for now even though it's obviously buggy...
|
||||
*/
|
||||
*addr_type += 1;
|
||||
}
|
||||
|
||||
switch (*addr_type) {
|
||||
case BDADDR_LE_PUBLIC:
|
||||
le_addr_type = ADDR_LE_DEV_PUBLIC;
|
||||
break;
|
||||
case BDADDR_LE_RANDOM:
|
||||
le_addr_type = ADDR_LE_DEV_RANDOM;
|
||||
break;
|
||||
default:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* The LE_PUBLIC address type is ignored because of BDADDR_ANY */
|
||||
hdev = hci_get_route(addr, BDADDR_ANY, BDADDR_LE_PUBLIC);
|
||||
if (!hdev)
|
||||
return -ENOENT;
|
||||
|
||||
hci_dev_lock(hdev);
|
||||
hcon = hci_conn_hash_lookup_le(hdev, addr, *addr_type);
|
||||
hcon = hci_conn_hash_lookup_le(hdev, addr, le_addr_type);
|
||||
hci_dev_unlock(hdev);
|
||||
hci_dev_put(hdev);
|
||||
|
||||
@@ -1103,7 +1123,7 @@ static ssize_t lowpan_control_write(struct file *fp,
|
||||
buf[buf_size] = '\0';
|
||||
|
||||
if (memcmp(buf, "connect ", 8) == 0) {
|
||||
ret = get_l2cap_conn(&buf[8], &addr, &addr_type, &conn);
|
||||
ret = get_l2cap_conn(&buf[8], &addr, &addr_type, &conn, false);
|
||||
if (ret == -EINVAL)
|
||||
return ret;
|
||||
|
||||
@@ -1140,7 +1160,7 @@ static ssize_t lowpan_control_write(struct file *fp,
|
||||
}
|
||||
|
||||
if (memcmp(buf, "disconnect ", 11) == 0) {
|
||||
ret = get_l2cap_conn(&buf[11], &addr, &addr_type, &conn);
|
||||
ret = get_l2cap_conn(&buf[11], &addr, &addr_type, &conn, true);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
||||
|
||||
Reference in New Issue
Block a user