blob: 9cdcfd9f907eae87741c413d8ab97b8a760060d2 [file] [log] [blame]
YOSHIFUJI Hideaki8e87d142007-02-09 23:24:33 +09001/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07002 BlueZ - Bluetooth protocol stack for Linux
Ron Shaffer2d0a0342010-05-28 11:53:46 -04003 Copyright (c) 2000-2001, 2010, Code Aurora Forum. All rights reserved.
Linus Torvalds1da177e2005-04-16 15:20:36 -07004
5 Written 2000,2001 by Maxim Krasnyansky <maxk@qualcomm.com>
6
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License version 2 as
9 published by the Free Software Foundation;
10
11 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
12 OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
13 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.
14 IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY
YOSHIFUJI Hideaki8e87d142007-02-09 23:24:33 +090015 CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES
16 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
17 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
Linus Torvalds1da177e2005-04-16 15:20:36 -070018 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
19
YOSHIFUJI Hideaki8e87d142007-02-09 23:24:33 +090020 ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,
21 COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS
Linus Torvalds1da177e2005-04-16 15:20:36 -070022 SOFTWARE IS DISCLAIMED.
23*/
24
25/* Bluetooth HCI event handling. */
26
Linus Torvalds1da177e2005-04-16 15:20:36 -070027#include <linux/module.h>
28
29#include <linux/types.h>
30#include <linux/errno.h>
31#include <linux/kernel.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070032#include <linux/slab.h>
33#include <linux/poll.h>
34#include <linux/fcntl.h>
35#include <linux/init.h>
36#include <linux/skbuff.h>
37#include <linux/interrupt.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070038#include <net/sock.h>
39
Andrei Emeltchenko70f230202010-12-01 16:58:25 +020040#include <linux/uaccess.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070041#include <asm/unaligned.h>
42
43#include <net/bluetooth/bluetooth.h>
44#include <net/bluetooth/hci_core.h>
45
Linus Torvalds1da177e2005-04-16 15:20:36 -070046/* Handle HCI Event packets */
47
Marcel Holtmanna9de9242007-10-20 13:33:56 +020048static void hci_cc_inquiry_cancel(struct hci_dev *hdev, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -070049{
Marcel Holtmanna9de9242007-10-20 13:33:56 +020050 __u8 status = *((__u8 *) skb->data);
Linus Torvalds1da177e2005-04-16 15:20:36 -070051
Marcel Holtmanna9de9242007-10-20 13:33:56 +020052 BT_DBG("%s status 0x%x", hdev->name, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -070053
Andre Guedese6d465c2011-11-09 17:14:26 -030054 if (status) {
55 hci_dev_lock(hdev);
56 mgmt_stop_discovery_failed(hdev, status);
57 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +020058 return;
Andre Guedese6d465c2011-11-09 17:14:26 -030059 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070060
Andre Guedes89352e72011-11-04 14:16:53 -030061 clear_bit(HCI_INQUIRY, &hdev->flags);
62
Johan Hedberg56e5cb82011-11-08 20:40:16 +020063 hci_dev_lock(hdev);
Johan Hedbergff9ef572012-01-04 14:23:45 +020064 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
Johan Hedberg56e5cb82011-11-08 20:40:16 +020065 hci_dev_unlock(hdev);
Marcel Holtmann6bd57412006-11-18 22:14:22 +010066
Johan Hedberg23bb5762010-12-21 23:01:27 +020067 hci_req_complete(hdev, HCI_OP_INQUIRY_CANCEL, status);
Marcel Holtmann6bd57412006-11-18 22:14:22 +010068
Marcel Holtmanna9de9242007-10-20 13:33:56 +020069 hci_conn_check_pending(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -070070}
71
Marcel Holtmanna9de9242007-10-20 13:33:56 +020072static void hci_cc_exit_periodic_inq(struct hci_dev *hdev, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -070073{
Marcel Holtmanna9de9242007-10-20 13:33:56 +020074 __u8 status = *((__u8 *) skb->data);
75
76 BT_DBG("%s status 0x%x", hdev->name, status);
77
78 if (status)
79 return;
80
Marcel Holtmanna9de9242007-10-20 13:33:56 +020081 hci_conn_check_pending(hdev);
82}
83
84static void hci_cc_remote_name_req_cancel(struct hci_dev *hdev, struct sk_buff *skb)
85{
86 BT_DBG("%s", hdev->name);
87}
88
89static void hci_cc_role_discovery(struct hci_dev *hdev, struct sk_buff *skb)
90{
91 struct hci_rp_role_discovery *rp = (void *) skb->data;
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 struct hci_conn *conn;
Linus Torvalds1da177e2005-04-16 15:20:36 -070093
Marcel Holtmanna9de9242007-10-20 13:33:56 +020094 BT_DBG("%s status 0x%x", hdev->name, rp->status);
Linus Torvalds1da177e2005-04-16 15:20:36 -070095
Marcel Holtmanna9de9242007-10-20 13:33:56 +020096 if (rp->status)
97 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -070098
Marcel Holtmanna9de9242007-10-20 13:33:56 +020099 hci_dev_lock(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200101 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(rp->handle));
102 if (conn) {
103 if (rp->role)
104 conn->link_mode &= ~HCI_LM_MASTER;
105 else
106 conn->link_mode |= HCI_LM_MASTER;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700107 }
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200108
109 hci_dev_unlock(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700110}
111
Marcel Holtmanne4e8e372008-07-14 20:13:47 +0200112static void hci_cc_read_link_policy(struct hci_dev *hdev, struct sk_buff *skb)
113{
114 struct hci_rp_read_link_policy *rp = (void *) skb->data;
115 struct hci_conn *conn;
116
117 BT_DBG("%s status 0x%x", hdev->name, rp->status);
118
119 if (rp->status)
120 return;
121
122 hci_dev_lock(hdev);
123
124 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(rp->handle));
125 if (conn)
126 conn->link_policy = __le16_to_cpu(rp->policy);
127
128 hci_dev_unlock(hdev);
129}
130
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200131static void hci_cc_write_link_policy(struct hci_dev *hdev, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700132{
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200133 struct hci_rp_write_link_policy *rp = (void *) skb->data;
134 struct hci_conn *conn;
135 void *sent;
136
137 BT_DBG("%s status 0x%x", hdev->name, rp->status);
138
139 if (rp->status)
140 return;
141
142 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_LINK_POLICY);
143 if (!sent)
144 return;
145
146 hci_dev_lock(hdev);
147
148 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(rp->handle));
Marcel Holtmanne4e8e372008-07-14 20:13:47 +0200149 if (conn)
Harvey Harrison83985312008-05-02 16:25:46 -0700150 conn->link_policy = get_unaligned_le16(sent + 2);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200151
152 hci_dev_unlock(hdev);
153}
154
Marcel Holtmanne4e8e372008-07-14 20:13:47 +0200155static void hci_cc_read_def_link_policy(struct hci_dev *hdev, struct sk_buff *skb)
156{
157 struct hci_rp_read_def_link_policy *rp = (void *) skb->data;
158
159 BT_DBG("%s status 0x%x", hdev->name, rp->status);
160
161 if (rp->status)
162 return;
163
164 hdev->link_policy = __le16_to_cpu(rp->policy);
165}
166
167static void hci_cc_write_def_link_policy(struct hci_dev *hdev, struct sk_buff *skb)
168{
169 __u8 status = *((__u8 *) skb->data);
170 void *sent;
171
172 BT_DBG("%s status 0x%x", hdev->name, status);
173
174 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_DEF_LINK_POLICY);
175 if (!sent)
176 return;
177
178 if (!status)
179 hdev->link_policy = get_unaligned_le16(sent);
180
Johan Hedberg23bb5762010-12-21 23:01:27 +0200181 hci_req_complete(hdev, HCI_OP_WRITE_DEF_LINK_POLICY, status);
Marcel Holtmanne4e8e372008-07-14 20:13:47 +0200182}
183
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200184static void hci_cc_reset(struct hci_dev *hdev, struct sk_buff *skb)
185{
186 __u8 status = *((__u8 *) skb->data);
187
188 BT_DBG("%s status 0x%x", hdev->name, status);
189
Gustavo F. Padovan10572132011-03-16 15:36:29 -0300190 clear_bit(HCI_RESET, &hdev->flags);
191
Johan Hedberg23bb5762010-12-21 23:01:27 +0200192 hci_req_complete(hdev, HCI_OP_RESET, status);
Andre Guedesd23264a2011-11-25 20:53:38 -0300193
Johan Hedberga297e972012-02-21 17:55:47 +0200194 /* Reset all non-persistent flags */
Johan Hedberg9f8ce962012-03-02 03:06:04 +0200195 hdev->dev_flags &= ~(BIT(HCI_LE_SCAN) | BIT(HCI_PENDING_CLASS));
Andre Guedes69775ff2012-02-23 16:50:05 +0200196
197 hdev->discovery.state = DISCOVERY_STOPPED;
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200198}
199
200static void hci_cc_write_local_name(struct hci_dev *hdev, struct sk_buff *skb)
201{
202 __u8 status = *((__u8 *) skb->data);
203 void *sent;
204
205 BT_DBG("%s status 0x%x", hdev->name, status);
206
207 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_LOCAL_NAME);
208 if (!sent)
209 return;
210
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200211 hci_dev_lock(hdev);
212
Johan Hedbergf51d5b22012-02-22 18:17:32 +0200213 if (test_bit(HCI_MGMT, &hdev->dev_flags))
214 mgmt_set_local_name_complete(hdev, sent, status);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +0200215 else if (!status)
216 memcpy(hdev->dev_name, sent, HCI_MAX_NAME_LENGTH);
Johan Hedbergf51d5b22012-02-22 18:17:32 +0200217
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200218 hci_dev_unlock(hdev);
Johan Hedberg3159d382012-02-24 13:47:56 +0200219
220 hci_req_complete(hdev, HCI_OP_WRITE_LOCAL_NAME, status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200221}
222
223static void hci_cc_read_local_name(struct hci_dev *hdev, struct sk_buff *skb)
224{
225 struct hci_rp_read_local_name *rp = (void *) skb->data;
226
227 BT_DBG("%s status 0x%x", hdev->name, rp->status);
228
229 if (rp->status)
230 return;
231
Johan Hedbergdb99b5f2012-02-22 20:14:22 +0200232 if (test_bit(HCI_SETUP, &hdev->dev_flags))
233 memcpy(hdev->dev_name, rp->name, HCI_MAX_NAME_LENGTH);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200234}
235
236static void hci_cc_write_auth_enable(struct hci_dev *hdev, struct sk_buff *skb)
237{
238 __u8 status = *((__u8 *) skb->data);
239 void *sent;
240
241 BT_DBG("%s status 0x%x", hdev->name, status);
242
243 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_AUTH_ENABLE);
244 if (!sent)
245 return;
246
247 if (!status) {
248 __u8 param = *((__u8 *) sent);
249
250 if (param == AUTH_ENABLED)
251 set_bit(HCI_AUTH, &hdev->flags);
252 else
253 clear_bit(HCI_AUTH, &hdev->flags);
254 }
255
Johan Hedberg33ef95e2012-02-16 23:56:27 +0200256 if (test_bit(HCI_MGMT, &hdev->dev_flags))
257 mgmt_auth_enable_complete(hdev, status);
258
Johan Hedberg23bb5762010-12-21 23:01:27 +0200259 hci_req_complete(hdev, HCI_OP_WRITE_AUTH_ENABLE, status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200260}
261
262static void hci_cc_write_encrypt_mode(struct hci_dev *hdev, struct sk_buff *skb)
263{
264 __u8 status = *((__u8 *) skb->data);
265 void *sent;
266
267 BT_DBG("%s status 0x%x", hdev->name, status);
268
269 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_ENCRYPT_MODE);
270 if (!sent)
271 return;
272
273 if (!status) {
274 __u8 param = *((__u8 *) sent);
275
276 if (param)
277 set_bit(HCI_ENCRYPT, &hdev->flags);
278 else
279 clear_bit(HCI_ENCRYPT, &hdev->flags);
280 }
281
Johan Hedberg23bb5762010-12-21 23:01:27 +0200282 hci_req_complete(hdev, HCI_OP_WRITE_ENCRYPT_MODE, status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200283}
284
285static void hci_cc_write_scan_enable(struct hci_dev *hdev, struct sk_buff *skb)
286{
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200287 __u8 param, status = *((__u8 *) skb->data);
288 int old_pscan, old_iscan;
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200289 void *sent;
290
291 BT_DBG("%s status 0x%x", hdev->name, status);
292
293 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_SCAN_ENABLE);
294 if (!sent)
295 return;
296
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200297 param = *((__u8 *) sent);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200298
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200299 hci_dev_lock(hdev);
300
Johan Hedberg2d7cee52011-11-07 22:16:03 +0200301 if (status != 0) {
Johan Hedberg744cf192011-11-08 20:40:14 +0200302 mgmt_write_scan_failed(hdev, param, status);
Johan Hedberg2d7cee52011-11-07 22:16:03 +0200303 hdev->discov_timeout = 0;
304 goto done;
305 }
306
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200307 old_pscan = test_and_clear_bit(HCI_PSCAN, &hdev->flags);
308 old_iscan = test_and_clear_bit(HCI_ISCAN, &hdev->flags);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200309
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200310 if (param & SCAN_INQUIRY) {
311 set_bit(HCI_ISCAN, &hdev->flags);
312 if (!old_iscan)
Johan Hedberg744cf192011-11-08 20:40:14 +0200313 mgmt_discoverable(hdev, 1);
Johan Hedberg16ab91a2011-11-07 22:16:02 +0200314 if (hdev->discov_timeout > 0) {
315 int to = msecs_to_jiffies(hdev->discov_timeout * 1000);
316 queue_delayed_work(hdev->workqueue, &hdev->discov_off,
317 to);
318 }
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200319 } else if (old_iscan)
Johan Hedberg744cf192011-11-08 20:40:14 +0200320 mgmt_discoverable(hdev, 0);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200321
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200322 if (param & SCAN_PAGE) {
323 set_bit(HCI_PSCAN, &hdev->flags);
324 if (!old_pscan)
Johan Hedberg744cf192011-11-08 20:40:14 +0200325 mgmt_connectable(hdev, 1);
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200326 } else if (old_pscan)
Johan Hedberg744cf192011-11-08 20:40:14 +0200327 mgmt_connectable(hdev, 0);
Johan Hedberg36f7fc72011-11-04 00:17:45 +0200328
329done:
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200330 hci_dev_unlock(hdev);
Johan Hedberg23bb5762010-12-21 23:01:27 +0200331 hci_req_complete(hdev, HCI_OP_WRITE_SCAN_ENABLE, status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200332}
333
334static void hci_cc_read_class_of_dev(struct hci_dev *hdev, struct sk_buff *skb)
335{
336 struct hci_rp_read_class_of_dev *rp = (void *) skb->data;
337
338 BT_DBG("%s status 0x%x", hdev->name, rp->status);
339
340 if (rp->status)
341 return;
342
343 memcpy(hdev->dev_class, rp->dev_class, 3);
344
345 BT_DBG("%s class 0x%.2x%.2x%.2x", hdev->name,
346 hdev->dev_class[2], hdev->dev_class[1], hdev->dev_class[0]);
347}
348
349static void hci_cc_write_class_of_dev(struct hci_dev *hdev, struct sk_buff *skb)
350{
351 __u8 status = *((__u8 *) skb->data);
352 void *sent;
353
354 BT_DBG("%s status 0x%x", hdev->name, status);
355
356 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_CLASS_OF_DEV);
357 if (!sent)
358 return;
359
Marcel Holtmann7f9a9032012-02-22 18:38:01 +0100360 hci_dev_lock(hdev);
361
362 if (status == 0)
363 memcpy(hdev->dev_class, sent, 3);
364
365 if (test_bit(HCI_MGMT, &hdev->dev_flags))
366 mgmt_set_class_of_dev_complete(hdev, sent, status);
367
368 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200369}
370
371static void hci_cc_read_voice_setting(struct hci_dev *hdev, struct sk_buff *skb)
372{
373 struct hci_rp_read_voice_setting *rp = (void *) skb->data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700374 __u16 setting;
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200375
376 BT_DBG("%s status 0x%x", hdev->name, rp->status);
377
378 if (rp->status)
379 return;
380
381 setting = __le16_to_cpu(rp->voice_setting);
382
Marcel Holtmannf383f272008-07-14 20:13:47 +0200383 if (hdev->voice_setting == setting)
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200384 return;
385
386 hdev->voice_setting = setting;
387
388 BT_DBG("%s voice setting 0x%04x", hdev->name, setting);
389
Gustavo F. Padovan3c547112011-12-14 22:58:44 -0200390 if (hdev->notify)
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200391 hdev->notify(hdev, HCI_NOTIFY_VOICE_SETTING);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200392}
393
394static void hci_cc_write_voice_setting(struct hci_dev *hdev, struct sk_buff *skb)
395{
396 __u8 status = *((__u8 *) skb->data);
Marcel Holtmannf383f272008-07-14 20:13:47 +0200397 __u16 setting;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700398 void *sent;
399
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200400 BT_DBG("%s status 0x%x", hdev->name, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700401
Marcel Holtmannf383f272008-07-14 20:13:47 +0200402 if (status)
403 return;
404
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200405 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_VOICE_SETTING);
406 if (!sent)
407 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700408
Marcel Holtmannf383f272008-07-14 20:13:47 +0200409 setting = get_unaligned_le16(sent);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700410
Marcel Holtmannf383f272008-07-14 20:13:47 +0200411 if (hdev->voice_setting == setting)
412 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700413
Marcel Holtmannf383f272008-07-14 20:13:47 +0200414 hdev->voice_setting = setting;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700415
Marcel Holtmannf383f272008-07-14 20:13:47 +0200416 BT_DBG("%s voice setting 0x%04x", hdev->name, setting);
417
Gustavo F. Padovan3c547112011-12-14 22:58:44 -0200418 if (hdev->notify)
Marcel Holtmannf383f272008-07-14 20:13:47 +0200419 hdev->notify(hdev, HCI_NOTIFY_VOICE_SETTING);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700420}
421
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200422static void hci_cc_host_buffer_size(struct hci_dev *hdev, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700423{
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200424 __u8 status = *((__u8 *) skb->data);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700425
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200426 BT_DBG("%s status 0x%x", hdev->name, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700427
Johan Hedberg23bb5762010-12-21 23:01:27 +0200428 hci_req_complete(hdev, HCI_OP_HOST_BUFFER_SIZE, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700429}
430
Marcel Holtmann333140b2008-07-14 20:13:48 +0200431static void hci_cc_write_ssp_mode(struct hci_dev *hdev, struct sk_buff *skb)
432{
433 __u8 status = *((__u8 *) skb->data);
434 void *sent;
435
436 BT_DBG("%s status 0x%x", hdev->name, status);
437
Marcel Holtmann333140b2008-07-14 20:13:48 +0200438 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_SSP_MODE);
439 if (!sent)
440 return;
441
Johan Hedberged2c4ee2012-02-17 00:56:28 +0200442 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedbergc0ecddc2012-02-22 12:38:31 +0200443 mgmt_ssp_enable_complete(hdev, *((u8 *) sent), status);
444 else if (!status) {
445 if (*((u8 *) sent))
446 set_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
447 else
448 clear_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
449 }
Marcel Holtmann333140b2008-07-14 20:13:48 +0200450}
451
Johan Hedbergd5859e22011-01-25 01:19:58 +0200452static u8 hci_get_inquiry_mode(struct hci_dev *hdev)
453{
454 if (hdev->features[6] & LMP_EXT_INQ)
455 return 2;
456
457 if (hdev->features[3] & LMP_RSSI_INQ)
458 return 1;
459
460 if (hdev->manufacturer == 11 && hdev->hci_rev == 0x00 &&
461 hdev->lmp_subver == 0x0757)
462 return 1;
463
464 if (hdev->manufacturer == 15) {
465 if (hdev->hci_rev == 0x03 && hdev->lmp_subver == 0x6963)
466 return 1;
467 if (hdev->hci_rev == 0x09 && hdev->lmp_subver == 0x6963)
468 return 1;
469 if (hdev->hci_rev == 0x00 && hdev->lmp_subver == 0x6965)
470 return 1;
471 }
472
473 if (hdev->manufacturer == 31 && hdev->hci_rev == 0x2005 &&
474 hdev->lmp_subver == 0x1805)
475 return 1;
476
477 return 0;
478}
479
480static void hci_setup_inquiry_mode(struct hci_dev *hdev)
481{
482 u8 mode;
483
484 mode = hci_get_inquiry_mode(hdev);
485
486 hci_send_cmd(hdev, HCI_OP_WRITE_INQUIRY_MODE, 1, &mode);
487}
488
489static void hci_setup_event_mask(struct hci_dev *hdev)
490{
491 /* The second byte is 0xff instead of 0x9f (two reserved bits
492 * disabled) since a Broadcom 1.2 dongle doesn't respond to the
493 * command otherwise */
494 u8 events[8] = { 0xff, 0xff, 0xfb, 0xff, 0x00, 0x00, 0x00, 0x00 };
495
Ville Tervo6de6c182011-05-27 11:16:21 +0300496 /* CSR 1.1 dongles does not accept any bitfield so don't try to set
497 * any event mask for pre 1.2 devices */
Andrei Emeltchenko5a13b092011-12-01 14:33:28 +0200498 if (hdev->hci_ver < BLUETOOTH_VER_1_2)
Ville Tervo6de6c182011-05-27 11:16:21 +0300499 return;
500
501 events[4] |= 0x01; /* Flow Specification Complete */
502 events[4] |= 0x02; /* Inquiry Result with RSSI */
503 events[4] |= 0x04; /* Read Remote Extended Features Complete */
504 events[5] |= 0x08; /* Synchronous Connection Complete */
505 events[5] |= 0x10; /* Synchronous Connection Changed */
Johan Hedbergd5859e22011-01-25 01:19:58 +0200506
507 if (hdev->features[3] & LMP_RSSI_INQ)
508 events[4] |= 0x04; /* Inquiry Result with RSSI */
509
510 if (hdev->features[5] & LMP_SNIFF_SUBR)
511 events[5] |= 0x20; /* Sniff Subrating */
512
513 if (hdev->features[5] & LMP_PAUSE_ENC)
514 events[5] |= 0x80; /* Encryption Key Refresh Complete */
515
516 if (hdev->features[6] & LMP_EXT_INQ)
517 events[5] |= 0x40; /* Extended Inquiry Result */
518
519 if (hdev->features[6] & LMP_NO_FLUSH)
520 events[7] |= 0x01; /* Enhanced Flush Complete */
521
522 if (hdev->features[7] & LMP_LSTO)
523 events[6] |= 0x80; /* Link Supervision Timeout Changed */
524
525 if (hdev->features[6] & LMP_SIMPLE_PAIR) {
526 events[6] |= 0x01; /* IO Capability Request */
527 events[6] |= 0x02; /* IO Capability Response */
528 events[6] |= 0x04; /* User Confirmation Request */
529 events[6] |= 0x08; /* User Passkey Request */
530 events[6] |= 0x10; /* Remote OOB Data Request */
531 events[6] |= 0x20; /* Simple Pairing Complete */
532 events[7] |= 0x04; /* User Passkey Notification */
533 events[7] |= 0x08; /* Keypress Notification */
534 events[7] |= 0x10; /* Remote Host Supported
535 * Features Notification */
536 }
537
538 if (hdev->features[4] & LMP_LE)
539 events[7] |= 0x20; /* LE Meta-Event */
540
541 hci_send_cmd(hdev, HCI_OP_SET_EVENT_MASK, sizeof(events), events);
542}
543
544static void hci_setup(struct hci_dev *hdev)
545{
Andrei Emeltchenkoe61ef492011-12-19 16:31:27 +0200546 if (hdev->dev_type != HCI_BREDR)
547 return;
548
Johan Hedbergd5859e22011-01-25 01:19:58 +0200549 hci_setup_event_mask(hdev);
550
Andrei Emeltchenkod095c1e2011-12-01 14:33:27 +0200551 if (hdev->hci_ver > BLUETOOTH_VER_1_1)
Johan Hedbergd5859e22011-01-25 01:19:58 +0200552 hci_send_cmd(hdev, HCI_OP_READ_LOCAL_COMMANDS, 0, NULL);
553
Johan Hedberg54d04db2012-02-22 15:47:48 +0200554 if (hdev->features[6] & LMP_SIMPLE_PAIR) {
555 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags)) {
556 u8 mode = 0x01;
557 hci_send_cmd(hdev, HCI_OP_WRITE_SSP_MODE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300558 sizeof(mode), &mode);
Johan Hedberg54d04db2012-02-22 15:47:48 +0200559 } else {
560 struct hci_cp_write_eir cp;
561
562 memset(hdev->eir, 0, sizeof(hdev->eir));
563 memset(&cp, 0, sizeof(cp));
564
565 hci_send_cmd(hdev, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
566 }
Johan Hedbergd5859e22011-01-25 01:19:58 +0200567 }
568
569 if (hdev->features[3] & LMP_RSSI_INQ)
570 hci_setup_inquiry_mode(hdev);
571
572 if (hdev->features[7] & LMP_INQ_TX_PWR)
573 hci_send_cmd(hdev, HCI_OP_READ_INQ_RSP_TX_POWER, 0, NULL);
Andre Guedes971e3a42011-06-30 19:20:52 -0300574
575 if (hdev->features[7] & LMP_EXTFEATURES) {
576 struct hci_cp_read_local_ext_features cp;
577
578 cp.page = 0x01;
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300579 hci_send_cmd(hdev, HCI_OP_READ_LOCAL_EXT_FEATURES, sizeof(cp),
580 &cp);
Andre Guedes971e3a42011-06-30 19:20:52 -0300581 }
Andre Guedese6100a22011-06-30 19:20:54 -0300582
Johan Hedberg47990ea2012-02-22 11:58:37 +0200583 if (test_bit(HCI_LINK_SECURITY, &hdev->dev_flags)) {
584 u8 enable = 1;
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300585 hci_send_cmd(hdev, HCI_OP_WRITE_AUTH_ENABLE, sizeof(enable),
586 &enable);
Johan Hedberg47990ea2012-02-22 11:58:37 +0200587 }
Johan Hedbergd5859e22011-01-25 01:19:58 +0200588}
589
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200590static void hci_cc_read_local_version(struct hci_dev *hdev, struct sk_buff *skb)
591{
592 struct hci_rp_read_local_version *rp = (void *) skb->data;
593
594 BT_DBG("%s status 0x%x", hdev->name, rp->status);
595
596 if (rp->status)
Andrei Emeltchenko28b8df72012-02-24 12:45:44 +0200597 goto done;
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200598
599 hdev->hci_ver = rp->hci_ver;
Marcel Holtmanne4e8e372008-07-14 20:13:47 +0200600 hdev->hci_rev = __le16_to_cpu(rp->hci_rev);
Johan Hedbergd5859e22011-01-25 01:19:58 +0200601 hdev->lmp_ver = rp->lmp_ver;
Marcel Holtmanne4e8e372008-07-14 20:13:47 +0200602 hdev->manufacturer = __le16_to_cpu(rp->manufacturer);
Johan Hedbergd5859e22011-01-25 01:19:58 +0200603 hdev->lmp_subver = __le16_to_cpu(rp->lmp_subver);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200604
605 BT_DBG("%s manufacturer %d hci ver %d:%d", hdev->name,
606 hdev->manufacturer,
607 hdev->hci_ver, hdev->hci_rev);
Johan Hedbergd5859e22011-01-25 01:19:58 +0200608
609 if (test_bit(HCI_INIT, &hdev->flags))
610 hci_setup(hdev);
Andrei Emeltchenko28b8df72012-02-24 12:45:44 +0200611
612done:
613 hci_req_complete(hdev, HCI_OP_READ_LOCAL_VERSION, rp->status);
Johan Hedbergd5859e22011-01-25 01:19:58 +0200614}
615
616static void hci_setup_link_policy(struct hci_dev *hdev)
617{
618 u16 link_policy = 0;
619
620 if (hdev->features[0] & LMP_RSWITCH)
621 link_policy |= HCI_LP_RSWITCH;
622 if (hdev->features[0] & LMP_HOLD)
623 link_policy |= HCI_LP_HOLD;
624 if (hdev->features[0] & LMP_SNIFF)
625 link_policy |= HCI_LP_SNIFF;
626 if (hdev->features[1] & LMP_PARK)
627 link_policy |= HCI_LP_PARK;
628
629 link_policy = cpu_to_le16(link_policy);
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300630 hci_send_cmd(hdev, HCI_OP_WRITE_DEF_LINK_POLICY, sizeof(link_policy),
631 &link_policy);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200632}
633
634static void hci_cc_read_local_commands(struct hci_dev *hdev, struct sk_buff *skb)
635{
636 struct hci_rp_read_local_commands *rp = (void *) skb->data;
637
638 BT_DBG("%s status 0x%x", hdev->name, rp->status);
639
640 if (rp->status)
Johan Hedbergd5859e22011-01-25 01:19:58 +0200641 goto done;
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200642
643 memcpy(hdev->commands, rp->commands, sizeof(hdev->commands));
Johan Hedbergd5859e22011-01-25 01:19:58 +0200644
645 if (test_bit(HCI_INIT, &hdev->flags) && (hdev->commands[5] & 0x10))
646 hci_setup_link_policy(hdev);
647
648done:
649 hci_req_complete(hdev, HCI_OP_READ_LOCAL_COMMANDS, rp->status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200650}
651
652static void hci_cc_read_local_features(struct hci_dev *hdev, struct sk_buff *skb)
653{
654 struct hci_rp_read_local_features *rp = (void *) skb->data;
655
656 BT_DBG("%s status 0x%x", hdev->name, rp->status);
657
658 if (rp->status)
659 return;
660
661 memcpy(hdev->features, rp->features, 8);
662
663 /* Adjust default settings according to features
664 * supported by device. */
665
666 if (hdev->features[0] & LMP_3SLOT)
667 hdev->pkt_type |= (HCI_DM3 | HCI_DH3);
668
669 if (hdev->features[0] & LMP_5SLOT)
670 hdev->pkt_type |= (HCI_DM5 | HCI_DH5);
671
672 if (hdev->features[1] & LMP_HV2) {
673 hdev->pkt_type |= (HCI_HV2);
674 hdev->esco_type |= (ESCO_HV2);
675 }
676
677 if (hdev->features[1] & LMP_HV3) {
678 hdev->pkt_type |= (HCI_HV3);
679 hdev->esco_type |= (ESCO_HV3);
680 }
681
682 if (hdev->features[3] & LMP_ESCO)
683 hdev->esco_type |= (ESCO_EV3);
684
685 if (hdev->features[4] & LMP_EV4)
686 hdev->esco_type |= (ESCO_EV4);
687
688 if (hdev->features[4] & LMP_EV5)
689 hdev->esco_type |= (ESCO_EV5);
690
Marcel Holtmannefc76882009-02-06 09:13:37 +0100691 if (hdev->features[5] & LMP_EDR_ESCO_2M)
692 hdev->esco_type |= (ESCO_2EV3);
693
694 if (hdev->features[5] & LMP_EDR_ESCO_3M)
695 hdev->esco_type |= (ESCO_3EV3);
696
697 if (hdev->features[5] & LMP_EDR_3S_ESCO)
698 hdev->esco_type |= (ESCO_2EV5 | ESCO_3EV5);
699
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200700 BT_DBG("%s features 0x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x", hdev->name,
701 hdev->features[0], hdev->features[1],
702 hdev->features[2], hdev->features[3],
703 hdev->features[4], hdev->features[5],
704 hdev->features[6], hdev->features[7]);
705}
706
Johan Hedberg8f984df2012-02-28 01:07:22 +0200707static void hci_set_le_support(struct hci_dev *hdev)
708{
709 struct hci_cp_write_le_host_supported cp;
710
711 memset(&cp, 0, sizeof(cp));
712
713 if (enable_le && test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
714 cp.le = 1;
715 cp.simul = !!(hdev->features[6] & LMP_SIMUL_LE_BR);
716 }
717
718 if (cp.le != !!(hdev->host_features[0] & LMP_HOST_LE))
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300719 hci_send_cmd(hdev, HCI_OP_WRITE_LE_HOST_SUPPORTED, sizeof(cp),
720 &cp);
Johan Hedberg8f984df2012-02-28 01:07:22 +0200721}
722
Andre Guedes971e3a42011-06-30 19:20:52 -0300723static void hci_cc_read_local_ext_features(struct hci_dev *hdev,
724 struct sk_buff *skb)
725{
726 struct hci_rp_read_local_ext_features *rp = (void *) skb->data;
727
728 BT_DBG("%s status 0x%x", hdev->name, rp->status);
729
730 if (rp->status)
Johan Hedberg8f984df2012-02-28 01:07:22 +0200731 goto done;
Andre Guedes971e3a42011-06-30 19:20:52 -0300732
Andre Guedesb5b32b62011-12-30 10:34:04 -0300733 switch (rp->page) {
734 case 0:
735 memcpy(hdev->features, rp->features, 8);
736 break;
737 case 1:
738 memcpy(hdev->host_features, rp->features, 8);
739 break;
740 }
Andre Guedes971e3a42011-06-30 19:20:52 -0300741
Johan Hedberg8f984df2012-02-28 01:07:22 +0200742 if (test_bit(HCI_INIT, &hdev->flags) && hdev->features[4] & LMP_LE)
743 hci_set_le_support(hdev);
744
745done:
Andre Guedes971e3a42011-06-30 19:20:52 -0300746 hci_req_complete(hdev, HCI_OP_READ_LOCAL_EXT_FEATURES, rp->status);
747}
748
Andrei Emeltchenko1e89cff2011-11-24 14:52:02 +0200749static void hci_cc_read_flow_control_mode(struct hci_dev *hdev,
750 struct sk_buff *skb)
751{
752 struct hci_rp_read_flow_control_mode *rp = (void *) skb->data;
753
754 BT_DBG("%s status 0x%x", hdev->name, rp->status);
755
756 if (rp->status)
757 return;
758
759 hdev->flow_ctl_mode = rp->mode;
760
761 hci_req_complete(hdev, HCI_OP_READ_FLOW_CONTROL_MODE, rp->status);
762}
763
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200764static void hci_cc_read_buffer_size(struct hci_dev *hdev, struct sk_buff *skb)
765{
766 struct hci_rp_read_buffer_size *rp = (void *) skb->data;
767
768 BT_DBG("%s status 0x%x", hdev->name, rp->status);
769
770 if (rp->status)
771 return;
772
773 hdev->acl_mtu = __le16_to_cpu(rp->acl_mtu);
774 hdev->sco_mtu = rp->sco_mtu;
775 hdev->acl_pkts = __le16_to_cpu(rp->acl_max_pkt);
776 hdev->sco_pkts = __le16_to_cpu(rp->sco_max_pkt);
777
778 if (test_bit(HCI_QUIRK_FIXUP_BUFFER_SIZE, &hdev->quirks)) {
779 hdev->sco_mtu = 64;
780 hdev->sco_pkts = 8;
781 }
782
783 hdev->acl_cnt = hdev->acl_pkts;
784 hdev->sco_cnt = hdev->sco_pkts;
785
786 BT_DBG("%s acl mtu %d:%d sco mtu %d:%d", hdev->name,
787 hdev->acl_mtu, hdev->acl_pkts,
788 hdev->sco_mtu, hdev->sco_pkts);
789}
790
791static void hci_cc_read_bd_addr(struct hci_dev *hdev, struct sk_buff *skb)
792{
793 struct hci_rp_read_bd_addr *rp = (void *) skb->data;
794
795 BT_DBG("%s status 0x%x", hdev->name, rp->status);
796
797 if (!rp->status)
798 bacpy(&hdev->bdaddr, &rp->bdaddr);
799
Johan Hedberg23bb5762010-12-21 23:01:27 +0200800 hci_req_complete(hdev, HCI_OP_READ_BD_ADDR, rp->status);
801}
802
Andrei Emeltchenko350ee4c2011-12-07 15:56:51 +0200803static void hci_cc_read_data_block_size(struct hci_dev *hdev,
804 struct sk_buff *skb)
805{
806 struct hci_rp_read_data_block_size *rp = (void *) skb->data;
807
808 BT_DBG("%s status 0x%x", hdev->name, rp->status);
809
810 if (rp->status)
811 return;
812
813 hdev->block_mtu = __le16_to_cpu(rp->max_acl_len);
814 hdev->block_len = __le16_to_cpu(rp->block_len);
815 hdev->num_blocks = __le16_to_cpu(rp->num_blocks);
816
817 hdev->block_cnt = hdev->num_blocks;
818
819 BT_DBG("%s blk mtu %d cnt %d len %d", hdev->name, hdev->block_mtu,
820 hdev->block_cnt, hdev->block_len);
821
822 hci_req_complete(hdev, HCI_OP_READ_DATA_BLOCK_SIZE, rp->status);
823}
824
Johan Hedberg23bb5762010-12-21 23:01:27 +0200825static void hci_cc_write_ca_timeout(struct hci_dev *hdev, struct sk_buff *skb)
826{
827 __u8 status = *((__u8 *) skb->data);
828
829 BT_DBG("%s status 0x%x", hdev->name, status);
830
831 hci_req_complete(hdev, HCI_OP_WRITE_CA_TIMEOUT, status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +0200832}
833
Andrei Emeltchenko928abaa2011-10-12 10:53:57 +0300834static void hci_cc_read_local_amp_info(struct hci_dev *hdev,
835 struct sk_buff *skb)
836{
837 struct hci_rp_read_local_amp_info *rp = (void *) skb->data;
838
839 BT_DBG("%s status 0x%x", hdev->name, rp->status);
840
841 if (rp->status)
842 return;
843
844 hdev->amp_status = rp->amp_status;
845 hdev->amp_total_bw = __le32_to_cpu(rp->total_bw);
846 hdev->amp_max_bw = __le32_to_cpu(rp->max_bw);
847 hdev->amp_min_latency = __le32_to_cpu(rp->min_latency);
848 hdev->amp_max_pdu = __le32_to_cpu(rp->max_pdu);
849 hdev->amp_type = rp->amp_type;
850 hdev->amp_pal_cap = __le16_to_cpu(rp->pal_cap);
851 hdev->amp_assoc_size = __le16_to_cpu(rp->max_assoc_size);
852 hdev->amp_be_flush_to = __le32_to_cpu(rp->be_flush_to);
853 hdev->amp_max_flush_to = __le32_to_cpu(rp->max_flush_to);
854
855 hci_req_complete(hdev, HCI_OP_READ_LOCAL_AMP_INFO, rp->status);
856}
857
Johan Hedbergb0916ea2011-01-10 13:44:55 +0200858static void hci_cc_delete_stored_link_key(struct hci_dev *hdev,
859 struct sk_buff *skb)
860{
861 __u8 status = *((__u8 *) skb->data);
862
863 BT_DBG("%s status 0x%x", hdev->name, status);
864
865 hci_req_complete(hdev, HCI_OP_DELETE_STORED_LINK_KEY, status);
866}
867
Johan Hedbergd5859e22011-01-25 01:19:58 +0200868static void hci_cc_set_event_mask(struct hci_dev *hdev, struct sk_buff *skb)
869{
870 __u8 status = *((__u8 *) skb->data);
871
872 BT_DBG("%s status 0x%x", hdev->name, status);
873
874 hci_req_complete(hdev, HCI_OP_SET_EVENT_MASK, status);
875}
876
877static void hci_cc_write_inquiry_mode(struct hci_dev *hdev,
878 struct sk_buff *skb)
879{
880 __u8 status = *((__u8 *) skb->data);
881
882 BT_DBG("%s status 0x%x", hdev->name, status);
883
884 hci_req_complete(hdev, HCI_OP_WRITE_INQUIRY_MODE, status);
885}
886
887static void hci_cc_read_inq_rsp_tx_power(struct hci_dev *hdev,
888 struct sk_buff *skb)
889{
890 __u8 status = *((__u8 *) skb->data);
891
892 BT_DBG("%s status 0x%x", hdev->name, status);
893
894 hci_req_complete(hdev, HCI_OP_READ_INQ_RSP_TX_POWER, status);
895}
896
897static void hci_cc_set_event_flt(struct hci_dev *hdev, struct sk_buff *skb)
898{
899 __u8 status = *((__u8 *) skb->data);
900
901 BT_DBG("%s status 0x%x", hdev->name, status);
902
903 hci_req_complete(hdev, HCI_OP_SET_EVENT_FLT, status);
904}
905
Johan Hedberg980e1a52011-01-22 06:10:07 +0200906static void hci_cc_pin_code_reply(struct hci_dev *hdev, struct sk_buff *skb)
907{
908 struct hci_rp_pin_code_reply *rp = (void *) skb->data;
909 struct hci_cp_pin_code_reply *cp;
910 struct hci_conn *conn;
911
912 BT_DBG("%s status 0x%x", hdev->name, rp->status);
913
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200914 hci_dev_lock(hdev);
915
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200916 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg744cf192011-11-08 20:40:14 +0200917 mgmt_pin_code_reply_complete(hdev, &rp->bdaddr, rp->status);
Johan Hedberg980e1a52011-01-22 06:10:07 +0200918
919 if (rp->status != 0)
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200920 goto unlock;
Johan Hedberg980e1a52011-01-22 06:10:07 +0200921
922 cp = hci_sent_cmd_data(hdev, HCI_OP_PIN_CODE_REPLY);
923 if (!cp)
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200924 goto unlock;
Johan Hedberg980e1a52011-01-22 06:10:07 +0200925
926 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &cp->bdaddr);
927 if (conn)
928 conn->pin_length = cp->pin_len;
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200929
930unlock:
931 hci_dev_unlock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +0200932}
933
934static void hci_cc_pin_code_neg_reply(struct hci_dev *hdev, struct sk_buff *skb)
935{
936 struct hci_rp_pin_code_neg_reply *rp = (void *) skb->data;
937
938 BT_DBG("%s status 0x%x", hdev->name, rp->status);
939
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200940 hci_dev_lock(hdev);
941
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200942 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg744cf192011-11-08 20:40:14 +0200943 mgmt_pin_code_neg_reply_complete(hdev, &rp->bdaddr,
Johan Hedberg980e1a52011-01-22 06:10:07 +0200944 rp->status);
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200945
946 hci_dev_unlock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +0200947}
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200948
Ville Tervo6ed58ec2011-02-10 22:38:48 -0300949static void hci_cc_le_read_buffer_size(struct hci_dev *hdev,
950 struct sk_buff *skb)
951{
952 struct hci_rp_le_read_buffer_size *rp = (void *) skb->data;
953
954 BT_DBG("%s status 0x%x", hdev->name, rp->status);
955
956 if (rp->status)
957 return;
958
959 hdev->le_mtu = __le16_to_cpu(rp->le_mtu);
960 hdev->le_pkts = rp->le_max_pkt;
961
962 hdev->le_cnt = hdev->le_pkts;
963
964 BT_DBG("%s le mtu %d:%d", hdev->name, hdev->le_mtu, hdev->le_pkts);
965
966 hci_req_complete(hdev, HCI_OP_LE_READ_BUFFER_SIZE, rp->status);
967}
Johan Hedberg980e1a52011-01-22 06:10:07 +0200968
Johan Hedberga5c29682011-02-19 12:05:57 -0300969static void hci_cc_user_confirm_reply(struct hci_dev *hdev, struct sk_buff *skb)
970{
971 struct hci_rp_user_confirm_reply *rp = (void *) skb->data;
972
973 BT_DBG("%s status 0x%x", hdev->name, rp->status);
974
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200975 hci_dev_lock(hdev);
976
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200977 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300978 mgmt_user_confirm_reply_complete(hdev, &rp->bdaddr, ACL_LINK, 0,
979 rp->status);
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200980
981 hci_dev_unlock(hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -0300982}
983
984static void hci_cc_user_confirm_neg_reply(struct hci_dev *hdev,
985 struct sk_buff *skb)
986{
987 struct hci_rp_user_confirm_reply *rp = (void *) skb->data;
988
989 BT_DBG("%s status 0x%x", hdev->name, rp->status);
990
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200991 hci_dev_lock(hdev);
992
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200993 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg744cf192011-11-08 20:40:14 +0200994 mgmt_user_confirm_neg_reply_complete(hdev, &rp->bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300995 ACL_LINK, 0, rp->status);
Johan Hedberg56e5cb82011-11-08 20:40:16 +0200996
997 hci_dev_unlock(hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -0300998}
999
Brian Gix1143d452011-11-23 08:28:34 -08001000static void hci_cc_user_passkey_reply(struct hci_dev *hdev, struct sk_buff *skb)
1001{
1002 struct hci_rp_user_confirm_reply *rp = (void *) skb->data;
1003
1004 BT_DBG("%s status 0x%x", hdev->name, rp->status);
1005
1006 hci_dev_lock(hdev);
1007
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02001008 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg272d90d2012-02-09 15:26:12 +02001009 mgmt_user_passkey_reply_complete(hdev, &rp->bdaddr, ACL_LINK,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001010 0, rp->status);
Brian Gix1143d452011-11-23 08:28:34 -08001011
1012 hci_dev_unlock(hdev);
1013}
1014
1015static void hci_cc_user_passkey_neg_reply(struct hci_dev *hdev,
1016 struct sk_buff *skb)
1017{
1018 struct hci_rp_user_confirm_reply *rp = (void *) skb->data;
1019
1020 BT_DBG("%s status 0x%x", hdev->name, rp->status);
1021
1022 hci_dev_lock(hdev);
1023
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02001024 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Brian Gix1143d452011-11-23 08:28:34 -08001025 mgmt_user_passkey_neg_reply_complete(hdev, &rp->bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001026 ACL_LINK, 0, rp->status);
Brian Gix1143d452011-11-23 08:28:34 -08001027
1028 hci_dev_unlock(hdev);
1029}
1030
Szymon Jancc35938b2011-03-22 13:12:21 +01001031static void hci_cc_read_local_oob_data_reply(struct hci_dev *hdev,
1032 struct sk_buff *skb)
1033{
1034 struct hci_rp_read_local_oob_data *rp = (void *) skb->data;
1035
1036 BT_DBG("%s status 0x%x", hdev->name, rp->status);
1037
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001038 hci_dev_lock(hdev);
Johan Hedberg744cf192011-11-08 20:40:14 +02001039 mgmt_read_local_oob_data_reply_complete(hdev, rp->hash,
Szymon Jancc35938b2011-03-22 13:12:21 +01001040 rp->randomizer, rp->status);
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001041 hci_dev_unlock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01001042}
1043
Andre Guedes07f7fa52011-12-02 21:13:31 +09001044static void hci_cc_le_set_scan_param(struct hci_dev *hdev, struct sk_buff *skb)
1045{
1046 __u8 status = *((__u8 *) skb->data);
1047
1048 BT_DBG("%s status 0x%x", hdev->name, status);
Andre Guedes7ba8b4b2012-02-03 17:47:59 -03001049
1050 hci_req_complete(hdev, HCI_OP_LE_SET_SCAN_PARAM, status);
Andre Guedes3fd24152012-02-03 17:48:01 -03001051
1052 if (status) {
1053 hci_dev_lock(hdev);
1054 mgmt_start_discovery_failed(hdev, status);
1055 hci_dev_unlock(hdev);
1056 return;
1057 }
Andre Guedes07f7fa52011-12-02 21:13:31 +09001058}
1059
Andre Guedeseb9d91f2011-05-26 16:23:52 -03001060static void hci_cc_le_set_scan_enable(struct hci_dev *hdev,
1061 struct sk_buff *skb)
1062{
1063 struct hci_cp_le_set_scan_enable *cp;
1064 __u8 status = *((__u8 *) skb->data);
1065
1066 BT_DBG("%s status 0x%x", hdev->name, status);
1067
Andre Guedeseb9d91f2011-05-26 16:23:52 -03001068 cp = hci_sent_cmd_data(hdev, HCI_OP_LE_SET_SCAN_ENABLE);
1069 if (!cp)
1070 return;
1071
Andrei Emeltchenko68a8aea2011-12-19 16:14:18 +02001072 switch (cp->enable) {
1073 case LE_SCANNING_ENABLED:
Andre Guedes7ba8b4b2012-02-03 17:47:59 -03001074 hci_req_complete(hdev, HCI_OP_LE_SET_SCAN_ENABLE, status);
1075
Andre Guedes3fd24152012-02-03 17:48:01 -03001076 if (status) {
1077 hci_dev_lock(hdev);
1078 mgmt_start_discovery_failed(hdev, status);
1079 hci_dev_unlock(hdev);
Andre Guedes7ba8b4b2012-02-03 17:47:59 -03001080 return;
Andre Guedes3fd24152012-02-03 17:48:01 -03001081 }
Andre Guedes7ba8b4b2012-02-03 17:47:59 -03001082
Andre Guedesd23264a2011-11-25 20:53:38 -03001083 set_bit(HCI_LE_SCAN, &hdev->dev_flags);
1084
Gustavo F. Padovandb323f22011-06-20 16:39:29 -03001085 cancel_delayed_work_sync(&hdev->adv_work);
Andre Guedesa8f13c82011-09-09 18:56:24 -03001086
1087 hci_dev_lock(hdev);
Andre Guedeseb9d91f2011-05-26 16:23:52 -03001088 hci_adv_entries_clear(hdev);
Andre Guedes343f9352012-02-17 20:39:37 -03001089 hci_discovery_set_state(hdev, DISCOVERY_FINDING);
Andre Guedesa8f13c82011-09-09 18:56:24 -03001090 hci_dev_unlock(hdev);
Andrei Emeltchenko68a8aea2011-12-19 16:14:18 +02001091 break;
1092
1093 case LE_SCANNING_DISABLED:
Andre Guedes7ba8b4b2012-02-03 17:47:59 -03001094 if (status)
1095 return;
1096
Andre Guedesd23264a2011-11-25 20:53:38 -03001097 clear_bit(HCI_LE_SCAN, &hdev->dev_flags);
1098
Andre Guedesd0843292012-01-02 19:18:11 -03001099 schedule_delayed_work(&hdev->adv_work, ADV_CLEAR_TIMEOUT);
Andre Guedes5e0452c2012-02-17 20:39:38 -03001100
1101 if (hdev->discovery.type == DISCOV_TYPE_INTERLEAVED) {
1102 mgmt_interleaved_discovery(hdev);
1103 } else {
1104 hci_dev_lock(hdev);
1105 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
1106 hci_dev_unlock(hdev);
1107 }
1108
Andrei Emeltchenko68a8aea2011-12-19 16:14:18 +02001109 break;
1110
1111 default:
1112 BT_ERR("Used reserved LE_Scan_Enable param %d", cp->enable);
1113 break;
Andre Guedes35815082011-05-26 16:23:53 -03001114 }
Andre Guedeseb9d91f2011-05-26 16:23:52 -03001115}
1116
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03001117static void hci_cc_le_ltk_reply(struct hci_dev *hdev, struct sk_buff *skb)
1118{
1119 struct hci_rp_le_ltk_reply *rp = (void *) skb->data;
1120
1121 BT_DBG("%s status 0x%x", hdev->name, rp->status);
1122
1123 if (rp->status)
1124 return;
1125
1126 hci_req_complete(hdev, HCI_OP_LE_LTK_REPLY, rp->status);
1127}
1128
1129static void hci_cc_le_ltk_neg_reply(struct hci_dev *hdev, struct sk_buff *skb)
1130{
1131 struct hci_rp_le_ltk_neg_reply *rp = (void *) skb->data;
1132
1133 BT_DBG("%s status 0x%x", hdev->name, rp->status);
1134
1135 if (rp->status)
1136 return;
1137
1138 hci_req_complete(hdev, HCI_OP_LE_LTK_NEG_REPLY, rp->status);
1139}
1140
Andre Guedesf9b49302011-06-30 19:20:53 -03001141static inline void hci_cc_write_le_host_supported(struct hci_dev *hdev,
1142 struct sk_buff *skb)
1143{
Johan Hedberg06199cf2012-02-22 16:37:11 +02001144 struct hci_cp_write_le_host_supported *sent;
Andre Guedesf9b49302011-06-30 19:20:53 -03001145 __u8 status = *((__u8 *) skb->data);
1146
1147 BT_DBG("%s status 0x%x", hdev->name, status);
1148
Johan Hedberg06199cf2012-02-22 16:37:11 +02001149 sent = hci_sent_cmd_data(hdev, HCI_OP_WRITE_LE_HOST_SUPPORTED);
Johan Hedberg8f984df2012-02-28 01:07:22 +02001150 if (!sent)
Andre Guedesf9b49302011-06-30 19:20:53 -03001151 return;
1152
Johan Hedberg8f984df2012-02-28 01:07:22 +02001153 if (!status) {
1154 if (sent->le)
1155 hdev->host_features[0] |= LMP_HOST_LE;
1156 else
1157 hdev->host_features[0] &= ~LMP_HOST_LE;
1158 }
1159
1160 if (test_bit(HCI_MGMT, &hdev->dev_flags) &&
1161 !test_bit(HCI_INIT, &hdev->flags))
1162 mgmt_le_enable_complete(hdev, sent->le, status);
1163
1164 hci_req_complete(hdev, HCI_OP_WRITE_LE_HOST_SUPPORTED, status);
Andre Guedesf9b49302011-06-30 19:20:53 -03001165}
1166
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001167static inline void hci_cs_inquiry(struct hci_dev *hdev, __u8 status)
1168{
1169 BT_DBG("%s status 0x%x", hdev->name, status);
1170
1171 if (status) {
Johan Hedberg23bb5762010-12-21 23:01:27 +02001172 hci_req_complete(hdev, HCI_OP_INQUIRY, status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001173 hci_conn_check_pending(hdev);
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001174 hci_dev_lock(hdev);
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02001175 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Andre Guedes7a135102011-11-09 17:14:25 -03001176 mgmt_start_discovery_failed(hdev, status);
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001177 hci_dev_unlock(hdev);
Johan Hedberg314b2382011-04-27 10:29:57 -04001178 return;
1179 }
1180
Andre Guedes89352e72011-11-04 14:16:53 -03001181 set_bit(HCI_INQUIRY, &hdev->flags);
1182
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001183 hci_dev_lock(hdev);
Andre Guedes343f9352012-02-17 20:39:37 -03001184 hci_discovery_set_state(hdev, DISCOVERY_FINDING);
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001185 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001186}
1187
Linus Torvalds1da177e2005-04-16 15:20:36 -07001188static inline void hci_cs_create_conn(struct hci_dev *hdev, __u8 status)
1189{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001190 struct hci_cp_create_conn *cp;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001191 struct hci_conn *conn;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001192
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001193 BT_DBG("%s status 0x%x", hdev->name, status);
1194
1195 cp = hci_sent_cmd_data(hdev, HCI_OP_CREATE_CONN);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001196 if (!cp)
1197 return;
1198
1199 hci_dev_lock(hdev);
1200
1201 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &cp->bdaddr);
1202
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001203 BT_DBG("%s bdaddr %s conn %p", hdev->name, batostr(&cp->bdaddr), conn);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001204
1205 if (status) {
1206 if (conn && conn->state == BT_CONNECT) {
Marcel Holtmann4c67bc72006-10-15 17:30:56 +02001207 if (status != 0x0c || conn->attempt > 2) {
1208 conn->state = BT_CLOSED;
1209 hci_proto_connect_cfm(conn, status);
1210 hci_conn_del(conn);
1211 } else
1212 conn->state = BT_CONNECT2;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001213 }
1214 } else {
1215 if (!conn) {
1216 conn = hci_conn_add(hdev, ACL_LINK, &cp->bdaddr);
1217 if (conn) {
Johan Hedberga0c808b2012-01-16 09:49:58 +02001218 conn->out = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001219 conn->link_mode |= HCI_LM_MASTER;
1220 } else
Gustavo F. Padovan893ef972010-07-18 15:13:37 -03001221 BT_ERR("No memory for new connection");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001222 }
1223 }
1224
1225 hci_dev_unlock(hdev);
1226}
1227
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001228static void hci_cs_add_sco(struct hci_dev *hdev, __u8 status)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001229{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001230 struct hci_cp_add_sco *cp;
1231 struct hci_conn *acl, *sco;
1232 __u16 handle;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001233
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001234 BT_DBG("%s status 0x%x", hdev->name, status);
1235
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001236 if (!status)
1237 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001238
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001239 cp = hci_sent_cmd_data(hdev, HCI_OP_ADD_SCO);
1240 if (!cp)
1241 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001242
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001243 handle = __le16_to_cpu(cp->handle);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001244
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001245 BT_DBG("%s handle %d", hdev->name, handle);
Marcel Holtmann6bd57412006-11-18 22:14:22 +01001246
1247 hci_dev_lock(hdev);
1248
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001249 acl = hci_conn_hash_lookup_handle(hdev, handle);
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +02001250 if (acl) {
1251 sco = acl->link;
1252 if (sco) {
1253 sco->state = BT_CLOSED;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001254
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +02001255 hci_proto_connect_cfm(sco, status);
1256 hci_conn_del(sco);
1257 }
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001258 }
Marcel Holtmann6bd57412006-11-18 22:14:22 +01001259
1260 hci_dev_unlock(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001261}
1262
Marcel Holtmannf8558552008-07-14 20:13:49 +02001263static void hci_cs_auth_requested(struct hci_dev *hdev, __u8 status)
1264{
1265 struct hci_cp_auth_requested *cp;
1266 struct hci_conn *conn;
1267
1268 BT_DBG("%s status 0x%x", hdev->name, status);
1269
1270 if (!status)
1271 return;
1272
1273 cp = hci_sent_cmd_data(hdev, HCI_OP_AUTH_REQUESTED);
1274 if (!cp)
1275 return;
1276
1277 hci_dev_lock(hdev);
1278
1279 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(cp->handle));
1280 if (conn) {
1281 if (conn->state == BT_CONFIG) {
1282 hci_proto_connect_cfm(conn, status);
1283 hci_conn_put(conn);
1284 }
1285 }
1286
1287 hci_dev_unlock(hdev);
1288}
1289
1290static void hci_cs_set_conn_encrypt(struct hci_dev *hdev, __u8 status)
1291{
1292 struct hci_cp_set_conn_encrypt *cp;
1293 struct hci_conn *conn;
1294
1295 BT_DBG("%s status 0x%x", hdev->name, status);
1296
1297 if (!status)
1298 return;
1299
1300 cp = hci_sent_cmd_data(hdev, HCI_OP_SET_CONN_ENCRYPT);
1301 if (!cp)
1302 return;
1303
1304 hci_dev_lock(hdev);
1305
1306 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(cp->handle));
1307 if (conn) {
1308 if (conn->state == BT_CONFIG) {
1309 hci_proto_connect_cfm(conn, status);
1310 hci_conn_put(conn);
1311 }
1312 }
1313
1314 hci_dev_unlock(hdev);
1315}
1316
Johan Hedberg127178d2010-11-18 22:22:29 +02001317static int hci_outgoing_auth_needed(struct hci_dev *hdev,
Szymon Janc138d22e2011-02-17 16:44:23 +01001318 struct hci_conn *conn)
Johan Hedberg392599b2010-11-18 22:22:28 +02001319{
Johan Hedberg392599b2010-11-18 22:22:28 +02001320 if (conn->state != BT_CONFIG || !conn->out)
1321 return 0;
1322
Johan Hedberg765c2a92011-01-19 12:06:52 +05301323 if (conn->pending_sec_level == BT_SECURITY_SDP)
Johan Hedberg392599b2010-11-18 22:22:28 +02001324 return 0;
1325
1326 /* Only request authentication for SSP connections or non-SSP
Vinicius Costa Gomese9bf2bf2011-09-02 14:51:20 -03001327 * devices with sec_level HIGH or if MITM protection is requested */
Johan Hedbergaa64a8b2012-01-18 21:33:12 +02001328 if (!hci_conn_ssp_enabled(conn) &&
Vinicius Costa Gomese9bf2bf2011-09-02 14:51:20 -03001329 conn->pending_sec_level != BT_SECURITY_HIGH &&
1330 !(conn->auth_type & 0x01))
Johan Hedberg392599b2010-11-18 22:22:28 +02001331 return 0;
1332
Johan Hedberg392599b2010-11-18 22:22:28 +02001333 return 1;
1334}
1335
Gustavo F. Padovan00abfe42012-03-01 00:37:10 -03001336static inline int hci_resolve_name(struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001337 struct inquiry_entry *e)
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001338{
1339 struct hci_cp_remote_name_req cp;
1340
1341 memset(&cp, 0, sizeof(cp));
1342
1343 bacpy(&cp.bdaddr, &e->data.bdaddr);
1344 cp.pscan_rep_mode = e->data.pscan_rep_mode;
1345 cp.pscan_mode = e->data.pscan_mode;
1346 cp.clock_offset = e->data.clock_offset;
1347
1348 return hci_send_cmd(hdev, HCI_OP_REMOTE_NAME_REQ, sizeof(cp), &cp);
1349}
1350
Johan Hedbergb644ba32012-01-17 21:48:47 +02001351static bool hci_resolve_next_name(struct hci_dev *hdev)
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001352{
1353 struct discovery_state *discov = &hdev->discovery;
1354 struct inquiry_entry *e;
1355
Johan Hedbergb644ba32012-01-17 21:48:47 +02001356 if (list_empty(&discov->resolve))
1357 return false;
1358
1359 e = hci_inquiry_cache_lookup_resolve(hdev, BDADDR_ANY, NAME_NEEDED);
Ram Malovanya431bd52012-07-19 10:26:09 +03001360 if (!e)
1361 return false;
1362
Johan Hedbergb644ba32012-01-17 21:48:47 +02001363 if (hci_resolve_name(hdev, e) == 0) {
1364 e->name_state = NAME_PENDING;
1365 return true;
1366 }
1367
1368 return false;
1369}
1370
1371static void hci_check_pending_name(struct hci_dev *hdev, struct hci_conn *conn,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001372 bdaddr_t *bdaddr, u8 *name, u8 name_len)
Johan Hedbergb644ba32012-01-17 21:48:47 +02001373{
1374 struct discovery_state *discov = &hdev->discovery;
1375 struct inquiry_entry *e;
1376
1377 if (conn && !test_and_set_bit(HCI_CONN_MGMT_CONNECTED, &conn->flags))
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001378 mgmt_device_connected(hdev, bdaddr, ACL_LINK, 0x00, 0, name,
1379 name_len, conn->dev_class);
Johan Hedbergb644ba32012-01-17 21:48:47 +02001380
1381 if (discov->state == DISCOVERY_STOPPED)
1382 return;
1383
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001384 if (discov->state == DISCOVERY_STOPPING)
1385 goto discov_complete;
1386
1387 if (discov->state != DISCOVERY_RESOLVING)
1388 return;
1389
1390 e = hci_inquiry_cache_lookup_resolve(hdev, bdaddr, NAME_PENDING);
Ram Malovanya8b8ad62012-07-19 10:26:10 +03001391 /* If the device was not found in a list of found devices names of which
1392 * are pending. there is no need to continue resolving a next name as it
1393 * will be done upon receiving another Remote Name Request Complete
1394 * Event */
1395 if (!e)
1396 return;
1397
1398 list_del(&e->list);
1399 if (name) {
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001400 e->name_state = NAME_KNOWN;
Ram Malovanya8b8ad62012-07-19 10:26:10 +03001401 mgmt_remote_name(hdev, bdaddr, ACL_LINK, 0x00,
1402 e->data.rssi, name, name_len);
Ram Malovany4a20bce2012-07-19 10:26:11 +03001403 } else {
1404 e->name_state = NAME_NOT_KNOWN;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001405 }
1406
Johan Hedbergb644ba32012-01-17 21:48:47 +02001407 if (hci_resolve_next_name(hdev))
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001408 return;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001409
1410discov_complete:
1411 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
1412}
1413
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001414static void hci_cs_remote_name_req(struct hci_dev *hdev, __u8 status)
1415{
Johan Hedberg127178d2010-11-18 22:22:29 +02001416 struct hci_cp_remote_name_req *cp;
1417 struct hci_conn *conn;
1418
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001419 BT_DBG("%s status 0x%x", hdev->name, status);
Johan Hedberg127178d2010-11-18 22:22:29 +02001420
1421 /* If successful wait for the name req complete event before
1422 * checking for the need to do authentication */
1423 if (!status)
1424 return;
1425
1426 cp = hci_sent_cmd_data(hdev, HCI_OP_REMOTE_NAME_REQ);
1427 if (!cp)
1428 return;
1429
1430 hci_dev_lock(hdev);
1431
1432 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &cp->bdaddr);
Johan Hedbergb644ba32012-01-17 21:48:47 +02001433
1434 if (test_bit(HCI_MGMT, &hdev->dev_flags))
1435 hci_check_pending_name(hdev, conn, &cp->bdaddr, NULL, 0);
1436
Johan Hedberg79c6c702011-04-28 11:28:55 -07001437 if (!conn)
1438 goto unlock;
1439
1440 if (!hci_outgoing_auth_needed(hdev, conn))
1441 goto unlock;
1442
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001443 if (!test_and_set_bit(HCI_CONN_AUTH_PEND, &conn->flags)) {
Johan Hedberg127178d2010-11-18 22:22:29 +02001444 struct hci_cp_auth_requested cp;
1445 cp.handle = __cpu_to_le16(conn->handle);
1446 hci_send_cmd(hdev, HCI_OP_AUTH_REQUESTED, sizeof(cp), &cp);
1447 }
1448
Johan Hedberg79c6c702011-04-28 11:28:55 -07001449unlock:
Johan Hedberg127178d2010-11-18 22:22:29 +02001450 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001451}
1452
Marcel Holtmann769be972008-07-14 20:13:49 +02001453static void hci_cs_read_remote_features(struct hci_dev *hdev, __u8 status)
1454{
1455 struct hci_cp_read_remote_features *cp;
1456 struct hci_conn *conn;
1457
1458 BT_DBG("%s status 0x%x", hdev->name, status);
1459
1460 if (!status)
1461 return;
1462
1463 cp = hci_sent_cmd_data(hdev, HCI_OP_READ_REMOTE_FEATURES);
1464 if (!cp)
1465 return;
1466
1467 hci_dev_lock(hdev);
1468
1469 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(cp->handle));
1470 if (conn) {
1471 if (conn->state == BT_CONFIG) {
Marcel Holtmann769be972008-07-14 20:13:49 +02001472 hci_proto_connect_cfm(conn, status);
1473 hci_conn_put(conn);
1474 }
1475 }
1476
1477 hci_dev_unlock(hdev);
1478}
1479
1480static void hci_cs_read_remote_ext_features(struct hci_dev *hdev, __u8 status)
1481{
1482 struct hci_cp_read_remote_ext_features *cp;
1483 struct hci_conn *conn;
1484
1485 BT_DBG("%s status 0x%x", hdev->name, status);
1486
1487 if (!status)
1488 return;
1489
1490 cp = hci_sent_cmd_data(hdev, HCI_OP_READ_REMOTE_EXT_FEATURES);
1491 if (!cp)
1492 return;
1493
1494 hci_dev_lock(hdev);
1495
1496 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(cp->handle));
1497 if (conn) {
1498 if (conn->state == BT_CONFIG) {
Marcel Holtmann769be972008-07-14 20:13:49 +02001499 hci_proto_connect_cfm(conn, status);
1500 hci_conn_put(conn);
1501 }
1502 }
1503
1504 hci_dev_unlock(hdev);
1505}
1506
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001507static void hci_cs_setup_sync_conn(struct hci_dev *hdev, __u8 status)
1508{
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001509 struct hci_cp_setup_sync_conn *cp;
1510 struct hci_conn *acl, *sco;
1511 __u16 handle;
1512
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001513 BT_DBG("%s status 0x%x", hdev->name, status);
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001514
1515 if (!status)
1516 return;
1517
1518 cp = hci_sent_cmd_data(hdev, HCI_OP_SETUP_SYNC_CONN);
1519 if (!cp)
1520 return;
1521
1522 handle = __le16_to_cpu(cp->handle);
1523
1524 BT_DBG("%s handle %d", hdev->name, handle);
1525
1526 hci_dev_lock(hdev);
1527
1528 acl = hci_conn_hash_lookup_handle(hdev, handle);
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +02001529 if (acl) {
1530 sco = acl->link;
1531 if (sco) {
1532 sco->state = BT_CLOSED;
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001533
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +02001534 hci_proto_connect_cfm(sco, status);
1535 hci_conn_del(sco);
1536 }
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001537 }
1538
1539 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001540}
1541
1542static void hci_cs_sniff_mode(struct hci_dev *hdev, __u8 status)
1543{
1544 struct hci_cp_sniff_mode *cp;
1545 struct hci_conn *conn;
1546
1547 BT_DBG("%s status 0x%x", hdev->name, status);
1548
1549 if (!status)
1550 return;
1551
1552 cp = hci_sent_cmd_data(hdev, HCI_OP_SNIFF_MODE);
1553 if (!cp)
1554 return;
1555
1556 hci_dev_lock(hdev);
1557
1558 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(cp->handle));
Marcel Holtmanne73439d2010-07-26 10:06:00 -04001559 if (conn) {
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001560 clear_bit(HCI_CONN_MODE_CHANGE_PEND, &conn->flags);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001561
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001562 if (test_and_clear_bit(HCI_CONN_SCO_SETUP_PEND, &conn->flags))
Marcel Holtmanne73439d2010-07-26 10:06:00 -04001563 hci_sco_setup(conn, status);
1564 }
1565
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001566 hci_dev_unlock(hdev);
1567}
1568
1569static void hci_cs_exit_sniff_mode(struct hci_dev *hdev, __u8 status)
1570{
1571 struct hci_cp_exit_sniff_mode *cp;
1572 struct hci_conn *conn;
1573
1574 BT_DBG("%s status 0x%x", hdev->name, status);
1575
1576 if (!status)
1577 return;
1578
1579 cp = hci_sent_cmd_data(hdev, HCI_OP_EXIT_SNIFF_MODE);
1580 if (!cp)
1581 return;
1582
1583 hci_dev_lock(hdev);
1584
1585 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(cp->handle));
Marcel Holtmanne73439d2010-07-26 10:06:00 -04001586 if (conn) {
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001587 clear_bit(HCI_CONN_MODE_CHANGE_PEND, &conn->flags);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001588
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001589 if (test_and_clear_bit(HCI_CONN_SCO_SETUP_PEND, &conn->flags))
Marcel Holtmanne73439d2010-07-26 10:06:00 -04001590 hci_sco_setup(conn, status);
1591 }
1592
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001593 hci_dev_unlock(hdev);
1594}
1595
Johan Hedberg88c3df12012-02-09 14:27:38 +02001596static void hci_cs_disconnect(struct hci_dev *hdev, u8 status)
1597{
1598 struct hci_cp_disconnect *cp;
1599 struct hci_conn *conn;
1600
1601 if (!status)
1602 return;
1603
1604 cp = hci_sent_cmd_data(hdev, HCI_OP_DISCONNECT);
1605 if (!cp)
1606 return;
1607
1608 hci_dev_lock(hdev);
1609
1610 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(cp->handle));
1611 if (conn)
1612 mgmt_disconnect_failed(hdev, &conn->dst, conn->type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001613 conn->dst_type, status);
Johan Hedberg88c3df12012-02-09 14:27:38 +02001614
1615 hci_dev_unlock(hdev);
1616}
1617
Ville Tervofcd89c02011-02-10 22:38:47 -03001618static void hci_cs_le_create_conn(struct hci_dev *hdev, __u8 status)
1619{
1620 struct hci_cp_le_create_conn *cp;
1621 struct hci_conn *conn;
1622
1623 BT_DBG("%s status 0x%x", hdev->name, status);
1624
1625 cp = hci_sent_cmd_data(hdev, HCI_OP_LE_CREATE_CONN);
1626 if (!cp)
1627 return;
1628
1629 hci_dev_lock(hdev);
1630
1631 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &cp->peer_addr);
1632
1633 BT_DBG("%s bdaddr %s conn %p", hdev->name, batostr(&cp->peer_addr),
1634 conn);
1635
1636 if (status) {
1637 if (conn && conn->state == BT_CONNECT) {
1638 conn->state = BT_CLOSED;
1639 hci_proto_connect_cfm(conn, status);
1640 hci_conn_del(conn);
1641 }
1642 } else {
1643 if (!conn) {
1644 conn = hci_conn_add(hdev, LE_LINK, &cp->peer_addr);
Andre Guedes29b79882011-05-31 14:20:54 -03001645 if (conn) {
1646 conn->dst_type = cp->peer_addr_type;
Johan Hedberga0c808b2012-01-16 09:49:58 +02001647 conn->out = true;
Andre Guedes29b79882011-05-31 14:20:54 -03001648 } else {
Ville Tervofcd89c02011-02-10 22:38:47 -03001649 BT_ERR("No memory for new connection");
Andre Guedes29b79882011-05-31 14:20:54 -03001650 }
Ville Tervofcd89c02011-02-10 22:38:47 -03001651 }
1652 }
1653
1654 hci_dev_unlock(hdev);
1655}
1656
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03001657static void hci_cs_le_start_enc(struct hci_dev *hdev, u8 status)
1658{
1659 BT_DBG("%s status 0x%x", hdev->name, status);
1660}
1661
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001662static inline void hci_inquiry_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
1663{
1664 __u8 status = *((__u8 *) skb->data);
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001665 struct discovery_state *discov = &hdev->discovery;
1666 struct inquiry_entry *e;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001667
1668 BT_DBG("%s status %d", hdev->name, status);
1669
Johan Hedberg23bb5762010-12-21 23:01:27 +02001670 hci_req_complete(hdev, HCI_OP_INQUIRY, status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001671
1672 hci_conn_check_pending(hdev);
Andre Guedes89352e72011-11-04 14:16:53 -03001673
1674 if (!test_and_clear_bit(HCI_INQUIRY, &hdev->flags))
1675 return;
1676
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02001677 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001678 return;
1679
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001680 hci_dev_lock(hdev);
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001681
Andre Guedes343f9352012-02-17 20:39:37 -03001682 if (discov->state != DISCOVERY_FINDING)
Johan Hedberg30dc78e2012-01-04 15:44:20 +02001683 goto unlock;
1684
1685 if (list_empty(&discov->resolve)) {
1686 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
1687 goto unlock;
1688 }
1689
1690 e = hci_inquiry_cache_lookup_resolve(hdev, BDADDR_ANY, NAME_NEEDED);
1691 if (e && hci_resolve_name(hdev, e) == 0) {
1692 e->name_state = NAME_PENDING;
1693 hci_discovery_set_state(hdev, DISCOVERY_RESOLVING);
1694 } else {
1695 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
1696 }
1697
1698unlock:
Johan Hedberg56e5cb82011-11-08 20:40:16 +02001699 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001700}
1701
Linus Torvalds1da177e2005-04-16 15:20:36 -07001702static inline void hci_inquiry_result_evt(struct hci_dev *hdev, struct sk_buff *skb)
1703{
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001704 struct inquiry_data data;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001705 struct inquiry_info *info = (void *) (skb->data + 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001706 int num_rsp = *((__u8 *) skb->data);
1707
1708 BT_DBG("%s num_rsp %d", hdev->name, num_rsp);
1709
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001710 if (!num_rsp)
1711 return;
1712
Linus Torvalds1da177e2005-04-16 15:20:36 -07001713 hci_dev_lock(hdev);
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001714
Johan Hedberge17acd42011-03-30 23:57:16 +03001715 for (; num_rsp; num_rsp--, info++) {
Johan Hedberg388fc8f2012-02-23 00:38:59 +02001716 bool name_known, ssp;
Johan Hedberg31754052012-01-04 13:39:52 +02001717
Linus Torvalds1da177e2005-04-16 15:20:36 -07001718 bacpy(&data.bdaddr, &info->bdaddr);
1719 data.pscan_rep_mode = info->pscan_rep_mode;
1720 data.pscan_period_mode = info->pscan_period_mode;
1721 data.pscan_mode = info->pscan_mode;
1722 memcpy(data.dev_class, info->dev_class, 3);
1723 data.clock_offset = info->clock_offset;
1724 data.rssi = 0x00;
Marcel Holtmann41a96212008-07-14 20:13:48 +02001725 data.ssp_mode = 0x00;
Johan Hedberg31754052012-01-04 13:39:52 +02001726
Johan Hedberg388fc8f2012-02-23 00:38:59 +02001727 name_known = hci_inquiry_cache_update(hdev, &data, false, &ssp);
Johan Hedberg48264f02011-11-09 13:58:58 +02001728 mgmt_device_found(hdev, &info->bdaddr, ACL_LINK, 0x00,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001729 info->dev_class, 0, !name_known, ssp, NULL,
1730 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001731 }
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001732
Linus Torvalds1da177e2005-04-16 15:20:36 -07001733 hci_dev_unlock(hdev);
1734}
1735
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001736static inline void hci_conn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001737{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001738 struct hci_ev_conn_complete *ev = (void *) skb->data;
1739 struct hci_conn *conn;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001740
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001741 BT_DBG("%s", hdev->name);
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001742
Linus Torvalds1da177e2005-04-16 15:20:36 -07001743 hci_dev_lock(hdev);
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001744
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001745 conn = hci_conn_hash_lookup_ba(hdev, ev->link_type, &ev->bdaddr);
Marcel Holtmann94992372009-04-19 19:30:03 +02001746 if (!conn) {
1747 if (ev->link_type != SCO_LINK)
1748 goto unlock;
1749
1750 conn = hci_conn_hash_lookup_ba(hdev, ESCO_LINK, &ev->bdaddr);
1751 if (!conn)
1752 goto unlock;
1753
1754 conn->type = SCO_LINK;
1755 }
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001756
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001757 if (!ev->status) {
1758 conn->handle = __le16_to_cpu(ev->handle);
Marcel Holtmann769be972008-07-14 20:13:49 +02001759
1760 if (conn->type == ACL_LINK) {
1761 conn->state = BT_CONFIG;
1762 hci_conn_hold(conn);
Marcel Holtmann052b30b2009-04-26 20:01:22 +02001763 conn->disc_timeout = HCI_DISCONN_TIMEOUT;
Marcel Holtmann769be972008-07-14 20:13:49 +02001764 } else
1765 conn->state = BT_CONNECTED;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001766
Marcel Holtmann9eba32b2009-08-22 14:19:26 -07001767 hci_conn_hold_device(conn);
Marcel Holtmann7d0db0a2008-07-14 20:13:51 +02001768 hci_conn_add_sysfs(conn);
1769
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001770 if (test_bit(HCI_AUTH, &hdev->flags))
1771 conn->link_mode |= HCI_LM_AUTH;
1772
1773 if (test_bit(HCI_ENCRYPT, &hdev->flags))
1774 conn->link_mode |= HCI_LM_ENCRYPT;
1775
1776 /* Get remote features */
1777 if (conn->type == ACL_LINK) {
1778 struct hci_cp_read_remote_features cp;
1779 cp.handle = ev->handle;
Marcel Holtmann769be972008-07-14 20:13:49 +02001780 hci_send_cmd(hdev, HCI_OP_READ_REMOTE_FEATURES,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001781 sizeof(cp), &cp);
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001782 }
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001783
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001784 /* Set packet type for incoming connection */
Andrei Emeltchenkod095c1e2011-12-01 14:33:27 +02001785 if (!conn->out && hdev->hci_ver < BLUETOOTH_VER_2_0) {
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001786 struct hci_cp_change_conn_ptype cp;
1787 cp.handle = ev->handle;
Marcel Holtmanna8746412008-07-14 20:13:46 +02001788 cp.pkt_type = cpu_to_le16(conn->pkt_type);
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001789 hci_send_cmd(hdev, HCI_OP_CHANGE_CONN_PTYPE, sizeof(cp),
1790 &cp);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001791 }
Johan Hedberg17d5c042011-01-22 06:09:08 +02001792 } else {
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001793 conn->state = BT_CLOSED;
Johan Hedberg17d5c042011-01-22 06:09:08 +02001794 if (conn->type == ACL_LINK)
Johan Hedberg744cf192011-11-08 20:40:14 +02001795 mgmt_connect_failed(hdev, &ev->bdaddr, conn->type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001796 conn->dst_type, ev->status);
Johan Hedberg17d5c042011-01-22 06:09:08 +02001797 }
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001798
Marcel Holtmanne73439d2010-07-26 10:06:00 -04001799 if (conn->type == ACL_LINK)
1800 hci_sco_setup(conn, ev->status);
Marcel Holtmann45bb4bf2005-08-09 20:27:49 -07001801
Marcel Holtmann769be972008-07-14 20:13:49 +02001802 if (ev->status) {
1803 hci_proto_connect_cfm(conn, ev->status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001804 hci_conn_del(conn);
Marcel Holtmannc89b6e62009-01-15 21:57:03 +01001805 } else if (ev->link_type != ACL_LINK)
1806 hci_proto_connect_cfm(conn, ev->status);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001807
1808unlock:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001809 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001810
1811 hci_conn_check_pending(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001812}
1813
Linus Torvalds1da177e2005-04-16 15:20:36 -07001814static inline void hci_conn_request_evt(struct hci_dev *hdev, struct sk_buff *skb)
1815{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001816 struct hci_ev_conn_request *ev = (void *) skb->data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001817 int mask = hdev->link_mode;
1818
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001819 BT_DBG("%s bdaddr %s type 0x%x", hdev->name,
1820 batostr(&ev->bdaddr), ev->link_type);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001821
1822 mask |= hci_proto_connect_ind(hdev, &ev->bdaddr, ev->link_type);
1823
Szymon Janc138d22e2011-02-17 16:44:23 +01001824 if ((mask & HCI_LM_ACCEPT) &&
1825 !hci_blacklist_lookup(hdev, &ev->bdaddr)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001826 /* Connection accepted */
Marcel Holtmannc7bdd502008-07-14 20:13:47 +02001827 struct inquiry_entry *ie;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001828 struct hci_conn *conn;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001829
1830 hci_dev_lock(hdev);
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001831
Andrei Emeltchenkocc11b9c2010-11-22 13:21:37 +02001832 ie = hci_inquiry_cache_lookup(hdev, &ev->bdaddr);
1833 if (ie)
Marcel Holtmannc7bdd502008-07-14 20:13:47 +02001834 memcpy(ie->data.dev_class, ev->dev_class, 3);
1835
Linus Torvalds1da177e2005-04-16 15:20:36 -07001836 conn = hci_conn_hash_lookup_ba(hdev, ev->link_type, &ev->bdaddr);
1837 if (!conn) {
Andrei Emeltchenkocc11b9c2010-11-22 13:21:37 +02001838 conn = hci_conn_add(hdev, ev->link_type, &ev->bdaddr);
1839 if (!conn) {
Gustavo F. Padovan893ef972010-07-18 15:13:37 -03001840 BT_ERR("No memory for new connection");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001841 hci_dev_unlock(hdev);
1842 return;
1843 }
1844 }
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001845
Linus Torvalds1da177e2005-04-16 15:20:36 -07001846 memcpy(conn->dev_class, ev->dev_class, 3);
1847 conn->state = BT_CONNECT;
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001848
Linus Torvalds1da177e2005-04-16 15:20:36 -07001849 hci_dev_unlock(hdev);
1850
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001851 if (ev->link_type == ACL_LINK || !lmp_esco_capable(hdev)) {
1852 struct hci_cp_accept_conn_req cp;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001853
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001854 bacpy(&cp.bdaddr, &ev->bdaddr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001855
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001856 if (lmp_rswitch_capable(hdev) && (mask & HCI_LM_MASTER))
1857 cp.role = 0x00; /* Become master */
1858 else
1859 cp.role = 0x01; /* Remain slave */
1860
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001861 hci_send_cmd(hdev, HCI_OP_ACCEPT_CONN_REQ, sizeof(cp),
1862 &cp);
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001863 } else {
1864 struct hci_cp_accept_sync_conn_req cp;
1865
1866 bacpy(&cp.bdaddr, &ev->bdaddr);
Marcel Holtmanna8746412008-07-14 20:13:46 +02001867 cp.pkt_type = cpu_to_le16(conn->pkt_type);
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001868
1869 cp.tx_bandwidth = cpu_to_le32(0x00001f40);
1870 cp.rx_bandwidth = cpu_to_le32(0x00001f40);
1871 cp.max_latency = cpu_to_le16(0xffff);
1872 cp.content_format = cpu_to_le16(hdev->voice_setting);
1873 cp.retrans_effort = 0xff;
1874
1875 hci_send_cmd(hdev, HCI_OP_ACCEPT_SYNC_CONN_REQ,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001876 sizeof(cp), &cp);
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02001877 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001878 } else {
1879 /* Connection rejected */
1880 struct hci_cp_reject_conn_req cp;
1881
1882 bacpy(&cp.bdaddr, &ev->bdaddr);
Andrei Emeltchenko9f5a0d72011-11-07 14:20:25 +02001883 cp.reason = HCI_ERROR_REJ_BAD_ADDR;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001884 hci_send_cmd(hdev, HCI_OP_REJECT_CONN_REQ, sizeof(cp), &cp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001885 }
1886}
1887
Linus Torvalds1da177e2005-04-16 15:20:36 -07001888static inline void hci_disconn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
1889{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001890 struct hci_ev_disconn_complete *ev = (void *) skb->data;
Marcel Holtmann04837f62006-07-03 10:02:33 +02001891 struct hci_conn *conn;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001892
1893 BT_DBG("%s status %d", hdev->name, ev->status);
1894
Linus Torvalds1da177e2005-04-16 15:20:36 -07001895 hci_dev_lock(hdev);
1896
Marcel Holtmann04837f62006-07-03 10:02:33 +02001897 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
Johan Hedbergf7520542011-01-20 12:34:39 +02001898 if (!conn)
1899 goto unlock;
Marcel Holtmann7d0db0a2008-07-14 20:13:51 +02001900
Johan Hedberg37d9ef72011-11-10 15:54:39 +02001901 if (ev->status == 0)
1902 conn->state = BT_CLOSED;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001903
Johan Hedbergb644ba32012-01-17 21:48:47 +02001904 if (test_and_clear_bit(HCI_CONN_MGMT_CONNECTED, &conn->flags) &&
1905 (conn->type == ACL_LINK || conn->type == LE_LINK)) {
Johan Hedberg37d9ef72011-11-10 15:54:39 +02001906 if (ev->status != 0)
Johan Hedberg88c3df12012-02-09 14:27:38 +02001907 mgmt_disconnect_failed(hdev, &conn->dst, conn->type,
1908 conn->dst_type, ev->status);
Johan Hedberg37d9ef72011-11-10 15:54:39 +02001909 else
Johan Hedbergafc747a2012-01-15 18:11:07 +02001910 mgmt_device_disconnected(hdev, &conn->dst, conn->type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001911 conn->dst_type);
Johan Hedberg37d9ef72011-11-10 15:54:39 +02001912 }
Johan Hedbergf7520542011-01-20 12:34:39 +02001913
Johan Hedberg37d9ef72011-11-10 15:54:39 +02001914 if (ev->status == 0) {
Vishal Agarwal6ec5bca2012-04-16 14:44:44 +05301915 if (conn->type == ACL_LINK && conn->flush_key)
1916 hci_remove_link_key(hdev, &conn->dst);
Johan Hedberg37d9ef72011-11-10 15:54:39 +02001917 hci_proto_disconn_cfm(conn, ev->reason);
1918 hci_conn_del(conn);
1919 }
Johan Hedbergf7520542011-01-20 12:34:39 +02001920
1921unlock:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001922 hci_dev_unlock(hdev);
1923}
1924
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001925static inline void hci_auth_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
1926{
1927 struct hci_ev_auth_complete *ev = (void *) skb->data;
1928 struct hci_conn *conn;
1929
1930 BT_DBG("%s status %d", hdev->name, ev->status);
1931
1932 hci_dev_lock(hdev);
1933
1934 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001935 if (!conn)
1936 goto unlock;
1937
1938 if (!ev->status) {
Johan Hedbergaa64a8b2012-01-18 21:33:12 +02001939 if (!hci_conn_ssp_enabled(conn) &&
1940 test_bit(HCI_CONN_REAUTH_PEND, &conn->flags)) {
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001941 BT_INFO("re-auth of legacy device is not possible.");
Johan Hedberg2a611692011-02-19 12:06:00 -03001942 } else {
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001943 conn->link_mode |= HCI_LM_AUTH;
1944 conn->sec_level = conn->pending_sec_level;
Johan Hedberg2a611692011-02-19 12:06:00 -03001945 }
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001946 } else {
Johan Hedbergbab73cb2012-02-09 16:07:29 +02001947 mgmt_auth_failed(hdev, &conn->dst, conn->type, conn->dst_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001948 ev->status);
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001949 }
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001950
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001951 clear_bit(HCI_CONN_AUTH_PEND, &conn->flags);
1952 clear_bit(HCI_CONN_REAUTH_PEND, &conn->flags);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001953
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001954 if (conn->state == BT_CONFIG) {
Johan Hedbergaa64a8b2012-01-18 21:33:12 +02001955 if (!ev->status && hci_conn_ssp_enabled(conn)) {
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001956 struct hci_cp_set_conn_encrypt cp;
1957 cp.handle = ev->handle;
1958 cp.encrypt = 0x01;
1959 hci_send_cmd(hdev, HCI_OP_SET_CONN_ENCRYPT, sizeof(cp),
1960 &cp);
Marcel Holtmann052b30b2009-04-26 20:01:22 +02001961 } else {
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001962 conn->state = BT_CONNECTED;
1963 hci_proto_connect_cfm(conn, ev->status);
Marcel Holtmann052b30b2009-04-26 20:01:22 +02001964 hci_conn_put(conn);
1965 }
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001966 } else {
1967 hci_auth_cfm(conn, ev->status);
Marcel Holtmann052b30b2009-04-26 20:01:22 +02001968
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001969 hci_conn_hold(conn);
1970 conn->disc_timeout = HCI_DISCONN_TIMEOUT;
1971 hci_conn_put(conn);
1972 }
1973
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001974 if (test_bit(HCI_CONN_ENCRYPT_PEND, &conn->flags)) {
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001975 if (!ev->status) {
1976 struct hci_cp_set_conn_encrypt cp;
1977 cp.handle = ev->handle;
1978 cp.encrypt = 0x01;
1979 hci_send_cmd(hdev, HCI_OP_SET_CONN_ENCRYPT, sizeof(cp),
1980 &cp);
1981 } else {
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001982 clear_bit(HCI_CONN_ENCRYPT_PEND, &conn->flags);
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001983 hci_encrypt_cfm(conn, ev->status, 0x00);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001984 }
1985 }
1986
Waldemar Rymarkiewiczd7556e22011-05-31 15:49:26 +02001987unlock:
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001988 hci_dev_unlock(hdev);
1989}
1990
1991static inline void hci_remote_name_evt(struct hci_dev *hdev, struct sk_buff *skb)
1992{
Johan Hedberg127178d2010-11-18 22:22:29 +02001993 struct hci_ev_remote_name *ev = (void *) skb->data;
1994 struct hci_conn *conn;
1995
Marcel Holtmanna9de9242007-10-20 13:33:56 +02001996 BT_DBG("%s", hdev->name);
1997
1998 hci_conn_check_pending(hdev);
Johan Hedberg127178d2010-11-18 22:22:29 +02001999
2000 hci_dev_lock(hdev);
2001
2002 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
Johan Hedbergb644ba32012-01-17 21:48:47 +02002003
2004 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
2005 goto check_auth;
2006
2007 if (ev->status == 0)
2008 hci_check_pending_name(hdev, conn, &ev->bdaddr, ev->name,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002009 strnlen(ev->name, HCI_MAX_NAME_LENGTH));
Johan Hedbergb644ba32012-01-17 21:48:47 +02002010 else
2011 hci_check_pending_name(hdev, conn, &ev->bdaddr, NULL, 0);
2012
2013check_auth:
Johan Hedberg79c6c702011-04-28 11:28:55 -07002014 if (!conn)
2015 goto unlock;
2016
2017 if (!hci_outgoing_auth_needed(hdev, conn))
2018 goto unlock;
2019
Johan Hedberg51a8efd2012-01-16 06:10:31 +02002020 if (!test_and_set_bit(HCI_CONN_AUTH_PEND, &conn->flags)) {
Johan Hedberg127178d2010-11-18 22:22:29 +02002021 struct hci_cp_auth_requested cp;
2022 cp.handle = __cpu_to_le16(conn->handle);
2023 hci_send_cmd(hdev, HCI_OP_AUTH_REQUESTED, sizeof(cp), &cp);
2024 }
2025
Johan Hedberg79c6c702011-04-28 11:28:55 -07002026unlock:
Johan Hedberg127178d2010-11-18 22:22:29 +02002027 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002028}
2029
2030static inline void hci_encrypt_change_evt(struct hci_dev *hdev, struct sk_buff *skb)
2031{
2032 struct hci_ev_encrypt_change *ev = (void *) skb->data;
2033 struct hci_conn *conn;
2034
2035 BT_DBG("%s status %d", hdev->name, ev->status);
2036
2037 hci_dev_lock(hdev);
2038
2039 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
2040 if (conn) {
2041 if (!ev->status) {
Marcel Holtmannae293192008-07-14 20:13:45 +02002042 if (ev->encrypt) {
2043 /* Encryption implies authentication */
2044 conn->link_mode |= HCI_LM_AUTH;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002045 conn->link_mode |= HCI_LM_ENCRYPT;
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -03002046 conn->sec_level = conn->pending_sec_level;
Marcel Holtmannae293192008-07-14 20:13:45 +02002047 } else
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002048 conn->link_mode &= ~HCI_LM_ENCRYPT;
2049 }
2050
Johan Hedberg51a8efd2012-01-16 06:10:31 +02002051 clear_bit(HCI_CONN_ENCRYPT_PEND, &conn->flags);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002052
Gustavo Padovana7d77232012-05-13 03:20:07 -03002053 if (ev->status && conn->state == BT_CONNECTED) {
2054 hci_acl_disconn(conn, 0x13);
2055 hci_conn_put(conn);
2056 goto unlock;
2057 }
2058
Marcel Holtmannf8558552008-07-14 20:13:49 +02002059 if (conn->state == BT_CONFIG) {
2060 if (!ev->status)
2061 conn->state = BT_CONNECTED;
2062
2063 hci_proto_connect_cfm(conn, ev->status);
2064 hci_conn_put(conn);
2065 } else
2066 hci_encrypt_cfm(conn, ev->status, ev->encrypt);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002067 }
2068
Gustavo Padovana7d77232012-05-13 03:20:07 -03002069unlock:
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002070 hci_dev_unlock(hdev);
2071}
2072
2073static inline void hci_change_link_key_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
2074{
2075 struct hci_ev_change_link_key_complete *ev = (void *) skb->data;
2076 struct hci_conn *conn;
2077
2078 BT_DBG("%s status %d", hdev->name, ev->status);
2079
2080 hci_dev_lock(hdev);
2081
2082 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
2083 if (conn) {
2084 if (!ev->status)
2085 conn->link_mode |= HCI_LM_SECURE;
2086
Johan Hedberg51a8efd2012-01-16 06:10:31 +02002087 clear_bit(HCI_CONN_AUTH_PEND, &conn->flags);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002088
2089 hci_key_change_cfm(conn, ev->status);
2090 }
2091
2092 hci_dev_unlock(hdev);
2093}
2094
2095static inline void hci_remote_features_evt(struct hci_dev *hdev, struct sk_buff *skb)
2096{
2097 struct hci_ev_remote_features *ev = (void *) skb->data;
2098 struct hci_conn *conn;
2099
2100 BT_DBG("%s status %d", hdev->name, ev->status);
2101
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002102 hci_dev_lock(hdev);
2103
2104 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
Johan Hedbergccd556f2010-11-10 17:11:51 +02002105 if (!conn)
2106 goto unlock;
Marcel Holtmann769be972008-07-14 20:13:49 +02002107
Johan Hedbergccd556f2010-11-10 17:11:51 +02002108 if (!ev->status)
2109 memcpy(conn->features, ev->features, 8);
2110
2111 if (conn->state != BT_CONFIG)
2112 goto unlock;
2113
2114 if (!ev->status && lmp_ssp_capable(hdev) && lmp_ssp_capable(conn)) {
2115 struct hci_cp_read_remote_ext_features cp;
2116 cp.handle = ev->handle;
2117 cp.page = 0x01;
2118 hci_send_cmd(hdev, HCI_OP_READ_REMOTE_EXT_FEATURES,
Marcel Holtmann769be972008-07-14 20:13:49 +02002119 sizeof(cp), &cp);
Johan Hedberg392599b2010-11-18 22:22:28 +02002120 goto unlock;
2121 }
2122
Johan Hedberg671267b2012-05-12 16:11:50 -03002123 if (!ev->status && !test_bit(HCI_CONN_MGMT_CONNECTED, &conn->flags)) {
Johan Hedberg127178d2010-11-18 22:22:29 +02002124 struct hci_cp_remote_name_req cp;
2125 memset(&cp, 0, sizeof(cp));
2126 bacpy(&cp.bdaddr, &conn->dst);
2127 cp.pscan_rep_mode = 0x02;
2128 hci_send_cmd(hdev, HCI_OP_REMOTE_NAME_REQ, sizeof(cp), &cp);
Johan Hedbergb644ba32012-01-17 21:48:47 +02002129 } else if (!test_and_set_bit(HCI_CONN_MGMT_CONNECTED, &conn->flags))
2130 mgmt_device_connected(hdev, &conn->dst, conn->type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002131 conn->dst_type, 0, NULL, 0,
2132 conn->dev_class);
Johan Hedberg392599b2010-11-18 22:22:28 +02002133
Johan Hedberg127178d2010-11-18 22:22:29 +02002134 if (!hci_outgoing_auth_needed(hdev, conn)) {
Johan Hedbergccd556f2010-11-10 17:11:51 +02002135 conn->state = BT_CONNECTED;
2136 hci_proto_connect_cfm(conn, ev->status);
2137 hci_conn_put(conn);
Marcel Holtmann769be972008-07-14 20:13:49 +02002138 }
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002139
Johan Hedbergccd556f2010-11-10 17:11:51 +02002140unlock:
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002141 hci_dev_unlock(hdev);
2142}
2143
2144static inline void hci_remote_version_evt(struct hci_dev *hdev, struct sk_buff *skb)
2145{
2146 BT_DBG("%s", hdev->name);
2147}
2148
2149static inline void hci_qos_setup_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
2150{
2151 BT_DBG("%s", hdev->name);
2152}
2153
2154static inline void hci_cmd_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
2155{
2156 struct hci_ev_cmd_complete *ev = (void *) skb->data;
2157 __u16 opcode;
2158
2159 skb_pull(skb, sizeof(*ev));
2160
2161 opcode = __le16_to_cpu(ev->opcode);
2162
2163 switch (opcode) {
2164 case HCI_OP_INQUIRY_CANCEL:
2165 hci_cc_inquiry_cancel(hdev, skb);
2166 break;
2167
2168 case HCI_OP_EXIT_PERIODIC_INQ:
2169 hci_cc_exit_periodic_inq(hdev, skb);
2170 break;
2171
2172 case HCI_OP_REMOTE_NAME_REQ_CANCEL:
2173 hci_cc_remote_name_req_cancel(hdev, skb);
2174 break;
2175
2176 case HCI_OP_ROLE_DISCOVERY:
2177 hci_cc_role_discovery(hdev, skb);
2178 break;
2179
Marcel Holtmanne4e8e372008-07-14 20:13:47 +02002180 case HCI_OP_READ_LINK_POLICY:
2181 hci_cc_read_link_policy(hdev, skb);
2182 break;
2183
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002184 case HCI_OP_WRITE_LINK_POLICY:
2185 hci_cc_write_link_policy(hdev, skb);
2186 break;
2187
Marcel Holtmanne4e8e372008-07-14 20:13:47 +02002188 case HCI_OP_READ_DEF_LINK_POLICY:
2189 hci_cc_read_def_link_policy(hdev, skb);
2190 break;
2191
2192 case HCI_OP_WRITE_DEF_LINK_POLICY:
2193 hci_cc_write_def_link_policy(hdev, skb);
2194 break;
2195
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002196 case HCI_OP_RESET:
2197 hci_cc_reset(hdev, skb);
2198 break;
2199
2200 case HCI_OP_WRITE_LOCAL_NAME:
2201 hci_cc_write_local_name(hdev, skb);
2202 break;
2203
2204 case HCI_OP_READ_LOCAL_NAME:
2205 hci_cc_read_local_name(hdev, skb);
2206 break;
2207
2208 case HCI_OP_WRITE_AUTH_ENABLE:
2209 hci_cc_write_auth_enable(hdev, skb);
2210 break;
2211
2212 case HCI_OP_WRITE_ENCRYPT_MODE:
2213 hci_cc_write_encrypt_mode(hdev, skb);
2214 break;
2215
2216 case HCI_OP_WRITE_SCAN_ENABLE:
2217 hci_cc_write_scan_enable(hdev, skb);
2218 break;
2219
2220 case HCI_OP_READ_CLASS_OF_DEV:
2221 hci_cc_read_class_of_dev(hdev, skb);
2222 break;
2223
2224 case HCI_OP_WRITE_CLASS_OF_DEV:
2225 hci_cc_write_class_of_dev(hdev, skb);
2226 break;
2227
2228 case HCI_OP_READ_VOICE_SETTING:
2229 hci_cc_read_voice_setting(hdev, skb);
2230 break;
2231
2232 case HCI_OP_WRITE_VOICE_SETTING:
2233 hci_cc_write_voice_setting(hdev, skb);
2234 break;
2235
2236 case HCI_OP_HOST_BUFFER_SIZE:
2237 hci_cc_host_buffer_size(hdev, skb);
2238 break;
2239
Marcel Holtmann333140b2008-07-14 20:13:48 +02002240 case HCI_OP_WRITE_SSP_MODE:
2241 hci_cc_write_ssp_mode(hdev, skb);
2242 break;
2243
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002244 case HCI_OP_READ_LOCAL_VERSION:
2245 hci_cc_read_local_version(hdev, skb);
2246 break;
2247
2248 case HCI_OP_READ_LOCAL_COMMANDS:
2249 hci_cc_read_local_commands(hdev, skb);
2250 break;
2251
2252 case HCI_OP_READ_LOCAL_FEATURES:
2253 hci_cc_read_local_features(hdev, skb);
2254 break;
2255
Andre Guedes971e3a42011-06-30 19:20:52 -03002256 case HCI_OP_READ_LOCAL_EXT_FEATURES:
2257 hci_cc_read_local_ext_features(hdev, skb);
2258 break;
2259
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002260 case HCI_OP_READ_BUFFER_SIZE:
2261 hci_cc_read_buffer_size(hdev, skb);
2262 break;
2263
2264 case HCI_OP_READ_BD_ADDR:
2265 hci_cc_read_bd_addr(hdev, skb);
2266 break;
2267
Andrei Emeltchenko350ee4c2011-12-07 15:56:51 +02002268 case HCI_OP_READ_DATA_BLOCK_SIZE:
2269 hci_cc_read_data_block_size(hdev, skb);
2270 break;
2271
Johan Hedberg23bb5762010-12-21 23:01:27 +02002272 case HCI_OP_WRITE_CA_TIMEOUT:
2273 hci_cc_write_ca_timeout(hdev, skb);
2274 break;
2275
Andrei Emeltchenko1e89cff2011-11-24 14:52:02 +02002276 case HCI_OP_READ_FLOW_CONTROL_MODE:
2277 hci_cc_read_flow_control_mode(hdev, skb);
2278 break;
2279
Andrei Emeltchenko928abaa2011-10-12 10:53:57 +03002280 case HCI_OP_READ_LOCAL_AMP_INFO:
2281 hci_cc_read_local_amp_info(hdev, skb);
2282 break;
2283
Johan Hedbergb0916ea2011-01-10 13:44:55 +02002284 case HCI_OP_DELETE_STORED_LINK_KEY:
2285 hci_cc_delete_stored_link_key(hdev, skb);
2286 break;
2287
Johan Hedbergd5859e22011-01-25 01:19:58 +02002288 case HCI_OP_SET_EVENT_MASK:
2289 hci_cc_set_event_mask(hdev, skb);
2290 break;
2291
2292 case HCI_OP_WRITE_INQUIRY_MODE:
2293 hci_cc_write_inquiry_mode(hdev, skb);
2294 break;
2295
2296 case HCI_OP_READ_INQ_RSP_TX_POWER:
2297 hci_cc_read_inq_rsp_tx_power(hdev, skb);
2298 break;
2299
2300 case HCI_OP_SET_EVENT_FLT:
2301 hci_cc_set_event_flt(hdev, skb);
2302 break;
2303
Johan Hedberg980e1a52011-01-22 06:10:07 +02002304 case HCI_OP_PIN_CODE_REPLY:
2305 hci_cc_pin_code_reply(hdev, skb);
2306 break;
2307
2308 case HCI_OP_PIN_CODE_NEG_REPLY:
2309 hci_cc_pin_code_neg_reply(hdev, skb);
2310 break;
2311
Szymon Jancc35938b2011-03-22 13:12:21 +01002312 case HCI_OP_READ_LOCAL_OOB_DATA:
2313 hci_cc_read_local_oob_data_reply(hdev, skb);
2314 break;
2315
Ville Tervo6ed58ec2011-02-10 22:38:48 -03002316 case HCI_OP_LE_READ_BUFFER_SIZE:
2317 hci_cc_le_read_buffer_size(hdev, skb);
2318 break;
2319
Johan Hedberga5c29682011-02-19 12:05:57 -03002320 case HCI_OP_USER_CONFIRM_REPLY:
2321 hci_cc_user_confirm_reply(hdev, skb);
2322 break;
2323
2324 case HCI_OP_USER_CONFIRM_NEG_REPLY:
2325 hci_cc_user_confirm_neg_reply(hdev, skb);
2326 break;
2327
Brian Gix1143d452011-11-23 08:28:34 -08002328 case HCI_OP_USER_PASSKEY_REPLY:
2329 hci_cc_user_passkey_reply(hdev, skb);
2330 break;
2331
2332 case HCI_OP_USER_PASSKEY_NEG_REPLY:
2333 hci_cc_user_passkey_neg_reply(hdev, skb);
Szymon Janc16cde992012-04-13 12:32:42 +02002334 break;
Andre Guedes07f7fa52011-12-02 21:13:31 +09002335
2336 case HCI_OP_LE_SET_SCAN_PARAM:
2337 hci_cc_le_set_scan_param(hdev, skb);
Brian Gix1143d452011-11-23 08:28:34 -08002338 break;
2339
Andre Guedeseb9d91f2011-05-26 16:23:52 -03002340 case HCI_OP_LE_SET_SCAN_ENABLE:
2341 hci_cc_le_set_scan_enable(hdev, skb);
2342 break;
2343
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03002344 case HCI_OP_LE_LTK_REPLY:
2345 hci_cc_le_ltk_reply(hdev, skb);
2346 break;
2347
2348 case HCI_OP_LE_LTK_NEG_REPLY:
2349 hci_cc_le_ltk_neg_reply(hdev, skb);
2350 break;
2351
Andre Guedesf9b49302011-06-30 19:20:53 -03002352 case HCI_OP_WRITE_LE_HOST_SUPPORTED:
2353 hci_cc_write_le_host_supported(hdev, skb);
2354 break;
2355
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002356 default:
2357 BT_DBG("%s opcode 0x%x", hdev->name, opcode);
2358 break;
2359 }
2360
Ville Tervo6bd32322011-02-16 16:32:41 +02002361 if (ev->opcode != HCI_OP_NOP)
2362 del_timer(&hdev->cmd_timer);
2363
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002364 if (ev->ncmd) {
2365 atomic_set(&hdev->cmd_cnt, 1);
2366 if (!skb_queue_empty(&hdev->cmd_q))
Gustavo F. Padovanc347b762011-12-14 23:53:47 -02002367 queue_work(hdev->workqueue, &hdev->cmd_work);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002368 }
2369}
2370
2371static inline void hci_cmd_status_evt(struct hci_dev *hdev, struct sk_buff *skb)
2372{
2373 struct hci_ev_cmd_status *ev = (void *) skb->data;
2374 __u16 opcode;
2375
2376 skb_pull(skb, sizeof(*ev));
2377
2378 opcode = __le16_to_cpu(ev->opcode);
2379
2380 switch (opcode) {
2381 case HCI_OP_INQUIRY:
2382 hci_cs_inquiry(hdev, ev->status);
2383 break;
2384
2385 case HCI_OP_CREATE_CONN:
2386 hci_cs_create_conn(hdev, ev->status);
2387 break;
2388
2389 case HCI_OP_ADD_SCO:
2390 hci_cs_add_sco(hdev, ev->status);
2391 break;
2392
Marcel Holtmannf8558552008-07-14 20:13:49 +02002393 case HCI_OP_AUTH_REQUESTED:
2394 hci_cs_auth_requested(hdev, ev->status);
2395 break;
2396
2397 case HCI_OP_SET_CONN_ENCRYPT:
2398 hci_cs_set_conn_encrypt(hdev, ev->status);
2399 break;
2400
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002401 case HCI_OP_REMOTE_NAME_REQ:
2402 hci_cs_remote_name_req(hdev, ev->status);
2403 break;
2404
Marcel Holtmann769be972008-07-14 20:13:49 +02002405 case HCI_OP_READ_REMOTE_FEATURES:
2406 hci_cs_read_remote_features(hdev, ev->status);
2407 break;
2408
2409 case HCI_OP_READ_REMOTE_EXT_FEATURES:
2410 hci_cs_read_remote_ext_features(hdev, ev->status);
2411 break;
2412
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002413 case HCI_OP_SETUP_SYNC_CONN:
2414 hci_cs_setup_sync_conn(hdev, ev->status);
2415 break;
2416
2417 case HCI_OP_SNIFF_MODE:
2418 hci_cs_sniff_mode(hdev, ev->status);
2419 break;
2420
2421 case HCI_OP_EXIT_SNIFF_MODE:
2422 hci_cs_exit_sniff_mode(hdev, ev->status);
2423 break;
2424
Johan Hedberg8962ee72011-01-20 12:40:27 +02002425 case HCI_OP_DISCONNECT:
Johan Hedberg88c3df12012-02-09 14:27:38 +02002426 hci_cs_disconnect(hdev, ev->status);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002427 break;
2428
Ville Tervofcd89c02011-02-10 22:38:47 -03002429 case HCI_OP_LE_CREATE_CONN:
2430 hci_cs_le_create_conn(hdev, ev->status);
2431 break;
2432
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03002433 case HCI_OP_LE_START_ENC:
2434 hci_cs_le_start_enc(hdev, ev->status);
2435 break;
2436
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002437 default:
2438 BT_DBG("%s opcode 0x%x", hdev->name, opcode);
2439 break;
2440 }
2441
Ville Tervo6bd32322011-02-16 16:32:41 +02002442 if (ev->opcode != HCI_OP_NOP)
2443 del_timer(&hdev->cmd_timer);
2444
Gustavo F. Padovan10572132011-03-16 15:36:29 -03002445 if (ev->ncmd && !test_bit(HCI_RESET, &hdev->flags)) {
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002446 atomic_set(&hdev->cmd_cnt, 1);
2447 if (!skb_queue_empty(&hdev->cmd_q))
Gustavo F. Padovanc347b762011-12-14 23:53:47 -02002448 queue_work(hdev->workqueue, &hdev->cmd_work);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002449 }
2450}
2451
2452static inline void hci_role_change_evt(struct hci_dev *hdev, struct sk_buff *skb)
2453{
2454 struct hci_ev_role_change *ev = (void *) skb->data;
2455 struct hci_conn *conn;
2456
2457 BT_DBG("%s status %d", hdev->name, ev->status);
2458
2459 hci_dev_lock(hdev);
2460
2461 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
2462 if (conn) {
2463 if (!ev->status) {
2464 if (ev->role)
2465 conn->link_mode &= ~HCI_LM_MASTER;
2466 else
2467 conn->link_mode |= HCI_LM_MASTER;
2468 }
2469
Johan Hedberg51a8efd2012-01-16 06:10:31 +02002470 clear_bit(HCI_CONN_RSWITCH_PEND, &conn->flags);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002471
2472 hci_role_switch_cfm(conn, ev->status, ev->role);
2473 }
2474
2475 hci_dev_unlock(hdev);
2476}
2477
Linus Torvalds1da177e2005-04-16 15:20:36 -07002478static inline void hci_num_comp_pkts_evt(struct hci_dev *hdev, struct sk_buff *skb)
2479{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002480 struct hci_ev_num_comp_pkts *ev = (void *) skb->data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002481 int i;
2482
Andrei Emeltchenko32ac5b92011-12-19 16:31:29 +02002483 if (hdev->flow_ctl_mode != HCI_FLOW_CTL_MODE_PACKET_BASED) {
2484 BT_ERR("Wrong event for mode %d", hdev->flow_ctl_mode);
2485 return;
2486 }
2487
Andrei Emeltchenkoc5993de2011-12-30 12:07:47 +02002488 if (skb->len < sizeof(*ev) || skb->len < sizeof(*ev) +
2489 ev->num_hndl * sizeof(struct hci_comp_pkts_info)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002490 BT_DBG("%s bad parameters", hdev->name);
2491 return;
2492 }
2493
Andrei Emeltchenkoc5993de2011-12-30 12:07:47 +02002494 BT_DBG("%s num_hndl %d", hdev->name, ev->num_hndl);
2495
Andrei Emeltchenko613a1c02011-12-19 16:31:30 +02002496 for (i = 0; i < ev->num_hndl; i++) {
2497 struct hci_comp_pkts_info *info = &ev->handles[i];
Linus Torvalds1da177e2005-04-16 15:20:36 -07002498 struct hci_conn *conn;
2499 __u16 handle, count;
2500
Andrei Emeltchenko613a1c02011-12-19 16:31:30 +02002501 handle = __le16_to_cpu(info->handle);
2502 count = __le16_to_cpu(info->count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002503
2504 conn = hci_conn_hash_lookup_handle(hdev, handle);
Andrei Emeltchenkof4280912011-12-07 15:56:52 +02002505 if (!conn)
2506 continue;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002507
Andrei Emeltchenkof4280912011-12-07 15:56:52 +02002508 conn->sent -= count;
2509
2510 switch (conn->type) {
2511 case ACL_LINK:
2512 hdev->acl_cnt += count;
2513 if (hdev->acl_cnt > hdev->acl_pkts)
2514 hdev->acl_cnt = hdev->acl_pkts;
2515 break;
2516
2517 case LE_LINK:
2518 if (hdev->le_pkts) {
2519 hdev->le_cnt += count;
2520 if (hdev->le_cnt > hdev->le_pkts)
2521 hdev->le_cnt = hdev->le_pkts;
2522 } else {
Andrei Emeltchenko70f230202010-12-01 16:58:25 +02002523 hdev->acl_cnt += count;
2524 if (hdev->acl_cnt > hdev->acl_pkts)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002525 hdev->acl_cnt = hdev->acl_pkts;
2526 }
Andrei Emeltchenkof4280912011-12-07 15:56:52 +02002527 break;
2528
2529 case SCO_LINK:
2530 hdev->sco_cnt += count;
2531 if (hdev->sco_cnt > hdev->sco_pkts)
2532 hdev->sco_cnt = hdev->sco_pkts;
2533 break;
2534
2535 default:
2536 BT_ERR("Unknown type %d conn %p", conn->type, conn);
2537 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002538 }
2539 }
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002540
Gustavo F. Padovan3eff45e2011-12-15 00:50:02 -02002541 queue_work(hdev->workqueue, &hdev->tx_work);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002542}
2543
Andrei Emeltchenko25e89e92012-01-04 12:41:58 +02002544static inline void hci_num_comp_blocks_evt(struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002545 struct sk_buff *skb)
Andrei Emeltchenko25e89e92012-01-04 12:41:58 +02002546{
2547 struct hci_ev_num_comp_blocks *ev = (void *) skb->data;
2548 int i;
2549
2550 if (hdev->flow_ctl_mode != HCI_FLOW_CTL_MODE_BLOCK_BASED) {
2551 BT_ERR("Wrong event for mode %d", hdev->flow_ctl_mode);
2552 return;
2553 }
2554
2555 if (skb->len < sizeof(*ev) || skb->len < sizeof(*ev) +
2556 ev->num_hndl * sizeof(struct hci_comp_blocks_info)) {
2557 BT_DBG("%s bad parameters", hdev->name);
2558 return;
2559 }
2560
2561 BT_DBG("%s num_blocks %d num_hndl %d", hdev->name, ev->num_blocks,
2562 ev->num_hndl);
2563
2564 for (i = 0; i < ev->num_hndl; i++) {
2565 struct hci_comp_blocks_info *info = &ev->handles[i];
2566 struct hci_conn *conn;
2567 __u16 handle, block_count;
2568
2569 handle = __le16_to_cpu(info->handle);
2570 block_count = __le16_to_cpu(info->blocks);
2571
2572 conn = hci_conn_hash_lookup_handle(hdev, handle);
2573 if (!conn)
2574 continue;
2575
2576 conn->sent -= block_count;
2577
2578 switch (conn->type) {
2579 case ACL_LINK:
2580 hdev->block_cnt += block_count;
2581 if (hdev->block_cnt > hdev->num_blocks)
2582 hdev->block_cnt = hdev->num_blocks;
2583 break;
2584
2585 default:
2586 BT_ERR("Unknown type %d conn %p", conn->type, conn);
2587 break;
2588 }
2589 }
2590
2591 queue_work(hdev->workqueue, &hdev->tx_work);
2592}
2593
Marcel Holtmann04837f62006-07-03 10:02:33 +02002594static inline void hci_mode_change_evt(struct hci_dev *hdev, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002595{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002596 struct hci_ev_mode_change *ev = (void *) skb->data;
Marcel Holtmann04837f62006-07-03 10:02:33 +02002597 struct hci_conn *conn;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002598
2599 BT_DBG("%s status %d", hdev->name, ev->status);
2600
2601 hci_dev_lock(hdev);
2602
Marcel Holtmann04837f62006-07-03 10:02:33 +02002603 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
2604 if (conn) {
2605 conn->mode = ev->mode;
2606 conn->interval = __le16_to_cpu(ev->interval);
2607
Johan Hedberg51a8efd2012-01-16 06:10:31 +02002608 if (!test_and_clear_bit(HCI_CONN_MODE_CHANGE_PEND, &conn->flags)) {
Marcel Holtmann04837f62006-07-03 10:02:33 +02002609 if (conn->mode == HCI_CM_ACTIVE)
Johan Hedberg58a681e2012-01-16 06:47:28 +02002610 set_bit(HCI_CONN_POWER_SAVE, &conn->flags);
Marcel Holtmann04837f62006-07-03 10:02:33 +02002611 else
Johan Hedberg58a681e2012-01-16 06:47:28 +02002612 clear_bit(HCI_CONN_POWER_SAVE, &conn->flags);
Marcel Holtmann04837f62006-07-03 10:02:33 +02002613 }
Marcel Holtmanne73439d2010-07-26 10:06:00 -04002614
Johan Hedberg51a8efd2012-01-16 06:10:31 +02002615 if (test_and_clear_bit(HCI_CONN_SCO_SETUP_PEND, &conn->flags))
Marcel Holtmanne73439d2010-07-26 10:06:00 -04002616 hci_sco_setup(conn, ev->status);
Marcel Holtmann04837f62006-07-03 10:02:33 +02002617 }
2618
2619 hci_dev_unlock(hdev);
2620}
2621
Linus Torvalds1da177e2005-04-16 15:20:36 -07002622static inline void hci_pin_code_request_evt(struct hci_dev *hdev, struct sk_buff *skb)
2623{
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002624 struct hci_ev_pin_code_req *ev = (void *) skb->data;
2625 struct hci_conn *conn;
2626
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002627 BT_DBG("%s", hdev->name);
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002628
2629 hci_dev_lock(hdev);
2630
2631 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
Waldemar Rymarkiewiczb6f98042011-09-23 10:01:30 +02002632 if (!conn)
2633 goto unlock;
2634
2635 if (conn->state == BT_CONNECTED) {
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002636 hci_conn_hold(conn);
2637 conn->disc_timeout = HCI_PAIRING_TIMEOUT;
2638 hci_conn_put(conn);
2639 }
2640
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02002641 if (!test_bit(HCI_PAIRABLE, &hdev->dev_flags))
Johan Hedberg03b555e2011-01-04 15:40:05 +02002642 hci_send_cmd(hdev, HCI_OP_PIN_CODE_NEG_REPLY,
2643 sizeof(ev->bdaddr), &ev->bdaddr);
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02002644 else if (test_bit(HCI_MGMT, &hdev->dev_flags)) {
Waldemar Rymarkiewicza770bb52011-04-28 12:07:59 +02002645 u8 secure;
2646
2647 if (conn->pending_sec_level == BT_SECURITY_HIGH)
2648 secure = 1;
2649 else
2650 secure = 0;
2651
Johan Hedberg744cf192011-11-08 20:40:14 +02002652 mgmt_pin_code_request(hdev, &ev->bdaddr, secure);
Waldemar Rymarkiewicza770bb52011-04-28 12:07:59 +02002653 }
Johan Hedberg980e1a52011-01-22 06:10:07 +02002654
Waldemar Rymarkiewiczb6f98042011-09-23 10:01:30 +02002655unlock:
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002656 hci_dev_unlock(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002657}
2658
Linus Torvalds1da177e2005-04-16 15:20:36 -07002659static inline void hci_link_key_request_evt(struct hci_dev *hdev, struct sk_buff *skb)
2660{
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002661 struct hci_ev_link_key_req *ev = (void *) skb->data;
2662 struct hci_cp_link_key_reply cp;
2663 struct hci_conn *conn;
2664 struct link_key *key;
2665
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002666 BT_DBG("%s", hdev->name);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002667
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02002668 if (!test_bit(HCI_LINK_KEYS, &hdev->dev_flags))
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002669 return;
2670
2671 hci_dev_lock(hdev);
2672
2673 key = hci_find_link_key(hdev, &ev->bdaddr);
2674 if (!key) {
2675 BT_DBG("%s link key not found for %s", hdev->name,
2676 batostr(&ev->bdaddr));
2677 goto not_found;
2678 }
2679
2680 BT_DBG("%s found key type %u for %s", hdev->name, key->type,
2681 batostr(&ev->bdaddr));
2682
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02002683 if (!test_bit(HCI_DEBUG_KEYS, &hdev->dev_flags) &&
Waldemar Rymarkiewiczb6020ba2011-04-28 12:07:53 +02002684 key->type == HCI_LK_DEBUG_COMBINATION) {
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002685 BT_DBG("%s ignoring debug key", hdev->name);
2686 goto not_found;
2687 }
2688
2689 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
Waldemar Rymarkiewicz60b83f52011-04-28 12:07:56 +02002690 if (conn) {
2691 if (key->type == HCI_LK_UNAUTH_COMBINATION &&
2692 conn->auth_type != 0xff &&
2693 (conn->auth_type & 0x01)) {
2694 BT_DBG("%s ignoring unauthenticated key", hdev->name);
2695 goto not_found;
2696 }
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002697
Waldemar Rymarkiewicz60b83f52011-04-28 12:07:56 +02002698 if (key->type == HCI_LK_COMBINATION && key->pin_len < 16 &&
2699 conn->pending_sec_level == BT_SECURITY_HIGH) {
2700 BT_DBG("%s ignoring key unauthenticated for high \
2701 security", hdev->name);
2702 goto not_found;
2703 }
2704
2705 conn->key_type = key->type;
2706 conn->pin_length = key->pin_len;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002707 }
2708
2709 bacpy(&cp.bdaddr, &ev->bdaddr);
2710 memcpy(cp.link_key, key->val, 16);
2711
2712 hci_send_cmd(hdev, HCI_OP_LINK_KEY_REPLY, sizeof(cp), &cp);
2713
2714 hci_dev_unlock(hdev);
2715
2716 return;
2717
2718not_found:
2719 hci_send_cmd(hdev, HCI_OP_LINK_KEY_NEG_REPLY, 6, &ev->bdaddr);
2720 hci_dev_unlock(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002721}
2722
Linus Torvalds1da177e2005-04-16 15:20:36 -07002723static inline void hci_link_key_notify_evt(struct hci_dev *hdev, struct sk_buff *skb)
2724{
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002725 struct hci_ev_link_key_notify *ev = (void *) skb->data;
2726 struct hci_conn *conn;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002727 u8 pin_len = 0;
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002728
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002729 BT_DBG("%s", hdev->name);
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002730
2731 hci_dev_lock(hdev);
2732
2733 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
2734 if (conn) {
2735 hci_conn_hold(conn);
2736 conn->disc_timeout = HCI_DISCONN_TIMEOUT;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002737 pin_len = conn->pin_length;
Waldemar Rymarkiewicz13d39312011-04-28 12:07:55 +02002738
2739 if (ev->key_type != HCI_LK_CHANGED_COMBINATION)
2740 conn->key_type = ev->key_type;
2741
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002742 hci_conn_put(conn);
2743 }
2744
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02002745 if (test_bit(HCI_LINK_KEYS, &hdev->dev_flags))
Johan Hedbergd25e28a2011-04-28 11:28:59 -07002746 hci_add_link_key(hdev, conn, 1, &ev->bdaddr, ev->link_key,
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002747 ev->key_type, pin_len);
2748
Marcel Holtmann052b30b2009-04-26 20:01:22 +02002749 hci_dev_unlock(hdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002750}
2751
Marcel Holtmann04837f62006-07-03 10:02:33 +02002752static inline void hci_clock_offset_evt(struct hci_dev *hdev, struct sk_buff *skb)
2753{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002754 struct hci_ev_clock_offset *ev = (void *) skb->data;
Marcel Holtmann04837f62006-07-03 10:02:33 +02002755 struct hci_conn *conn;
2756
2757 BT_DBG("%s status %d", hdev->name, ev->status);
2758
2759 hci_dev_lock(hdev);
2760
2761 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002762 if (conn && !ev->status) {
2763 struct inquiry_entry *ie;
2764
Andrei Emeltchenkocc11b9c2010-11-22 13:21:37 +02002765 ie = hci_inquiry_cache_lookup(hdev, &conn->dst);
2766 if (ie) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002767 ie->data.clock_offset = ev->clock_offset;
2768 ie->timestamp = jiffies;
2769 }
2770 }
2771
2772 hci_dev_unlock(hdev);
2773}
2774
Marcel Holtmanna8746412008-07-14 20:13:46 +02002775static inline void hci_pkt_type_change_evt(struct hci_dev *hdev, struct sk_buff *skb)
2776{
2777 struct hci_ev_pkt_type_change *ev = (void *) skb->data;
2778 struct hci_conn *conn;
2779
2780 BT_DBG("%s status %d", hdev->name, ev->status);
2781
2782 hci_dev_lock(hdev);
2783
2784 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
2785 if (conn && !ev->status)
2786 conn->pkt_type = __le16_to_cpu(ev->pkt_type);
2787
2788 hci_dev_unlock(hdev);
2789}
2790
Marcel Holtmann85a1e932005-08-09 20:28:02 -07002791static inline void hci_pscan_rep_mode_evt(struct hci_dev *hdev, struct sk_buff *skb)
2792{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002793 struct hci_ev_pscan_rep_mode *ev = (void *) skb->data;
Marcel Holtmann85a1e932005-08-09 20:28:02 -07002794 struct inquiry_entry *ie;
2795
2796 BT_DBG("%s", hdev->name);
2797
2798 hci_dev_lock(hdev);
2799
Andrei Emeltchenkocc11b9c2010-11-22 13:21:37 +02002800 ie = hci_inquiry_cache_lookup(hdev, &ev->bdaddr);
2801 if (ie) {
Marcel Holtmann85a1e932005-08-09 20:28:02 -07002802 ie->data.pscan_rep_mode = ev->pscan_rep_mode;
2803 ie->timestamp = jiffies;
2804 }
2805
2806 hci_dev_unlock(hdev);
2807}
2808
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002809static inline void hci_inquiry_result_with_rssi_evt(struct hci_dev *hdev, struct sk_buff *skb)
2810{
2811 struct inquiry_data data;
2812 int num_rsp = *((__u8 *) skb->data);
Johan Hedberg388fc8f2012-02-23 00:38:59 +02002813 bool name_known, ssp;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002814
2815 BT_DBG("%s num_rsp %d", hdev->name, num_rsp);
2816
2817 if (!num_rsp)
2818 return;
2819
2820 hci_dev_lock(hdev);
2821
2822 if ((skb->len - 1) / num_rsp != sizeof(struct inquiry_info_with_rssi)) {
Szymon Janc138d22e2011-02-17 16:44:23 +01002823 struct inquiry_info_with_rssi_and_pscan_mode *info;
2824 info = (void *) (skb->data + 1);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002825
Johan Hedberge17acd42011-03-30 23:57:16 +03002826 for (; num_rsp; num_rsp--, info++) {
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002827 bacpy(&data.bdaddr, &info->bdaddr);
2828 data.pscan_rep_mode = info->pscan_rep_mode;
2829 data.pscan_period_mode = info->pscan_period_mode;
2830 data.pscan_mode = info->pscan_mode;
2831 memcpy(data.dev_class, info->dev_class, 3);
2832 data.clock_offset = info->clock_offset;
2833 data.rssi = info->rssi;
Marcel Holtmann41a96212008-07-14 20:13:48 +02002834 data.ssp_mode = 0x00;
Johan Hedberg31754052012-01-04 13:39:52 +02002835
2836 name_known = hci_inquiry_cache_update(hdev, &data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002837 false, &ssp);
Johan Hedberg48264f02011-11-09 13:58:58 +02002838 mgmt_device_found(hdev, &info->bdaddr, ACL_LINK, 0x00,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002839 info->dev_class, info->rssi,
2840 !name_known, ssp, NULL, 0);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002841 }
2842 } else {
2843 struct inquiry_info_with_rssi *info = (void *) (skb->data + 1);
2844
Johan Hedberge17acd42011-03-30 23:57:16 +03002845 for (; num_rsp; num_rsp--, info++) {
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002846 bacpy(&data.bdaddr, &info->bdaddr);
2847 data.pscan_rep_mode = info->pscan_rep_mode;
2848 data.pscan_period_mode = info->pscan_period_mode;
2849 data.pscan_mode = 0x00;
2850 memcpy(data.dev_class, info->dev_class, 3);
2851 data.clock_offset = info->clock_offset;
2852 data.rssi = info->rssi;
Marcel Holtmann41a96212008-07-14 20:13:48 +02002853 data.ssp_mode = 0x00;
Johan Hedberg31754052012-01-04 13:39:52 +02002854 name_known = hci_inquiry_cache_update(hdev, &data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002855 false, &ssp);
Johan Hedberg48264f02011-11-09 13:58:58 +02002856 mgmt_device_found(hdev, &info->bdaddr, ACL_LINK, 0x00,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002857 info->dev_class, info->rssi,
2858 !name_known, ssp, NULL, 0);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002859 }
2860 }
2861
2862 hci_dev_unlock(hdev);
2863}
2864
2865static inline void hci_remote_ext_features_evt(struct hci_dev *hdev, struct sk_buff *skb)
2866{
Marcel Holtmann41a96212008-07-14 20:13:48 +02002867 struct hci_ev_remote_ext_features *ev = (void *) skb->data;
2868 struct hci_conn *conn;
2869
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002870 BT_DBG("%s", hdev->name);
Marcel Holtmann41a96212008-07-14 20:13:48 +02002871
Marcel Holtmann41a96212008-07-14 20:13:48 +02002872 hci_dev_lock(hdev);
2873
2874 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
Johan Hedbergccd556f2010-11-10 17:11:51 +02002875 if (!conn)
2876 goto unlock;
Marcel Holtmann41a96212008-07-14 20:13:48 +02002877
Johan Hedbergccd556f2010-11-10 17:11:51 +02002878 if (!ev->status && ev->page == 0x01) {
2879 struct inquiry_entry *ie;
Marcel Holtmann41a96212008-07-14 20:13:48 +02002880
Andrei Emeltchenkocc11b9c2010-11-22 13:21:37 +02002881 ie = hci_inquiry_cache_lookup(hdev, &conn->dst);
2882 if (ie)
Johan Hedberg02b7cc62012-02-28 02:28:43 +02002883 ie->data.ssp_mode = (ev->features[0] & LMP_HOST_SSP);
Marcel Holtmann769be972008-07-14 20:13:49 +02002884
Johan Hedberg02b7cc62012-02-28 02:28:43 +02002885 if (ev->features[0] & LMP_HOST_SSP)
Johan Hedberg58a681e2012-01-16 06:47:28 +02002886 set_bit(HCI_CONN_SSP_ENABLED, &conn->flags);
Marcel Holtmann41a96212008-07-14 20:13:48 +02002887 }
2888
Johan Hedbergccd556f2010-11-10 17:11:51 +02002889 if (conn->state != BT_CONFIG)
2890 goto unlock;
2891
Johan Hedberg671267b2012-05-12 16:11:50 -03002892 if (!ev->status && !test_bit(HCI_CONN_MGMT_CONNECTED, &conn->flags)) {
Johan Hedberg127178d2010-11-18 22:22:29 +02002893 struct hci_cp_remote_name_req cp;
2894 memset(&cp, 0, sizeof(cp));
2895 bacpy(&cp.bdaddr, &conn->dst);
2896 cp.pscan_rep_mode = 0x02;
2897 hci_send_cmd(hdev, HCI_OP_REMOTE_NAME_REQ, sizeof(cp), &cp);
Johan Hedbergb644ba32012-01-17 21:48:47 +02002898 } else if (!test_and_set_bit(HCI_CONN_MGMT_CONNECTED, &conn->flags))
2899 mgmt_device_connected(hdev, &conn->dst, conn->type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002900 conn->dst_type, 0, NULL, 0,
2901 conn->dev_class);
Johan Hedberg392599b2010-11-18 22:22:28 +02002902
Johan Hedberg127178d2010-11-18 22:22:29 +02002903 if (!hci_outgoing_auth_needed(hdev, conn)) {
Johan Hedbergccd556f2010-11-10 17:11:51 +02002904 conn->state = BT_CONNECTED;
2905 hci_proto_connect_cfm(conn, ev->status);
2906 hci_conn_put(conn);
2907 }
2908
2909unlock:
Marcel Holtmann41a96212008-07-14 20:13:48 +02002910 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002911}
2912
2913static inline void hci_sync_conn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
2914{
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02002915 struct hci_ev_sync_conn_complete *ev = (void *) skb->data;
2916 struct hci_conn *conn;
2917
2918 BT_DBG("%s status %d", hdev->name, ev->status);
2919
2920 hci_dev_lock(hdev);
2921
2922 conn = hci_conn_hash_lookup_ba(hdev, ev->link_type, &ev->bdaddr);
Marcel Holtmann9dc0a3a2008-07-14 20:13:46 +02002923 if (!conn) {
2924 if (ev->link_type == ESCO_LINK)
2925 goto unlock;
2926
2927 conn = hci_conn_hash_lookup_ba(hdev, ESCO_LINK, &ev->bdaddr);
2928 if (!conn)
2929 goto unlock;
2930
2931 conn->type = SCO_LINK;
2932 }
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02002933
Marcel Holtmann732547f2009-04-19 19:14:14 +02002934 switch (ev->status) {
2935 case 0x00:
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02002936 conn->handle = __le16_to_cpu(ev->handle);
2937 conn->state = BT_CONNECTED;
Marcel Holtmann7d0db0a2008-07-14 20:13:51 +02002938
Marcel Holtmann9eba32b2009-08-22 14:19:26 -07002939 hci_conn_hold_device(conn);
Marcel Holtmann7d0db0a2008-07-14 20:13:51 +02002940 hci_conn_add_sysfs(conn);
Marcel Holtmann732547f2009-04-19 19:14:14 +02002941 break;
2942
Stephen Coe705e5712010-02-16 11:29:44 -05002943 case 0x11: /* Unsupported Feature or Parameter Value */
Marcel Holtmann732547f2009-04-19 19:14:14 +02002944 case 0x1c: /* SCO interval rejected */
Nick Pelly1038a002010-02-03 11:42:26 -08002945 case 0x1a: /* Unsupported Remote Feature */
Marcel Holtmann732547f2009-04-19 19:14:14 +02002946 case 0x1f: /* Unspecified error */
2947 if (conn->out && conn->attempt < 2) {
2948 conn->pkt_type = (hdev->esco_type & SCO_ESCO_MASK) |
2949 (hdev->esco_type & EDR_ESCO_MASK);
2950 hci_setup_sync(conn, conn->link->handle);
2951 goto unlock;
2952 }
2953 /* fall through */
2954
2955 default:
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02002956 conn->state = BT_CLOSED;
Marcel Holtmann732547f2009-04-19 19:14:14 +02002957 break;
2958 }
Marcel Holtmannb6a0dc82007-10-20 14:55:10 +02002959
2960 hci_proto_connect_cfm(conn, ev->status);
2961 if (ev->status)
2962 hci_conn_del(conn);
2963
2964unlock:
2965 hci_dev_unlock(hdev);
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002966}
2967
2968static inline void hci_sync_conn_changed_evt(struct hci_dev *hdev, struct sk_buff *skb)
2969{
2970 BT_DBG("%s", hdev->name);
2971}
2972
Marcel Holtmann04837f62006-07-03 10:02:33 +02002973static inline void hci_sniff_subrate_evt(struct hci_dev *hdev, struct sk_buff *skb)
2974{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002975 struct hci_ev_sniff_subrate *ev = (void *) skb->data;
Marcel Holtmann04837f62006-07-03 10:02:33 +02002976
2977 BT_DBG("%s status %d", hdev->name, ev->status);
Marcel Holtmann04837f62006-07-03 10:02:33 +02002978}
2979
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002980static inline void hci_extended_inquiry_result_evt(struct hci_dev *hdev, struct sk_buff *skb)
2981{
2982 struct inquiry_data data;
2983 struct extended_inquiry_info *info = (void *) (skb->data + 1);
2984 int num_rsp = *((__u8 *) skb->data);
2985
2986 BT_DBG("%s num_rsp %d", hdev->name, num_rsp);
2987
2988 if (!num_rsp)
2989 return;
2990
2991 hci_dev_lock(hdev);
2992
Johan Hedberge17acd42011-03-30 23:57:16 +03002993 for (; num_rsp; num_rsp--, info++) {
Johan Hedberg388fc8f2012-02-23 00:38:59 +02002994 bool name_known, ssp;
Johan Hedberg561aafb2012-01-04 13:31:59 +02002995
Marcel Holtmanna9de9242007-10-20 13:33:56 +02002996 bacpy(&data.bdaddr, &info->bdaddr);
Szymon Janc138d22e2011-02-17 16:44:23 +01002997 data.pscan_rep_mode = info->pscan_rep_mode;
2998 data.pscan_period_mode = info->pscan_period_mode;
2999 data.pscan_mode = 0x00;
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003000 memcpy(data.dev_class, info->dev_class, 3);
Szymon Janc138d22e2011-02-17 16:44:23 +01003001 data.clock_offset = info->clock_offset;
3002 data.rssi = info->rssi;
Marcel Holtmann41a96212008-07-14 20:13:48 +02003003 data.ssp_mode = 0x01;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003004
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02003005 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg4ddb1932012-01-15 20:04:43 +02003006 name_known = eir_has_data_type(info->data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003007 sizeof(info->data),
3008 EIR_NAME_COMPLETE);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003009 else
3010 name_known = true;
3011
Johan Hedberg388fc8f2012-02-23 00:38:59 +02003012 name_known = hci_inquiry_cache_update(hdev, &data, name_known,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003013 &ssp);
Johan Hedberg48264f02011-11-09 13:58:58 +02003014 mgmt_device_found(hdev, &info->bdaddr, ACL_LINK, 0x00,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003015 info->dev_class, info->rssi, !name_known,
3016 ssp, info->data, sizeof(info->data));
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003017 }
3018
3019 hci_dev_unlock(hdev);
3020}
3021
Johan Hedberg17fa4b92011-01-25 13:28:33 +02003022static inline u8 hci_get_auth_req(struct hci_conn *conn)
3023{
3024 /* If remote requests dedicated bonding follow that lead */
3025 if (conn->remote_auth == 0x02 || conn->remote_auth == 0x03) {
3026 /* If both remote and local IO capabilities allow MITM
3027 * protection then require it, otherwise don't */
3028 if (conn->remote_cap == 0x03 || conn->io_capability == 0x03)
3029 return 0x02;
3030 else
3031 return 0x03;
3032 }
3033
3034 /* If remote requests no-bonding follow that lead */
3035 if (conn->remote_auth == 0x00 || conn->remote_auth == 0x01)
Waldemar Rymarkiewicz58797bf2011-04-28 12:07:58 +02003036 return conn->remote_auth | (conn->auth_type & 0x01);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02003037
3038 return conn->auth_type;
3039}
3040
Marcel Holtmann04936842008-07-14 20:13:48 +02003041static inline void hci_io_capa_request_evt(struct hci_dev *hdev, struct sk_buff *skb)
3042{
3043 struct hci_ev_io_capa_request *ev = (void *) skb->data;
3044 struct hci_conn *conn;
3045
3046 BT_DBG("%s", hdev->name);
3047
3048 hci_dev_lock(hdev);
3049
3050 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
Johan Hedberg03b555e2011-01-04 15:40:05 +02003051 if (!conn)
3052 goto unlock;
Marcel Holtmann04936842008-07-14 20:13:48 +02003053
Johan Hedberg03b555e2011-01-04 15:40:05 +02003054 hci_conn_hold(conn);
3055
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02003056 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg03b555e2011-01-04 15:40:05 +02003057 goto unlock;
3058
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02003059 if (test_bit(HCI_PAIRABLE, &hdev->dev_flags) ||
Johan Hedberg03b555e2011-01-04 15:40:05 +02003060 (conn->remote_auth & ~0x01) == HCI_AT_NO_BONDING) {
Johan Hedberg17fa4b92011-01-25 13:28:33 +02003061 struct hci_cp_io_capability_reply cp;
3062
3063 bacpy(&cp.bdaddr, &ev->bdaddr);
Hemant Gupta7a7f1e72012-01-16 13:34:29 +05303064 /* Change the IO capability from KeyboardDisplay
3065 * to DisplayYesNo as it is not supported by BT spec. */
3066 cp.capability = (conn->io_capability == 0x04) ?
3067 0x01 : conn->io_capability;
Johan Hedberg7cbc9bd2011-04-28 11:29:04 -07003068 conn->auth_type = hci_get_auth_req(conn);
3069 cp.authentication = conn->auth_type;
Johan Hedberg17fa4b92011-01-25 13:28:33 +02003070
Johan Hedberg58a681e2012-01-16 06:47:28 +02003071 if ((conn->out || test_bit(HCI_CONN_REMOTE_OOB, &conn->flags)) &&
Szymon Jancce85ee12011-03-22 13:12:23 +01003072 hci_find_remote_oob_data(hdev, &conn->dst))
3073 cp.oob_data = 0x01;
3074 else
3075 cp.oob_data = 0x00;
3076
Johan Hedberg17fa4b92011-01-25 13:28:33 +02003077 hci_send_cmd(hdev, HCI_OP_IO_CAPABILITY_REPLY,
3078 sizeof(cp), &cp);
Johan Hedberg03b555e2011-01-04 15:40:05 +02003079 } else {
3080 struct hci_cp_io_capability_neg_reply cp;
3081
3082 bacpy(&cp.bdaddr, &ev->bdaddr);
Andrei Emeltchenko9f5a0d72011-11-07 14:20:25 +02003083 cp.reason = HCI_ERROR_PAIRING_NOT_ALLOWED;
Johan Hedberg03b555e2011-01-04 15:40:05 +02003084
3085 hci_send_cmd(hdev, HCI_OP_IO_CAPABILITY_NEG_REPLY,
3086 sizeof(cp), &cp);
3087 }
3088
3089unlock:
3090 hci_dev_unlock(hdev);
3091}
3092
3093static inline void hci_io_capa_reply_evt(struct hci_dev *hdev, struct sk_buff *skb)
3094{
3095 struct hci_ev_io_capa_reply *ev = (void *) skb->data;
3096 struct hci_conn *conn;
3097
3098 BT_DBG("%s", hdev->name);
3099
3100 hci_dev_lock(hdev);
3101
3102 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
3103 if (!conn)
3104 goto unlock;
3105
Johan Hedberg03b555e2011-01-04 15:40:05 +02003106 conn->remote_cap = ev->capability;
Johan Hedberg03b555e2011-01-04 15:40:05 +02003107 conn->remote_auth = ev->authentication;
Johan Hedberg58a681e2012-01-16 06:47:28 +02003108 if (ev->oob_data)
3109 set_bit(HCI_CONN_REMOTE_OOB, &conn->flags);
Johan Hedberg03b555e2011-01-04 15:40:05 +02003110
3111unlock:
Marcel Holtmann04936842008-07-14 20:13:48 +02003112 hci_dev_unlock(hdev);
3113}
3114
Johan Hedberga5c29682011-02-19 12:05:57 -03003115static inline void hci_user_confirm_request_evt(struct hci_dev *hdev,
3116 struct sk_buff *skb)
3117{
3118 struct hci_ev_user_confirm_req *ev = (void *) skb->data;
Johan Hedberg55bc1a32011-04-28 11:28:56 -07003119 int loc_mitm, rem_mitm, confirm_hint = 0;
Johan Hedberg7a828902011-04-28 11:28:53 -07003120 struct hci_conn *conn;
Johan Hedberga5c29682011-02-19 12:05:57 -03003121
3122 BT_DBG("%s", hdev->name);
3123
3124 hci_dev_lock(hdev);
3125
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02003126 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg7a828902011-04-28 11:28:53 -07003127 goto unlock;
Johan Hedberga5c29682011-02-19 12:05:57 -03003128
Johan Hedberg7a828902011-04-28 11:28:53 -07003129 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
3130 if (!conn)
3131 goto unlock;
3132
3133 loc_mitm = (conn->auth_type & 0x01);
3134 rem_mitm = (conn->remote_auth & 0x01);
3135
3136 /* If we require MITM but the remote device can't provide that
3137 * (it has NoInputNoOutput) then reject the confirmation
3138 * request. The only exception is when we're dedicated bonding
3139 * initiators (connect_cfm_cb set) since then we always have the MITM
3140 * bit set. */
3141 if (!conn->connect_cfm_cb && loc_mitm && conn->remote_cap == 0x03) {
3142 BT_DBG("Rejecting request: remote device can't provide MITM");
3143 hci_send_cmd(hdev, HCI_OP_USER_CONFIRM_NEG_REPLY,
3144 sizeof(ev->bdaddr), &ev->bdaddr);
3145 goto unlock;
3146 }
3147
3148 /* If no side requires MITM protection; auto-accept */
3149 if ((!loc_mitm || conn->remote_cap == 0x03) &&
3150 (!rem_mitm || conn->io_capability == 0x03)) {
Johan Hedberg55bc1a32011-04-28 11:28:56 -07003151
3152 /* If we're not the initiators request authorization to
3153 * proceed from user space (mgmt_user_confirm with
3154 * confirm_hint set to 1). */
Johan Hedberg51a8efd2012-01-16 06:10:31 +02003155 if (!test_bit(HCI_CONN_AUTH_PEND, &conn->flags)) {
Johan Hedberg55bc1a32011-04-28 11:28:56 -07003156 BT_DBG("Confirming auto-accept as acceptor");
3157 confirm_hint = 1;
3158 goto confirm;
3159 }
3160
Johan Hedberg9f616562011-04-28 11:28:54 -07003161 BT_DBG("Auto-accept of user confirmation with %ums delay",
3162 hdev->auto_accept_delay);
3163
3164 if (hdev->auto_accept_delay > 0) {
3165 int delay = msecs_to_jiffies(hdev->auto_accept_delay);
3166 mod_timer(&conn->auto_accept_timer, jiffies + delay);
3167 goto unlock;
3168 }
3169
Johan Hedberg7a828902011-04-28 11:28:53 -07003170 hci_send_cmd(hdev, HCI_OP_USER_CONFIRM_REPLY,
3171 sizeof(ev->bdaddr), &ev->bdaddr);
3172 goto unlock;
3173 }
3174
Johan Hedberg55bc1a32011-04-28 11:28:56 -07003175confirm:
Johan Hedberg272d90d2012-02-09 15:26:12 +02003176 mgmt_user_confirm_request(hdev, &ev->bdaddr, ACL_LINK, 0, ev->passkey,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003177 confirm_hint);
Johan Hedberg7a828902011-04-28 11:28:53 -07003178
3179unlock:
Johan Hedberga5c29682011-02-19 12:05:57 -03003180 hci_dev_unlock(hdev);
3181}
3182
Brian Gix1143d452011-11-23 08:28:34 -08003183static inline void hci_user_passkey_request_evt(struct hci_dev *hdev,
3184 struct sk_buff *skb)
3185{
3186 struct hci_ev_user_passkey_req *ev = (void *) skb->data;
3187
3188 BT_DBG("%s", hdev->name);
3189
3190 hci_dev_lock(hdev);
3191
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02003192 if (test_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg272d90d2012-02-09 15:26:12 +02003193 mgmt_user_passkey_request(hdev, &ev->bdaddr, ACL_LINK, 0);
Brian Gix1143d452011-11-23 08:28:34 -08003194
3195 hci_dev_unlock(hdev);
3196}
3197
Marcel Holtmann04936842008-07-14 20:13:48 +02003198static inline void hci_simple_pair_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
3199{
3200 struct hci_ev_simple_pair_complete *ev = (void *) skb->data;
3201 struct hci_conn *conn;
3202
3203 BT_DBG("%s", hdev->name);
3204
3205 hci_dev_lock(hdev);
3206
3207 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &ev->bdaddr);
Johan Hedberg2a611692011-02-19 12:06:00 -03003208 if (!conn)
3209 goto unlock;
Marcel Holtmann04936842008-07-14 20:13:48 +02003210
Johan Hedberg2a611692011-02-19 12:06:00 -03003211 /* To avoid duplicate auth_failed events to user space we check
3212 * the HCI_CONN_AUTH_PEND flag which will be set if we
3213 * initiated the authentication. A traditional auth_complete
3214 * event gets always produced as initiator and is also mapped to
3215 * the mgmt_auth_failed event */
Johan Hedberg51a8efd2012-01-16 06:10:31 +02003216 if (!test_bit(HCI_CONN_AUTH_PEND, &conn->flags) && ev->status != 0)
Johan Hedbergbab73cb2012-02-09 16:07:29 +02003217 mgmt_auth_failed(hdev, &conn->dst, conn->type, conn->dst_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003218 ev->status);
Johan Hedberg2a611692011-02-19 12:06:00 -03003219
3220 hci_conn_put(conn);
3221
3222unlock:
Marcel Holtmann04936842008-07-14 20:13:48 +02003223 hci_dev_unlock(hdev);
3224}
3225
Marcel Holtmann41a96212008-07-14 20:13:48 +02003226static inline void hci_remote_host_features_evt(struct hci_dev *hdev, struct sk_buff *skb)
3227{
3228 struct hci_ev_remote_host_features *ev = (void *) skb->data;
3229 struct inquiry_entry *ie;
3230
3231 BT_DBG("%s", hdev->name);
3232
3233 hci_dev_lock(hdev);
3234
Andrei Emeltchenkocc11b9c2010-11-22 13:21:37 +02003235 ie = hci_inquiry_cache_lookup(hdev, &ev->bdaddr);
3236 if (ie)
Johan Hedberg02b7cc62012-02-28 02:28:43 +02003237 ie->data.ssp_mode = (ev->features[0] & LMP_HOST_SSP);
Marcel Holtmann41a96212008-07-14 20:13:48 +02003238
3239 hci_dev_unlock(hdev);
3240}
3241
Szymon Janc2763eda2011-03-22 13:12:22 +01003242static inline void hci_remote_oob_data_request_evt(struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003243 struct sk_buff *skb)
Szymon Janc2763eda2011-03-22 13:12:22 +01003244{
3245 struct hci_ev_remote_oob_data_request *ev = (void *) skb->data;
3246 struct oob_data *data;
3247
3248 BT_DBG("%s", hdev->name);
3249
3250 hci_dev_lock(hdev);
3251
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02003252 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
Szymon Jance1ba1f12011-04-06 13:01:59 +02003253 goto unlock;
3254
Szymon Janc2763eda2011-03-22 13:12:22 +01003255 data = hci_find_remote_oob_data(hdev, &ev->bdaddr);
3256 if (data) {
3257 struct hci_cp_remote_oob_data_reply cp;
3258
3259 bacpy(&cp.bdaddr, &ev->bdaddr);
3260 memcpy(cp.hash, data->hash, sizeof(cp.hash));
3261 memcpy(cp.randomizer, data->randomizer, sizeof(cp.randomizer));
3262
3263 hci_send_cmd(hdev, HCI_OP_REMOTE_OOB_DATA_REPLY, sizeof(cp),
3264 &cp);
3265 } else {
3266 struct hci_cp_remote_oob_data_neg_reply cp;
3267
3268 bacpy(&cp.bdaddr, &ev->bdaddr);
3269 hci_send_cmd(hdev, HCI_OP_REMOTE_OOB_DATA_NEG_REPLY, sizeof(cp),
3270 &cp);
3271 }
3272
Szymon Jance1ba1f12011-04-06 13:01:59 +02003273unlock:
Szymon Janc2763eda2011-03-22 13:12:22 +01003274 hci_dev_unlock(hdev);
3275}
3276
Ville Tervofcd89c02011-02-10 22:38:47 -03003277static inline void hci_le_conn_complete_evt(struct hci_dev *hdev, struct sk_buff *skb)
3278{
3279 struct hci_ev_le_conn_complete *ev = (void *) skb->data;
3280 struct hci_conn *conn;
3281
3282 BT_DBG("%s status %d", hdev->name, ev->status);
3283
3284 hci_dev_lock(hdev);
3285
3286 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &ev->bdaddr);
Ville Tervob62f3282011-02-10 22:38:50 -03003287 if (!conn) {
3288 conn = hci_conn_add(hdev, LE_LINK, &ev->bdaddr);
3289 if (!conn) {
3290 BT_ERR("No memory for new connection");
3291 hci_dev_unlock(hdev);
3292 return;
3293 }
Andre Guedes29b79882011-05-31 14:20:54 -03003294
3295 conn->dst_type = ev->bdaddr_type;
Ville Tervob62f3282011-02-10 22:38:50 -03003296 }
Ville Tervofcd89c02011-02-10 22:38:47 -03003297
3298 if (ev->status) {
Johan Hedberg48264f02011-11-09 13:58:58 +02003299 mgmt_connect_failed(hdev, &ev->bdaddr, conn->type,
3300 conn->dst_type, ev->status);
Ville Tervofcd89c02011-02-10 22:38:47 -03003301 hci_proto_connect_cfm(conn, ev->status);
3302 conn->state = BT_CLOSED;
3303 hci_conn_del(conn);
3304 goto unlock;
3305 }
3306
Johan Hedbergb644ba32012-01-17 21:48:47 +02003307 if (!test_and_set_bit(HCI_CONN_MGMT_CONNECTED, &conn->flags))
3308 mgmt_device_connected(hdev, &ev->bdaddr, conn->type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003309 conn->dst_type, 0, NULL, 0, NULL);
Vinicius Costa Gomes83bc71b2011-05-06 18:41:43 -03003310
Vinicius Costa Gomes7b5c0d52011-06-09 18:50:50 -03003311 conn->sec_level = BT_SECURITY_LOW;
Ville Tervofcd89c02011-02-10 22:38:47 -03003312 conn->handle = __le16_to_cpu(ev->handle);
3313 conn->state = BT_CONNECTED;
3314
3315 hci_conn_hold_device(conn);
3316 hci_conn_add_sysfs(conn);
3317
3318 hci_proto_connect_cfm(conn, ev->status);
3319
3320unlock:
3321 hci_dev_unlock(hdev);
3322}
3323
Andre Guedes9aa04c92011-05-26 16:23:51 -03003324static inline void hci_le_adv_report_evt(struct hci_dev *hdev,
3325 struct sk_buff *skb)
3326{
Andre Guedese95beb42011-09-26 20:48:35 -03003327 u8 num_reports = skb->data[0];
3328 void *ptr = &skb->data[1];
Andre Guedes3c9e9192012-01-10 18:20:50 -03003329 s8 rssi;
Andre Guedes9aa04c92011-05-26 16:23:51 -03003330
3331 hci_dev_lock(hdev);
3332
Andre Guedese95beb42011-09-26 20:48:35 -03003333 while (num_reports--) {
3334 struct hci_ev_le_advertising_info *ev = ptr;
Andre Guedes9aa04c92011-05-26 16:23:51 -03003335
Andre Guedes9aa04c92011-05-26 16:23:51 -03003336 hci_add_adv_entry(hdev, ev);
Andre Guedese95beb42011-09-26 20:48:35 -03003337
Andre Guedes3c9e9192012-01-10 18:20:50 -03003338 rssi = ev->data[ev->length];
3339 mgmt_device_found(hdev, &ev->bdaddr, LE_LINK, ev->bdaddr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003340 NULL, rssi, 0, 1, ev->data, ev->length);
Andre Guedes3c9e9192012-01-10 18:20:50 -03003341
Andre Guedese95beb42011-09-26 20:48:35 -03003342 ptr += sizeof(*ev) + ev->length + 1;
Andre Guedes9aa04c92011-05-26 16:23:51 -03003343 }
3344
3345 hci_dev_unlock(hdev);
3346}
3347
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003348static inline void hci_le_ltk_request_evt(struct hci_dev *hdev,
3349 struct sk_buff *skb)
3350{
3351 struct hci_ev_le_ltk_req *ev = (void *) skb->data;
3352 struct hci_cp_le_ltk_reply cp;
Vinicius Costa Gomesbea710f2011-07-07 18:59:37 -03003353 struct hci_cp_le_ltk_neg_reply neg;
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003354 struct hci_conn *conn;
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -03003355 struct smp_ltk *ltk;
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003356
3357 BT_DBG("%s handle %d", hdev->name, cpu_to_le16(ev->handle));
3358
3359 hci_dev_lock(hdev);
3360
3361 conn = hci_conn_hash_lookup_handle(hdev, __le16_to_cpu(ev->handle));
Vinicius Costa Gomesbea710f2011-07-07 18:59:37 -03003362 if (conn == NULL)
3363 goto not_found;
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003364
Vinicius Costa Gomesbea710f2011-07-07 18:59:37 -03003365 ltk = hci_find_ltk(hdev, ev->ediv, ev->random);
3366 if (ltk == NULL)
3367 goto not_found;
3368
3369 memcpy(cp.ltk, ltk->val, sizeof(ltk->val));
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003370 cp.handle = cpu_to_le16(conn->handle);
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -03003371
3372 if (ltk->authenticated)
3373 conn->sec_level = BT_SECURITY_HIGH;
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003374
3375 hci_send_cmd(hdev, HCI_OP_LE_LTK_REPLY, sizeof(cp), &cp);
3376
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -03003377 if (ltk->type & HCI_SMP_STK) {
3378 list_del(&ltk->list);
3379 kfree(ltk);
3380 }
3381
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003382 hci_dev_unlock(hdev);
Vinicius Costa Gomesbea710f2011-07-07 18:59:37 -03003383
3384 return;
3385
3386not_found:
3387 neg.handle = ev->handle;
3388 hci_send_cmd(hdev, HCI_OP_LE_LTK_NEG_REPLY, sizeof(neg), &neg);
3389 hci_dev_unlock(hdev);
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003390}
3391
Ville Tervofcd89c02011-02-10 22:38:47 -03003392static inline void hci_le_meta_evt(struct hci_dev *hdev, struct sk_buff *skb)
3393{
3394 struct hci_ev_le_meta *le_ev = (void *) skb->data;
3395
3396 skb_pull(skb, sizeof(*le_ev));
3397
3398 switch (le_ev->subevent) {
3399 case HCI_EV_LE_CONN_COMPLETE:
3400 hci_le_conn_complete_evt(hdev, skb);
3401 break;
3402
Andre Guedes9aa04c92011-05-26 16:23:51 -03003403 case HCI_EV_LE_ADVERTISING_REPORT:
3404 hci_le_adv_report_evt(hdev, skb);
3405 break;
3406
Vinicius Costa Gomesa7a595f2011-06-09 18:50:47 -03003407 case HCI_EV_LE_LTK_REQ:
3408 hci_le_ltk_request_evt(hdev, skb);
3409 break;
3410
Ville Tervofcd89c02011-02-10 22:38:47 -03003411 default:
3412 break;
3413 }
3414}
3415
Linus Torvalds1da177e2005-04-16 15:20:36 -07003416void hci_event_packet(struct hci_dev *hdev, struct sk_buff *skb)
3417{
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003418 struct hci_event_hdr *hdr = (void *) skb->data;
3419 __u8 event = hdr->evt;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003420
3421 skb_pull(skb, HCI_EVENT_HDR_SIZE);
3422
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003423 switch (event) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003424 case HCI_EV_INQUIRY_COMPLETE:
3425 hci_inquiry_complete_evt(hdev, skb);
3426 break;
3427
3428 case HCI_EV_INQUIRY_RESULT:
3429 hci_inquiry_result_evt(hdev, skb);
3430 break;
3431
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003432 case HCI_EV_CONN_COMPLETE:
3433 hci_conn_complete_evt(hdev, skb);
Marcel Holtmann21d9e302005-09-13 01:32:25 +02003434 break;
3435
Linus Torvalds1da177e2005-04-16 15:20:36 -07003436 case HCI_EV_CONN_REQUEST:
3437 hci_conn_request_evt(hdev, skb);
3438 break;
3439
Linus Torvalds1da177e2005-04-16 15:20:36 -07003440 case HCI_EV_DISCONN_COMPLETE:
3441 hci_disconn_complete_evt(hdev, skb);
3442 break;
3443
Linus Torvalds1da177e2005-04-16 15:20:36 -07003444 case HCI_EV_AUTH_COMPLETE:
3445 hci_auth_complete_evt(hdev, skb);
3446 break;
3447
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003448 case HCI_EV_REMOTE_NAME:
3449 hci_remote_name_evt(hdev, skb);
3450 break;
3451
Linus Torvalds1da177e2005-04-16 15:20:36 -07003452 case HCI_EV_ENCRYPT_CHANGE:
3453 hci_encrypt_change_evt(hdev, skb);
3454 break;
3455
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003456 case HCI_EV_CHANGE_LINK_KEY_COMPLETE:
3457 hci_change_link_key_complete_evt(hdev, skb);
3458 break;
3459
3460 case HCI_EV_REMOTE_FEATURES:
3461 hci_remote_features_evt(hdev, skb);
3462 break;
3463
3464 case HCI_EV_REMOTE_VERSION:
3465 hci_remote_version_evt(hdev, skb);
3466 break;
3467
3468 case HCI_EV_QOS_SETUP_COMPLETE:
3469 hci_qos_setup_complete_evt(hdev, skb);
3470 break;
3471
3472 case HCI_EV_CMD_COMPLETE:
3473 hci_cmd_complete_evt(hdev, skb);
3474 break;
3475
3476 case HCI_EV_CMD_STATUS:
3477 hci_cmd_status_evt(hdev, skb);
3478 break;
3479
3480 case HCI_EV_ROLE_CHANGE:
3481 hci_role_change_evt(hdev, skb);
3482 break;
3483
3484 case HCI_EV_NUM_COMP_PKTS:
3485 hci_num_comp_pkts_evt(hdev, skb);
3486 break;
3487
3488 case HCI_EV_MODE_CHANGE:
3489 hci_mode_change_evt(hdev, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003490 break;
3491
3492 case HCI_EV_PIN_CODE_REQ:
3493 hci_pin_code_request_evt(hdev, skb);
3494 break;
3495
3496 case HCI_EV_LINK_KEY_REQ:
3497 hci_link_key_request_evt(hdev, skb);
3498 break;
3499
3500 case HCI_EV_LINK_KEY_NOTIFY:
3501 hci_link_key_notify_evt(hdev, skb);
3502 break;
3503
3504 case HCI_EV_CLOCK_OFFSET:
3505 hci_clock_offset_evt(hdev, skb);
3506 break;
3507
Marcel Holtmanna8746412008-07-14 20:13:46 +02003508 case HCI_EV_PKT_TYPE_CHANGE:
3509 hci_pkt_type_change_evt(hdev, skb);
3510 break;
3511
Marcel Holtmann85a1e932005-08-09 20:28:02 -07003512 case HCI_EV_PSCAN_REP_MODE:
3513 hci_pscan_rep_mode_evt(hdev, skb);
3514 break;
3515
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003516 case HCI_EV_INQUIRY_RESULT_WITH_RSSI:
3517 hci_inquiry_result_with_rssi_evt(hdev, skb);
3518 break;
3519
3520 case HCI_EV_REMOTE_EXT_FEATURES:
3521 hci_remote_ext_features_evt(hdev, skb);
3522 break;
3523
3524 case HCI_EV_SYNC_CONN_COMPLETE:
3525 hci_sync_conn_complete_evt(hdev, skb);
3526 break;
3527
3528 case HCI_EV_SYNC_CONN_CHANGED:
3529 hci_sync_conn_changed_evt(hdev, skb);
3530 break;
3531
Marcel Holtmann04837f62006-07-03 10:02:33 +02003532 case HCI_EV_SNIFF_SUBRATE:
3533 hci_sniff_subrate_evt(hdev, skb);
3534 break;
3535
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003536 case HCI_EV_EXTENDED_INQUIRY_RESULT:
3537 hci_extended_inquiry_result_evt(hdev, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003538 break;
3539
Marcel Holtmann04936842008-07-14 20:13:48 +02003540 case HCI_EV_IO_CAPA_REQUEST:
3541 hci_io_capa_request_evt(hdev, skb);
3542 break;
3543
Johan Hedberg03b555e2011-01-04 15:40:05 +02003544 case HCI_EV_IO_CAPA_REPLY:
3545 hci_io_capa_reply_evt(hdev, skb);
3546 break;
3547
Johan Hedberga5c29682011-02-19 12:05:57 -03003548 case HCI_EV_USER_CONFIRM_REQUEST:
3549 hci_user_confirm_request_evt(hdev, skb);
3550 break;
3551
Brian Gix1143d452011-11-23 08:28:34 -08003552 case HCI_EV_USER_PASSKEY_REQUEST:
3553 hci_user_passkey_request_evt(hdev, skb);
3554 break;
3555
Marcel Holtmann04936842008-07-14 20:13:48 +02003556 case HCI_EV_SIMPLE_PAIR_COMPLETE:
3557 hci_simple_pair_complete_evt(hdev, skb);
3558 break;
3559
Marcel Holtmann41a96212008-07-14 20:13:48 +02003560 case HCI_EV_REMOTE_HOST_FEATURES:
3561 hci_remote_host_features_evt(hdev, skb);
3562 break;
3563
Ville Tervofcd89c02011-02-10 22:38:47 -03003564 case HCI_EV_LE_META:
3565 hci_le_meta_evt(hdev, skb);
3566 break;
3567
Szymon Janc2763eda2011-03-22 13:12:22 +01003568 case HCI_EV_REMOTE_OOB_DATA_REQUEST:
3569 hci_remote_oob_data_request_evt(hdev, skb);
3570 break;
3571
Andrei Emeltchenko25e89e92012-01-04 12:41:58 +02003572 case HCI_EV_NUM_COMP_BLOCKS:
3573 hci_num_comp_blocks_evt(hdev, skb);
3574 break;
3575
Marcel Holtmanna9de9242007-10-20 13:33:56 +02003576 default:
3577 BT_DBG("%s event 0x%x", hdev->name, event);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003578 break;
3579 }
3580
3581 kfree_skb(skb);
3582 hdev->stat.evt_rx++;
3583}