diff options
| author | Johan Hedberg <johan.hedberg@nokia.com> | 2010-12-21 16:01:27 -0500 |
|---|---|---|
| committer | Gustavo F. Padovan <padovan@profusion.mobi> | 2010-12-22 19:58:07 -0500 |
| commit | 23bb57633df97ede067ea26f3cdc8a7ba2cd8109 (patch) | |
| tree | 3aa9a25b17da84975691af7159b030f810560e38 | |
| parent | c71e97bfaadfa727669fcfcf12301744fd169091 (diff) | |
Bluetooth: Fix __hci_request synchronization for hci_open_dev
The initialization function used by hci_open_dev (hci_init_req) sends
many different HCI commands. The __hci_request function should only
return when all of these commands have completed (or a timeout occurs).
Several of these commands cause hci_req_complete to be called which
causes __hci_request to return prematurely.
This patch fixes the issue by adding a new hdev->req_last_cmd variable
which is set during the initialization procedure. The hci_req_complete
function will no longer mark the request as complete until the command
matching hdev->req_last_cmd completes.
Signed-off-by: Johan Hedberg <johan.hedberg@nokia.com>
Acked-by: Marcel Holtmann <marcel@holtmann.org>
Signed-off-by: Gustavo F. Padovan <padovan@profusion.mobi>
| -rw-r--r-- | include/net/bluetooth/hci_core.h | 3 | ||||
| -rw-r--r-- | net/bluetooth/hci_core.c | 15 | ||||
| -rw-r--r-- | net/bluetooth/hci_event.c | 33 |
3 files changed, 37 insertions, 14 deletions
diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h index 3786ee83604e..a29feb01854e 100644 --- a/include/net/bluetooth/hci_core.h +++ b/include/net/bluetooth/hci_core.h | |||
| @@ -129,6 +129,7 @@ struct hci_dev { | |||
| 129 | wait_queue_head_t req_wait_q; | 129 | wait_queue_head_t req_wait_q; |
| 130 | __u32 req_status; | 130 | __u32 req_status; |
| 131 | __u32 req_result; | 131 | __u32 req_result; |
| 132 | __u16 req_last_cmd; | ||
| 132 | 133 | ||
| 133 | struct inquiry_cache inq_cache; | 134 | struct inquiry_cache inq_cache; |
| 134 | struct hci_conn_hash conn_hash; | 135 | struct hci_conn_hash conn_hash; |
| @@ -693,6 +694,6 @@ struct hci_sec_filter { | |||
| 693 | #define hci_req_lock(d) mutex_lock(&d->req_lock) | 694 | #define hci_req_lock(d) mutex_lock(&d->req_lock) |
| 694 | #define hci_req_unlock(d) mutex_unlock(&d->req_lock) | 695 | #define hci_req_unlock(d) mutex_unlock(&d->req_lock) |
| 695 | 696 | ||
| 696 | void hci_req_complete(struct hci_dev *hdev, int result); | 697 | void hci_req_complete(struct hci_dev *hdev, __u16 cmd, int result); |
| 697 | 698 | ||
| 698 | #endif /* __HCI_CORE_H */ | 699 | #endif /* __HCI_CORE_H */ |
diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c index 1a4ec97d5ac4..8b602d881fd7 100644 --- a/net/bluetooth/hci_core.c +++ b/net/bluetooth/hci_core.c | |||
| @@ -91,9 +91,16 @@ static void hci_notify(struct hci_dev *hdev, int event) | |||
| 91 | 91 | ||
| 92 | /* ---- HCI requests ---- */ | 92 | /* ---- HCI requests ---- */ |
| 93 | 93 | ||
| 94 | void hci_req_complete(struct hci_dev *hdev, int result) | 94 | void hci_req_complete(struct hci_dev *hdev, __u16 cmd, int result) |
| 95 | { | 95 | { |
| 96 | BT_DBG("%s result 0x%2.2x", hdev->name, result); | 96 | BT_DBG("%s command 0x%04x result 0x%2.2x", hdev->name, cmd, result); |
| 97 | |||
| 98 | /* If the request has set req_last_cmd (typical for multi-HCI | ||
| 99 | * command requests) check if the completed command matches | ||
| 100 | * this, and if not just return. Single HCI command requests | ||
| 101 | * typically leave req_last_cmd as 0 */ | ||
| 102 | if (hdev->req_last_cmd && cmd != hdev->req_last_cmd) | ||
| 103 | return; | ||
| 97 | 104 | ||
| 98 | if (hdev->req_status == HCI_REQ_PEND) { | 105 | if (hdev->req_status == HCI_REQ_PEND) { |
| 99 | hdev->req_result = result; | 106 | hdev->req_result = result; |
| @@ -149,7 +156,7 @@ static int __hci_request(struct hci_dev *hdev, void (*req)(struct hci_dev *hdev, | |||
| 149 | break; | 156 | break; |
| 150 | } | 157 | } |
| 151 | 158 | ||
| 152 | hdev->req_status = hdev->req_result = 0; | 159 | hdev->req_last_cmd = hdev->req_status = hdev->req_result = 0; |
| 153 | 160 | ||
| 154 | BT_DBG("%s end: err %d", hdev->name, err); | 161 | BT_DBG("%s end: err %d", hdev->name, err); |
| 155 | 162 | ||
| @@ -252,6 +259,8 @@ static void hci_init_req(struct hci_dev *hdev, unsigned long opt) | |||
| 252 | /* Connection accept timeout ~20 secs */ | 259 | /* Connection accept timeout ~20 secs */ |
| 253 | param = cpu_to_le16(0x7d00); | 260 | param = cpu_to_le16(0x7d00); |
| 254 | hci_send_cmd(hdev, HCI_OP_WRITE_CA_TIMEOUT, 2, ¶m); | 261 | hci_send_cmd(hdev, HCI_OP_WRITE_CA_TIMEOUT, 2, ¶m); |
| 262 | |||
| 263 | hdev->req_last_cmd = HCI_OP_WRITE_CA_TIMEOUT; | ||
| 255 | } | 264 | } |
| 256 | 265 | ||
| 257 | static void hci_scan_req(struct hci_dev *hdev, unsigned long opt) | 266 | static void hci_scan_req(struct hci_dev *hdev, unsigned long opt) |
diff --git a/net/bluetooth/hci_event.c b/net/bluetooth/hci_event.c index 8923b36a67a2..38100170d380 100644 --- a/net/bluetooth/hci_event.c +++ b/net/bluetooth/hci_event.c | |||
| @@ -58,7 +58,7 @@ static void hci_cc_inquiry_cancel(struct hci_dev *hdev, struct sk_buff *skb) | |||
| 58 | 58 | ||
| 59 | clear_bit(HCI_INQUIRY, &hdev->flags); | 59 | clear_bit(HCI_INQUIRY, &hdev->flags); |
| 60 | 60 | ||
| 61 | hci_req_complete(hdev, status); | 61 | hci_req_complete(hdev, HCI_OP_INQUIRY_CANCEL, status); |
| 62 | 62 | ||
| 63 | hci_conn_check_pending(hdev); | 63 | hci_conn_check_pending(hdev); |
| 64 | } | 64 | } |
| @@ -174,7 +174,7 @@ static void hci_cc_write_def_link_policy(struct hci_dev *hdev, struct sk_buff *s | |||
| 174 | if (!status) | 174 | if (!status) |
| 175 | hdev->link_policy = get_unaligned_le16(sent); | 175 | hdev->link_policy = get_unaligned_le16(sent); |
| 176 | 176 | ||
| 177 | hci_req_complete(hdev, status); | 177 | hci_req_complete(hdev, HCI_OP_WRITE_DEF_LINK_POLICY, status); |
| 178 | } | 178 | } |
| 179 | 179 | ||
| 180 | static void hci_cc_reset(struct hci_dev *hdev, struct sk_buff *skb) | 180 | static void hci_cc_reset(struct hci_dev *hdev, struct sk_buff *skb) |
| @@ -183,7 +183,7 @@ static void hci_cc_reset(struct hci_dev *hdev, struct sk_buff *skb) | |||
| 183 | 183 | ||
| 184 | BT_DBG("%s status 0x%x", hdev->name, status); | 184 | BT_DBG("%s status 0x%x", hdev->name, status); |
| 185 | 185 | ||
| 186 | hci_req_complete(hdev, status); | 186 | hci_req_complete(hdev, HCI_OP_RESET, status); |
| 187 | } | 187 | } |
| 188 | 188 | ||
| 189 | static void hci_cc_write_local_name(struct hci_dev *hdev, struct sk_buff *skb) | 189 | static void hci_cc_write_local_name(struct hci_dev *hdev, struct sk_buff *skb) |
| @@ -235,7 +235,7 @@ static void hci_cc_write_auth_enable(struct hci_dev *hdev, struct sk_buff *skb) | |||
| 235 | clear_bit(HCI_AUTH, &hdev->flags); | 235 | clear_bit(HCI_AUTH, &hdev->flags); |
| 236 | } | 236 | } |
| 237 | 237 | ||
| 238 | hci_req_complete(hdev, status); | 238 | hci_req_complete(hdev, HCI_OP_WRITE_AUTH_ENABLE, status); |
| 239 | } | 239 | } |
| 240 | 240 | ||
| 241 | static void hci_cc_write_encrypt_mode(struct hci_dev *hdev, struct sk_buff *skb) | 241 | static void hci_cc_write_encrypt_mode(struct hci_dev *hdev, struct sk_buff *skb) |
| @@ -258,7 +258,7 @@ static void hci_cc_write_encrypt_mode(struct hci_dev *hdev, struct sk_buff *skb) | |||
| 258 | clear_bit(HCI_ENCRYPT, &hdev->flags); | 258 | clear_bit(HCI_ENCRYPT, &hdev->flags); |
| 259 | } | 259 | } |
| 260 | 260 | ||
| 261 | hci_req_complete(hdev, status); | 261 | hci_req_complete(hdev, HCI_OP_WRITE_ENCRYPT_MODE, status); |
| 262 | } | 262 | } |
| 263 | 263 | ||
| 264 | static void hci_cc_write_scan_enable(struct hci_dev *hdev, struct sk_buff *skb) | 264 | static void hci_cc_write_scan_enable(struct hci_dev *hdev, struct sk_buff *skb) |
| @@ -285,7 +285,7 @@ static void hci_cc_write_scan_enable(struct hci_dev *hdev, struct sk_buff *skb) | |||
| 285 | set_bit(HCI_PSCAN, &hdev->flags); | 285 | set_bit(HCI_PSCAN, &hdev->flags); |
| 286 | } | 286 | } |
| 287 | 287 | ||
| 288 | hci_req_complete(hdev, status); | 288 | hci_req_complete(hdev, HCI_OP_WRITE_SCAN_ENABLE, status); |
| 289 | } | 289 | } |
| 290 | 290 | ||
| 291 | static void hci_cc_read_class_of_dev(struct hci_dev *hdev, struct sk_buff *skb) | 291 | static void hci_cc_read_class_of_dev(struct hci_dev *hdev, struct sk_buff *skb) |
| @@ -383,7 +383,7 @@ static void hci_cc_host_buffer_size(struct hci_dev *hdev, struct sk_buff *skb) | |||
| 383 | 383 | ||
| 384 | BT_DBG("%s status 0x%x", hdev->name, status); | 384 | BT_DBG("%s status 0x%x", hdev->name, status); |
| 385 | 385 | ||
| 386 | hci_req_complete(hdev, status); | 386 | hci_req_complete(hdev, HCI_OP_HOST_BUFFER_SIZE, status); |
| 387 | } | 387 | } |
| 388 | 388 | ||
| 389 | static void hci_cc_read_ssp_mode(struct hci_dev *hdev, struct sk_buff *skb) | 389 | static void hci_cc_read_ssp_mode(struct hci_dev *hdev, struct sk_buff *skb) |
| @@ -536,7 +536,16 @@ static void hci_cc_read_bd_addr(struct hci_dev *hdev, struct sk_buff *skb) | |||
| 536 | if (!rp->status) | 536 | if (!rp->status) |
| 537 | bacpy(&hdev->bdaddr, &rp->bdaddr); | 537 | bacpy(&hdev->bdaddr, &rp->bdaddr); |
| 538 | 538 | ||
| 539 | hci_req_complete(hdev, rp->status); | 539 | hci_req_complete(hdev, HCI_OP_READ_BD_ADDR, rp->status); |
| 540 | } | ||
| 541 | |||
| 542 | static void hci_cc_write_ca_timeout(struct hci_dev *hdev, struct sk_buff *skb) | ||
| 543 | { | ||
| 544 | __u8 status = *((__u8 *) skb->data); | ||
| 545 | |||
| 546 | BT_DBG("%s status 0x%x", hdev->name, status); | ||
| 547 | |||
| 548 | hci_req_complete(hdev, HCI_OP_WRITE_CA_TIMEOUT, status); | ||
| 540 | } | 549 | } |
| 541 | 550 | ||
| 542 | static inline void hci_cs_inquiry(struct hci_dev *hdev, __u8 status) | 551 | static inline void hci_cs_inquiry(struct hci_dev *hdev, __u8 status) |
| @@ -544,7 +553,7 @@ static inline void hci_cs_inquiry(struct hci_dev *hdev, __u8 status) | |||
| 544 | BT_DBG("%s status 0x%x", hdev->name, status); | 553 | BT_DBG("%s status 0x%x", hdev->name, status); |
| 545 | 554 | ||
| 546 | if (status) { | 555 | if (status) { |
| 547 | hci_req_complete(hdev, status); | 556 | hci_req_complete(hdev, HCI_OP_INQUIRY, status); |
| 548 | 557 | ||
| 549 | hci_conn_check_pending(hdev); | 558 | hci_conn_check_pending(hdev); |
| 550 | } else | 559 | } else |
| @@ -871,7 +880,7 @@ static inline void hci_inquiry_complete_evt(struct hci_dev *hdev, struct sk_buff | |||
| 871 | 880 | ||
| 872 | clear_bit(HCI_INQUIRY, &hdev->flags); | 881 | clear_bit(HCI_INQUIRY, &hdev->flags); |
| 873 | 882 | ||
| 874 | hci_req_complete(hdev, status); | 883 | hci_req_complete(hdev, HCI_OP_INQUIRY, status); |
| 875 | 884 | ||
| 876 | hci_conn_check_pending(hdev); | 885 | hci_conn_check_pending(hdev); |
| 877 | } | 886 | } |
| @@ -1379,6 +1388,10 @@ static inline void hci_cmd_complete_evt(struct hci_dev *hdev, struct sk_buff *sk | |||
| 1379 | hci_cc_read_bd_addr(hdev, skb); | 1388 | hci_cc_read_bd_addr(hdev, skb); |
| 1380 | break; | 1389 | break; |
| 1381 | 1390 | ||
| 1391 | case HCI_OP_WRITE_CA_TIMEOUT: | ||
| 1392 | hci_cc_write_ca_timeout(hdev, skb); | ||
| 1393 | break; | ||
| 1394 | |||
| 1382 | default: | 1395 | default: |
| 1383 | BT_DBG("%s opcode 0x%x", hdev->name, opcode); | 1396 | BT_DBG("%s opcode 0x%x", hdev->name, opcode); |
| 1384 | break; | 1397 | break; |
