Commit 01b1cb87 authored by Johan Hedberg's avatar Johan Hedberg Committed by Marcel Holtmann

Bluetooth: Run page scan updates through hdev->req_workqueue

Since Add/Remove Device perform the page scan updates independently
from the HCI command completion we've introduced a potential race when
multiple mgmt commands are queued. Doing the page scan updates through
the req_workqueue ensures that the state changes are performed in a
race-free manner.

At the same time, to make the request helper more widely usable,
extend it to also cover Inquiry Scan changes since those are behind
the same HCI command. This is also reflected in the new name of the
API as well as the work struct name.
Signed-off-by: default avatarJohan Hedberg <johan.hedberg@intel.com>
Signed-off-by: default avatarMarcel Holtmann <marcel@holtmann.org>
parent ad2c8c73
...@@ -329,6 +329,7 @@ struct hci_dev { ...@@ -329,6 +329,7 @@ struct hci_dev {
struct work_struct discov_update; struct work_struct discov_update;
struct work_struct bg_scan_update; struct work_struct bg_scan_update;
struct work_struct scan_update;
struct delayed_work le_scan_disable; struct delayed_work le_scan_disable;
struct delayed_work le_scan_restart; struct delayed_work le_scan_restart;
......
...@@ -2176,7 +2176,7 @@ static void hci_conn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb) ...@@ -2176,7 +2176,7 @@ static void hci_conn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
hci_send_cmd(hdev, HCI_OP_READ_REMOTE_FEATURES, hci_send_cmd(hdev, HCI_OP_READ_REMOTE_FEATURES,
sizeof(cp), &cp); sizeof(cp), &cp);
hci_update_page_scan(hdev); hci_req_update_scan(hdev);
} }
/* Set packet type for incoming connection */ /* Set packet type for incoming connection */
...@@ -2362,7 +2362,7 @@ static void hci_disconn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb) ...@@ -2362,7 +2362,7 @@ static void hci_disconn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
if (test_bit(HCI_CONN_FLUSH_KEY, &conn->flags)) if (test_bit(HCI_CONN_FLUSH_KEY, &conn->flags))
hci_remove_link_key(hdev, &conn->dst); hci_remove_link_key(hdev, &conn->dst);
hci_update_page_scan(hdev); hci_req_update_scan(hdev);
} }
params = hci_conn_params_lookup(hdev, &conn->dst, conn->dst_type); params = hci_conn_params_lookup(hdev, &conn->dst, conn->dst_type);
......
...@@ -637,7 +637,7 @@ static bool disconnected_whitelist_entries(struct hci_dev *hdev) ...@@ -637,7 +637,7 @@ static bool disconnected_whitelist_entries(struct hci_dev *hdev)
return false; return false;
} }
void __hci_update_page_scan(struct hci_request *req) void __hci_req_update_scan(struct hci_request *req)
{ {
struct hci_dev *hdev = req->hdev; struct hci_dev *hdev = req->hdev;
u8 scan; u8 scan;
...@@ -657,22 +657,29 @@ void __hci_update_page_scan(struct hci_request *req) ...@@ -657,22 +657,29 @@ void __hci_update_page_scan(struct hci_request *req)
else else
scan = SCAN_DISABLED; scan = SCAN_DISABLED;
if (test_bit(HCI_PSCAN, &hdev->flags) == !!(scan & SCAN_PAGE))
return;
if (hci_dev_test_flag(hdev, HCI_DISCOVERABLE)) if (hci_dev_test_flag(hdev, HCI_DISCOVERABLE))
scan |= SCAN_INQUIRY; scan |= SCAN_INQUIRY;
if (test_bit(HCI_PSCAN, &hdev->flags) == !!(scan & SCAN_PAGE) &&
test_bit(HCI_ISCAN, &hdev->flags) == !!(scan & SCAN_INQUIRY))
return;
hci_req_add(req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan); hci_req_add(req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
} }
void hci_update_page_scan(struct hci_dev *hdev) static int update_scan(struct hci_request *req, unsigned long opt)
{ {
struct hci_request req; hci_dev_lock(req->hdev);
__hci_req_update_scan(req);
hci_dev_unlock(req->hdev);
return 0;
}
hci_req_init(&req, hdev); static void scan_update_work(struct work_struct *work)
__hci_update_page_scan(&req); {
hci_req_run(&req, NULL); struct hci_dev *hdev = container_of(work, struct hci_dev, scan_update);
hci_req_sync(hdev, update_scan, 0, HCI_CMD_TIMEOUT, NULL);
} }
/* This function controls the background scanning based on hdev->pend_le_conns /* This function controls the background scanning based on hdev->pend_le_conns
...@@ -1270,6 +1277,7 @@ void hci_request_setup(struct hci_dev *hdev) ...@@ -1270,6 +1277,7 @@ void hci_request_setup(struct hci_dev *hdev)
{ {
INIT_WORK(&hdev->discov_update, discov_update); INIT_WORK(&hdev->discov_update, discov_update);
INIT_WORK(&hdev->bg_scan_update, bg_scan_update); INIT_WORK(&hdev->bg_scan_update, bg_scan_update);
INIT_WORK(&hdev->scan_update, scan_update_work);
INIT_DELAYED_WORK(&hdev->le_scan_disable, le_scan_disable_work); INIT_DELAYED_WORK(&hdev->le_scan_disable, le_scan_disable_work);
INIT_DELAYED_WORK(&hdev->le_scan_restart, le_scan_restart_work); INIT_DELAYED_WORK(&hdev->le_scan_restart, le_scan_restart_work);
} }
...@@ -1280,6 +1288,7 @@ void hci_request_cancel_all(struct hci_dev *hdev) ...@@ -1280,6 +1288,7 @@ void hci_request_cancel_all(struct hci_dev *hdev)
cancel_work_sync(&hdev->discov_update); cancel_work_sync(&hdev->discov_update);
cancel_work_sync(&hdev->bg_scan_update); cancel_work_sync(&hdev->bg_scan_update);
cancel_work_sync(&hdev->scan_update);
cancel_delayed_work_sync(&hdev->le_scan_disable); cancel_delayed_work_sync(&hdev->le_scan_disable);
cancel_delayed_work_sync(&hdev->le_scan_restart); cancel_delayed_work_sync(&hdev->le_scan_restart);
} }
...@@ -61,8 +61,12 @@ void hci_req_add_le_passive_scan(struct hci_request *req); ...@@ -61,8 +61,12 @@ void hci_req_add_le_passive_scan(struct hci_request *req);
/* Returns true if HCI commands were queued */ /* Returns true if HCI commands were queued */
bool hci_req_stop_discovery(struct hci_request *req); bool hci_req_stop_discovery(struct hci_request *req);
void hci_update_page_scan(struct hci_dev *hdev); static inline void hci_req_update_scan(struct hci_dev *hdev)
void __hci_update_page_scan(struct hci_request *req); {
queue_work(hdev->req_workqueue, &hdev->scan_update);
}
void __hci_req_update_scan(struct hci_request *req);
int hci_update_random_address(struct hci_request *req, bool require_privacy, int hci_update_random_address(struct hci_request *req, bool require_privacy,
u8 *own_addr_type); u8 *own_addr_type);
......
...@@ -1810,7 +1810,7 @@ static void set_discoverable_complete(struct hci_dev *hdev, u8 status, ...@@ -1810,7 +1810,7 @@ static void set_discoverable_complete(struct hci_dev *hdev, u8 status,
* entries. * entries.
*/ */
hci_req_init(&req, hdev); hci_req_init(&req, hdev);
__hci_update_page_scan(&req); __hci_req_update_scan(&req);
update_class(&req); update_class(&req);
hci_req_run(&req, NULL); hci_req_run(&req, NULL);
...@@ -2058,7 +2058,7 @@ static void set_connectable_complete(struct hci_dev *hdev, u8 status, ...@@ -2058,7 +2058,7 @@ static void set_connectable_complete(struct hci_dev *hdev, u8 status,
if (conn_changed || discov_changed) { if (conn_changed || discov_changed) {
new_settings(hdev, cmd->sk); new_settings(hdev, cmd->sk);
hci_update_page_scan(hdev); hci_req_update_scan(hdev);
if (discov_changed) if (discov_changed)
mgmt_update_adv_data(hdev); mgmt_update_adv_data(hdev);
hci_update_background_scan(hdev); hci_update_background_scan(hdev);
...@@ -2092,7 +2092,7 @@ static int set_connectable_update_settings(struct hci_dev *hdev, ...@@ -2092,7 +2092,7 @@ static int set_connectable_update_settings(struct hci_dev *hdev,
return err; return err;
if (changed) { if (changed) {
hci_update_page_scan(hdev); hci_req_update_scan(hdev);
hci_update_background_scan(hdev); hci_update_background_scan(hdev);
return new_settings(hdev, sk); return new_settings(hdev, sk);
} }
...@@ -5041,7 +5041,7 @@ static int set_bredr(struct sock *sk, struct hci_dev *hdev, void *data, u16 len) ...@@ -5041,7 +5041,7 @@ static int set_bredr(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
hci_req_init(&req, hdev); hci_req_init(&req, hdev);
write_fast_connectable(&req, false); write_fast_connectable(&req, false);
__hci_update_page_scan(&req); __hci_req_update_scan(&req);
/* Since only the advertising data flags will change, there /* Since only the advertising data flags will change, there
* is no need to update the scan response data. * is no need to update the scan response data.
...@@ -5927,7 +5927,7 @@ static int add_device(struct sock *sk, struct hci_dev *hdev, ...@@ -5927,7 +5927,7 @@ static int add_device(struct sock *sk, struct hci_dev *hdev,
if (err) if (err)
goto unlock; goto unlock;
hci_update_page_scan(hdev); hci_req_update_scan(hdev);
goto added; goto added;
} }
...@@ -6024,7 +6024,7 @@ static int remove_device(struct sock *sk, struct hci_dev *hdev, ...@@ -6024,7 +6024,7 @@ static int remove_device(struct sock *sk, struct hci_dev *hdev,
goto unlock; goto unlock;
} }
hci_update_page_scan(hdev); hci_req_update_scan(hdev);
device_removed(sk, hdev, &cp->addr.bdaddr, device_removed(sk, hdev, &cp->addr.bdaddr,
cp->addr.type); cp->addr.type);
...@@ -6089,7 +6089,7 @@ static int remove_device(struct sock *sk, struct hci_dev *hdev, ...@@ -6089,7 +6089,7 @@ static int remove_device(struct sock *sk, struct hci_dev *hdev,
kfree(b); kfree(b);
} }
hci_update_page_scan(hdev); hci_req_update_scan(hdev);
list_for_each_entry_safe(p, tmp, &hdev->le_conn_params, list) { list_for_each_entry_safe(p, tmp, &hdev->le_conn_params, list) {
if (p->auto_connect == HCI_AUTO_CONN_DISABLED) if (p->auto_connect == HCI_AUTO_CONN_DISABLED)
...@@ -7397,7 +7397,7 @@ static int powered_update_hci(struct hci_dev *hdev) ...@@ -7397,7 +7397,7 @@ static int powered_update_hci(struct hci_dev *hdev)
write_fast_connectable(&req, true); write_fast_connectable(&req, true);
else else
write_fast_connectable(&req, false); write_fast_connectable(&req, false);
__hci_update_page_scan(&req); __hci_req_update_scan(&req);
update_class(&req); update_class(&req);
update_name(&req); update_name(&req);
update_eir(&req); update_eir(&req);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment