diff options
| author | Andre Guedes <andre.guedes@openbossa.org> | 2012-02-03 15:48:00 -0500 |
|---|---|---|
| committer | Johan Hedberg <johan.hedberg@intel.com> | 2012-02-13 10:01:34 -0500 |
| commit | 28b75a89480df99a17c8facd5c33985847d06bb6 (patch) | |
| tree | ddb050dd79624ed01e12d1d072f8be646a15815f | |
| parent | 7ba8b4be38e7c83b2b13333a82a0ecde921a7390 (diff) | |
Bluetooth: Add hci_le_scan()
We are not supposed to block in start_discovery() because
start_discovery code is running in write() syscall context
and this would block the write operation on the mgmt socket.
This way, we cannot directly call hci_do_le_scan() to scan
LE devices in start_discovery(). To overcome this issue a
derefered work (hdev->le_scan) was created so we can properly
call hci_do_le_scan().
The helper function hci_le_scan() simply set LE scan parameters
and queue hdev->le_scan work. The work is queued on system_long_wq
since it can sleep for a few seconds in the worst case (timeout).
Signed-off-by: Andre Guedes <andre.guedes@openbossa.org>
Acked-by: Marcel Holtmann <marcel@holtmann.org>
Signed-off-by: Johan Hedberg <johan.hedberg@intel.com>
| -rw-r--r-- | include/net/bluetooth/hci_core.h | 6 | ||||
| -rw-r--r-- | net/bluetooth/hci_core.c | 35 |
2 files changed, 41 insertions, 0 deletions
diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h index 3e70872bffea..7107790817a5 100644 --- a/include/net/bluetooth/hci_core.h +++ b/include/net/bluetooth/hci_core.h | |||
| @@ -126,6 +126,7 @@ struct le_scan_params { | |||
| 126 | u8 type; | 126 | u8 type; |
| 127 | u16 interval; | 127 | u16 interval; |
| 128 | u16 window; | 128 | u16 window; |
| 129 | int timeout; | ||
| 129 | }; | 130 | }; |
| 130 | 131 | ||
| 131 | #define NUM_REASSEMBLY 4 | 132 | #define NUM_REASSEMBLY 4 |
| @@ -269,6 +270,9 @@ struct hci_dev { | |||
| 269 | 270 | ||
| 270 | struct delayed_work le_scan_disable; | 271 | struct delayed_work le_scan_disable; |
| 271 | 272 | ||
| 273 | struct work_struct le_scan; | ||
| 274 | struct le_scan_params le_scan_params; | ||
| 275 | |||
| 272 | int (*open)(struct hci_dev *hdev); | 276 | int (*open)(struct hci_dev *hdev); |
| 273 | int (*close)(struct hci_dev *hdev); | 277 | int (*close)(struct hci_dev *hdev); |
| 274 | int (*flush)(struct hci_dev *hdev); | 278 | int (*flush)(struct hci_dev *hdev); |
| @@ -1033,5 +1037,7 @@ void hci_le_ltk_neg_reply(struct hci_conn *conn); | |||
| 1033 | 1037 | ||
| 1034 | int hci_do_inquiry(struct hci_dev *hdev, u8 length); | 1038 | int hci_do_inquiry(struct hci_dev *hdev, u8 length); |
| 1035 | int hci_cancel_inquiry(struct hci_dev *hdev); | 1039 | int hci_cancel_inquiry(struct hci_dev *hdev); |
| 1040 | int hci_le_scan(struct hci_dev *hdev, u8 type, u16 interval, u16 window, | ||
| 1041 | int timeout); | ||
| 1036 | 1042 | ||
| 1037 | #endif /* __HCI_CORE_H */ | 1043 | #endif /* __HCI_CORE_H */ |
diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c index ae86cdd80ac0..3d09f4b4ca68 100644 --- a/net/bluetooth/hci_core.c +++ b/net/bluetooth/hci_core.c | |||
| @@ -735,6 +735,8 @@ static int hci_dev_do_close(struct hci_dev *hdev) | |||
| 735 | { | 735 | { |
| 736 | BT_DBG("%s %p", hdev->name, hdev); | 736 | BT_DBG("%s %p", hdev->name, hdev); |
| 737 | 737 | ||
| 738 | cancel_work_sync(&hdev->le_scan); | ||
| 739 | |||
| 738 | hci_req_cancel(hdev, ENODEV); | 740 | hci_req_cancel(hdev, ENODEV); |
| 739 | hci_req_lock(hdev); | 741 | hci_req_lock(hdev); |
| 740 | 742 | ||
| @@ -1668,6 +1670,37 @@ static void le_scan_disable_work(struct work_struct *work) | |||
| 1668 | hci_send_cmd(hdev, HCI_OP_LE_SET_SCAN_ENABLE, sizeof(cp), &cp); | 1670 | hci_send_cmd(hdev, HCI_OP_LE_SET_SCAN_ENABLE, sizeof(cp), &cp); |
| 1669 | } | 1671 | } |
| 1670 | 1672 | ||
| 1673 | static void le_scan_work(struct work_struct *work) | ||
| 1674 | { | ||
| 1675 | struct hci_dev *hdev = container_of(work, struct hci_dev, le_scan); | ||
| 1676 | struct le_scan_params *param = &hdev->le_scan_params; | ||
| 1677 | |||
| 1678 | BT_DBG("%s", hdev->name); | ||
| 1679 | |||
| 1680 | hci_do_le_scan(hdev, param->type, param->interval, | ||
| 1681 | param->window, param->timeout); | ||
| 1682 | } | ||
| 1683 | |||
| 1684 | int hci_le_scan(struct hci_dev *hdev, u8 type, u16 interval, u16 window, | ||
| 1685 | int timeout) | ||
| 1686 | { | ||
| 1687 | struct le_scan_params *param = &hdev->le_scan_params; | ||
| 1688 | |||
| 1689 | BT_DBG("%s", hdev->name); | ||
| 1690 | |||
| 1691 | if (work_busy(&hdev->le_scan)) | ||
| 1692 | return -EINPROGRESS; | ||
| 1693 | |||
| 1694 | param->type = type; | ||
| 1695 | param->interval = interval; | ||
| 1696 | param->window = window; | ||
| 1697 | param->timeout = timeout; | ||
| 1698 | |||
| 1699 | queue_work(system_long_wq, &hdev->le_scan); | ||
| 1700 | |||
| 1701 | return 0; | ||
| 1702 | } | ||
| 1703 | |||
| 1671 | /* Register HCI device */ | 1704 | /* Register HCI device */ |
| 1672 | int hci_register_dev(struct hci_dev *hdev) | 1705 | int hci_register_dev(struct hci_dev *hdev) |
| 1673 | { | 1706 | { |
| @@ -1754,6 +1787,8 @@ int hci_register_dev(struct hci_dev *hdev) | |||
| 1754 | 1787 | ||
| 1755 | atomic_set(&hdev->promisc, 0); | 1788 | atomic_set(&hdev->promisc, 0); |
| 1756 | 1789 | ||
| 1790 | INIT_WORK(&hdev->le_scan, le_scan_work); | ||
| 1791 | |||
| 1757 | INIT_DELAYED_WORK(&hdev->le_scan_disable, le_scan_disable_work); | 1792 | INIT_DELAYED_WORK(&hdev->le_scan_disable, le_scan_disable_work); |
| 1758 | 1793 | ||
| 1759 | write_unlock(&hci_dev_list_lock); | 1794 | write_unlock(&hci_dev_list_lock); |
