blob: bad23d5fdd3535b88af384ec1ae0ea9ed8603418 [file] [log] [blame]
Johan Hedberg03811012010-12-08 00:21:06 +02001/*
2 BlueZ - Bluetooth protocol stack for Linux
Johan Hedbergea585ab2012-02-17 14:50:39 +02003
Johan Hedberg03811012010-12-08 00:21:06 +02004 Copyright (C) 2010 Nokia Corporation
Johan Hedbergea585ab2012-02-17 14:50:39 +02005 Copyright (C) 2011-2012 Intel Corporation
Johan Hedberg03811012010-12-08 00:21:06 +02006
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
15 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
18 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
19
20 ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,
21 COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS
22 SOFTWARE IS DISCLAIMED.
23*/
24
25/* Bluetooth HCI Management interface */
26
Paul Gortmaker3a9a2312011-05-27 09:12:25 -040027#include <linux/module.h>
Johan Hedberg03811012010-12-08 00:21:06 +020028#include <asm/unaligned.h>
29
30#include <net/bluetooth/bluetooth.h>
31#include <net/bluetooth/hci_core.h>
32#include <net/bluetooth/mgmt.h>
Marcel Holtmannac4b7232013-10-10 14:54:16 -070033
34#include "smp.h"
Johan Hedberg03811012010-12-08 00:21:06 +020035
Johan Hedberg2da9c552012-02-17 14:39:28 +020036#define MGMT_VERSION 1
Marcel Holtmann404566442014-01-28 15:39:01 -080037#define MGMT_REVISION 5
Johan Hedberg02d98122010-12-13 21:07:04 +020038
Johan Hedberge70bb2e2012-02-13 16:59:33 +020039static const u16 mgmt_commands[] = {
40 MGMT_OP_READ_INDEX_LIST,
41 MGMT_OP_READ_INFO,
42 MGMT_OP_SET_POWERED,
43 MGMT_OP_SET_DISCOVERABLE,
44 MGMT_OP_SET_CONNECTABLE,
45 MGMT_OP_SET_FAST_CONNECTABLE,
46 MGMT_OP_SET_PAIRABLE,
47 MGMT_OP_SET_LINK_SECURITY,
48 MGMT_OP_SET_SSP,
49 MGMT_OP_SET_HS,
50 MGMT_OP_SET_LE,
51 MGMT_OP_SET_DEV_CLASS,
52 MGMT_OP_SET_LOCAL_NAME,
53 MGMT_OP_ADD_UUID,
54 MGMT_OP_REMOVE_UUID,
55 MGMT_OP_LOAD_LINK_KEYS,
56 MGMT_OP_LOAD_LONG_TERM_KEYS,
57 MGMT_OP_DISCONNECT,
58 MGMT_OP_GET_CONNECTIONS,
59 MGMT_OP_PIN_CODE_REPLY,
60 MGMT_OP_PIN_CODE_NEG_REPLY,
61 MGMT_OP_SET_IO_CAPABILITY,
62 MGMT_OP_PAIR_DEVICE,
63 MGMT_OP_CANCEL_PAIR_DEVICE,
64 MGMT_OP_UNPAIR_DEVICE,
65 MGMT_OP_USER_CONFIRM_REPLY,
66 MGMT_OP_USER_CONFIRM_NEG_REPLY,
67 MGMT_OP_USER_PASSKEY_REPLY,
68 MGMT_OP_USER_PASSKEY_NEG_REPLY,
69 MGMT_OP_READ_LOCAL_OOB_DATA,
70 MGMT_OP_ADD_REMOTE_OOB_DATA,
71 MGMT_OP_REMOVE_REMOTE_OOB_DATA,
72 MGMT_OP_START_DISCOVERY,
73 MGMT_OP_STOP_DISCOVERY,
74 MGMT_OP_CONFIRM_NAME,
75 MGMT_OP_BLOCK_DEVICE,
76 MGMT_OP_UNBLOCK_DEVICE,
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -070077 MGMT_OP_SET_DEVICE_ID,
Johan Hedberg4375f102013-09-25 13:26:10 +030078 MGMT_OP_SET_ADVERTISING,
Johan Hedberg0663ca22013-10-02 13:43:14 +030079 MGMT_OP_SET_BREDR,
Marcel Holtmannd13eafc2013-10-02 04:41:30 -070080 MGMT_OP_SET_STATIC_ADDRESS,
Marcel Holtmann7f72134e2013-10-11 14:44:58 -070081 MGMT_OP_SET_SCAN_PARAMS,
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -080082 MGMT_OP_SET_SECURE_CONN,
Marcel Holtmann4e39ac82014-01-31 11:55:22 -080083 MGMT_OP_SET_DEBUG_KEYS,
Johan Hedberg62b04cd2014-02-23 19:42:27 +020084 MGMT_OP_SET_PRIVACY,
Johan Hedberg41edf162014-02-18 10:19:35 +020085 MGMT_OP_LOAD_IRKS,
Johan Hedberge70bb2e2012-02-13 16:59:33 +020086};
87
88static const u16 mgmt_events[] = {
89 MGMT_EV_CONTROLLER_ERROR,
90 MGMT_EV_INDEX_ADDED,
91 MGMT_EV_INDEX_REMOVED,
92 MGMT_EV_NEW_SETTINGS,
93 MGMT_EV_CLASS_OF_DEV_CHANGED,
94 MGMT_EV_LOCAL_NAME_CHANGED,
95 MGMT_EV_NEW_LINK_KEY,
96 MGMT_EV_NEW_LONG_TERM_KEY,
97 MGMT_EV_DEVICE_CONNECTED,
98 MGMT_EV_DEVICE_DISCONNECTED,
99 MGMT_EV_CONNECT_FAILED,
100 MGMT_EV_PIN_CODE_REQUEST,
101 MGMT_EV_USER_CONFIRM_REQUEST,
102 MGMT_EV_USER_PASSKEY_REQUEST,
103 MGMT_EV_AUTH_FAILED,
104 MGMT_EV_DEVICE_FOUND,
105 MGMT_EV_DISCOVERING,
106 MGMT_EV_DEVICE_BLOCKED,
107 MGMT_EV_DEVICE_UNBLOCKED,
108 MGMT_EV_DEVICE_UNPAIRED,
Johan Hedberg92a25252012-09-06 18:39:26 +0300109 MGMT_EV_PASSKEY_NOTIFY,
Marcel Holtmann1b60ef22014-02-21 21:35:30 -0800110 MGMT_EV_NEW_IRK,
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200111};
112
Marcel Holtmann17b02e62012-03-01 14:32:37 -0800113#define CACHE_TIMEOUT msecs_to_jiffies(2 * 1000)
Johan Hedberg7d785252011-12-15 00:47:39 +0200114
Johan Hedberg4b34ee782012-02-21 14:13:02 +0200115#define hdev_is_powered(hdev) (test_bit(HCI_UP, &hdev->flags) && \
116 !test_bit(HCI_AUTO_OFF, &hdev->dev_flags))
117
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200118struct pending_cmd {
119 struct list_head list;
Johan Hedbergfc2f4b12011-11-09 13:58:56 +0200120 u16 opcode;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200121 int index;
Szymon Jancc68fb7f2011-03-22 13:12:19 +0100122 void *param;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200123 struct sock *sk;
Johan Hedberge9a416b2011-02-19 12:05:56 -0300124 void *user_data;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200125};
126
Johan Hedbergca69b792011-11-11 18:10:00 +0200127/* HCI to MGMT error code conversion table */
128static u8 mgmt_status_table[] = {
129 MGMT_STATUS_SUCCESS,
130 MGMT_STATUS_UNKNOWN_COMMAND, /* Unknown Command */
131 MGMT_STATUS_NOT_CONNECTED, /* No Connection */
132 MGMT_STATUS_FAILED, /* Hardware Failure */
133 MGMT_STATUS_CONNECT_FAILED, /* Page Timeout */
134 MGMT_STATUS_AUTH_FAILED, /* Authentication Failed */
Johan Hedbergeadd6632014-01-13 17:15:53 +0200135 MGMT_STATUS_AUTH_FAILED, /* PIN or Key Missing */
Johan Hedbergca69b792011-11-11 18:10:00 +0200136 MGMT_STATUS_NO_RESOURCES, /* Memory Full */
137 MGMT_STATUS_TIMEOUT, /* Connection Timeout */
138 MGMT_STATUS_NO_RESOURCES, /* Max Number of Connections */
139 MGMT_STATUS_NO_RESOURCES, /* Max Number of SCO Connections */
140 MGMT_STATUS_ALREADY_CONNECTED, /* ACL Connection Exists */
141 MGMT_STATUS_BUSY, /* Command Disallowed */
142 MGMT_STATUS_NO_RESOURCES, /* Rejected Limited Resources */
143 MGMT_STATUS_REJECTED, /* Rejected Security */
144 MGMT_STATUS_REJECTED, /* Rejected Personal */
145 MGMT_STATUS_TIMEOUT, /* Host Timeout */
146 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported Feature */
147 MGMT_STATUS_INVALID_PARAMS, /* Invalid Parameters */
148 MGMT_STATUS_DISCONNECTED, /* OE User Ended Connection */
149 MGMT_STATUS_NO_RESOURCES, /* OE Low Resources */
150 MGMT_STATUS_DISCONNECTED, /* OE Power Off */
151 MGMT_STATUS_DISCONNECTED, /* Connection Terminated */
152 MGMT_STATUS_BUSY, /* Repeated Attempts */
153 MGMT_STATUS_REJECTED, /* Pairing Not Allowed */
154 MGMT_STATUS_FAILED, /* Unknown LMP PDU */
155 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported Remote Feature */
156 MGMT_STATUS_REJECTED, /* SCO Offset Rejected */
157 MGMT_STATUS_REJECTED, /* SCO Interval Rejected */
158 MGMT_STATUS_REJECTED, /* Air Mode Rejected */
159 MGMT_STATUS_INVALID_PARAMS, /* Invalid LMP Parameters */
160 MGMT_STATUS_FAILED, /* Unspecified Error */
161 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported LMP Parameter Value */
162 MGMT_STATUS_FAILED, /* Role Change Not Allowed */
163 MGMT_STATUS_TIMEOUT, /* LMP Response Timeout */
164 MGMT_STATUS_FAILED, /* LMP Error Transaction Collision */
165 MGMT_STATUS_FAILED, /* LMP PDU Not Allowed */
166 MGMT_STATUS_REJECTED, /* Encryption Mode Not Accepted */
167 MGMT_STATUS_FAILED, /* Unit Link Key Used */
168 MGMT_STATUS_NOT_SUPPORTED, /* QoS Not Supported */
169 MGMT_STATUS_TIMEOUT, /* Instant Passed */
170 MGMT_STATUS_NOT_SUPPORTED, /* Pairing Not Supported */
171 MGMT_STATUS_FAILED, /* Transaction Collision */
172 MGMT_STATUS_INVALID_PARAMS, /* Unacceptable Parameter */
173 MGMT_STATUS_REJECTED, /* QoS Rejected */
174 MGMT_STATUS_NOT_SUPPORTED, /* Classification Not Supported */
175 MGMT_STATUS_REJECTED, /* Insufficient Security */
176 MGMT_STATUS_INVALID_PARAMS, /* Parameter Out Of Range */
177 MGMT_STATUS_BUSY, /* Role Switch Pending */
178 MGMT_STATUS_FAILED, /* Slot Violation */
179 MGMT_STATUS_FAILED, /* Role Switch Failed */
180 MGMT_STATUS_INVALID_PARAMS, /* EIR Too Large */
181 MGMT_STATUS_NOT_SUPPORTED, /* Simple Pairing Not Supported */
182 MGMT_STATUS_BUSY, /* Host Busy Pairing */
183 MGMT_STATUS_REJECTED, /* Rejected, No Suitable Channel */
184 MGMT_STATUS_BUSY, /* Controller Busy */
185 MGMT_STATUS_INVALID_PARAMS, /* Unsuitable Connection Interval */
186 MGMT_STATUS_TIMEOUT, /* Directed Advertising Timeout */
187 MGMT_STATUS_AUTH_FAILED, /* Terminated Due to MIC Failure */
188 MGMT_STATUS_CONNECT_FAILED, /* Connection Establishment Failed */
189 MGMT_STATUS_CONNECT_FAILED, /* MAC Connection Failed */
190};
191
192static u8 mgmt_status(u8 hci_status)
193{
194 if (hci_status < ARRAY_SIZE(mgmt_status_table))
195 return mgmt_status_table[hci_status];
196
197 return MGMT_STATUS_FAILED;
198}
199
Szymon Janc4e51eae2011-02-25 19:05:48 +0100200static int cmd_status(struct sock *sk, u16 index, u16 cmd, u8 status)
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200201{
202 struct sk_buff *skb;
203 struct mgmt_hdr *hdr;
204 struct mgmt_ev_cmd_status *ev;
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300205 int err;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200206
Szymon Janc34eb5252011-02-28 14:10:08 +0100207 BT_DBG("sock %p, index %u, cmd %u, status %u", sk, index, cmd, status);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200208
Andre Guedes790eff42012-06-07 19:05:46 -0300209 skb = alloc_skb(sizeof(*hdr) + sizeof(*ev), GFP_KERNEL);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200210 if (!skb)
211 return -ENOMEM;
212
213 hdr = (void *) skb_put(skb, sizeof(*hdr));
214
Syam Sidhardhan612dfce2012-10-29 22:37:36 +0530215 hdr->opcode = __constant_cpu_to_le16(MGMT_EV_CMD_STATUS);
Szymon Janc4e51eae2011-02-25 19:05:48 +0100216 hdr->index = cpu_to_le16(index);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200217 hdr->len = cpu_to_le16(sizeof(*ev));
218
219 ev = (void *) skb_put(skb, sizeof(*ev));
220 ev->status = status;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200221 ev->opcode = cpu_to_le16(cmd);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200222
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300223 err = sock_queue_rcv_skb(sk, skb);
224 if (err < 0)
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200225 kfree_skb(skb);
226
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300227 return err;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200228}
229
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200230static int cmd_complete(struct sock *sk, u16 index, u16 cmd, u8 status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300231 void *rp, size_t rp_len)
Johan Hedberg02d98122010-12-13 21:07:04 +0200232{
233 struct sk_buff *skb;
234 struct mgmt_hdr *hdr;
235 struct mgmt_ev_cmd_complete *ev;
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300236 int err;
Johan Hedberg02d98122010-12-13 21:07:04 +0200237
238 BT_DBG("sock %p", sk);
239
Andre Guedes790eff42012-06-07 19:05:46 -0300240 skb = alloc_skb(sizeof(*hdr) + sizeof(*ev) + rp_len, GFP_KERNEL);
Johan Hedberg02d98122010-12-13 21:07:04 +0200241 if (!skb)
242 return -ENOMEM;
243
244 hdr = (void *) skb_put(skb, sizeof(*hdr));
Johan Hedberga38528f2011-01-22 06:46:43 +0200245
Syam Sidhardhan612dfce2012-10-29 22:37:36 +0530246 hdr->opcode = __constant_cpu_to_le16(MGMT_EV_CMD_COMPLETE);
Szymon Janc4e51eae2011-02-25 19:05:48 +0100247 hdr->index = cpu_to_le16(index);
Johan Hedberga38528f2011-01-22 06:46:43 +0200248 hdr->len = cpu_to_le16(sizeof(*ev) + rp_len);
Johan Hedberg02d98122010-12-13 21:07:04 +0200249
Johan Hedberga38528f2011-01-22 06:46:43 +0200250 ev = (void *) skb_put(skb, sizeof(*ev) + rp_len);
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200251 ev->opcode = cpu_to_le16(cmd);
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200252 ev->status = status;
Szymon Janc8020c162011-02-28 14:09:50 +0100253
254 if (rp)
255 memcpy(ev->data, rp, rp_len);
Johan Hedberg02d98122010-12-13 21:07:04 +0200256
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300257 err = sock_queue_rcv_skb(sk, skb);
258 if (err < 0)
Johan Hedberg02d98122010-12-13 21:07:04 +0200259 kfree_skb(skb);
260
Marcel Holtmanne5f0e152012-02-22 11:59:01 +0100261 return err;
Johan Hedberg02d98122010-12-13 21:07:04 +0200262}
263
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300264static int read_version(struct sock *sk, struct hci_dev *hdev, void *data,
265 u16 data_len)
Johan Hedberga38528f2011-01-22 06:46:43 +0200266{
267 struct mgmt_rp_read_version rp;
268
269 BT_DBG("sock %p", sk);
270
271 rp.version = MGMT_VERSION;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200272 rp.revision = __constant_cpu_to_le16(MGMT_REVISION);
Johan Hedberga38528f2011-01-22 06:46:43 +0200273
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200274 return cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_VERSION, 0, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300275 sizeof(rp));
Johan Hedberga38528f2011-01-22 06:46:43 +0200276}
277
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300278static int read_commands(struct sock *sk, struct hci_dev *hdev, void *data,
279 u16 data_len)
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200280{
281 struct mgmt_rp_read_commands *rp;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200282 const u16 num_commands = ARRAY_SIZE(mgmt_commands);
283 const u16 num_events = ARRAY_SIZE(mgmt_events);
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +0200284 __le16 *opcode;
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200285 size_t rp_size;
286 int i, err;
287
288 BT_DBG("sock %p", sk);
289
290 rp_size = sizeof(*rp) + ((num_commands + num_events) * sizeof(u16));
291
292 rp = kmalloc(rp_size, GFP_KERNEL);
293 if (!rp)
294 return -ENOMEM;
295
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200296 rp->num_commands = __constant_cpu_to_le16(num_commands);
297 rp->num_events = __constant_cpu_to_le16(num_events);
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200298
299 for (i = 0, opcode = rp->opcodes; i < num_commands; i++, opcode++)
300 put_unaligned_le16(mgmt_commands[i], opcode);
301
302 for (i = 0; i < num_events; i++, opcode++)
303 put_unaligned_le16(mgmt_events[i], opcode);
304
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200305 err = cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_COMMANDS, 0, rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300306 rp_size);
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200307 kfree(rp);
308
309 return err;
310}
311
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300312static int read_index_list(struct sock *sk, struct hci_dev *hdev, void *data,
313 u16 data_len)
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200314{
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200315 struct mgmt_rp_read_index_list *rp;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +0200316 struct hci_dev *d;
Johan Hedberga38528f2011-01-22 06:46:43 +0200317 size_t rp_len;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200318 u16 count;
Johan Hedberg476e44c2012-10-19 20:10:46 +0300319 int err;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200320
321 BT_DBG("sock %p", sk);
322
323 read_lock(&hci_dev_list_lock);
324
325 count = 0;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +0300326 list_for_each_entry(d, &hci_dev_list, list) {
Marcel Holtmann1514b892013-10-06 08:25:01 -0700327 if (d->dev_type == HCI_BREDR)
328 count++;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200329 }
330
Johan Hedberga38528f2011-01-22 06:46:43 +0200331 rp_len = sizeof(*rp) + (2 * count);
332 rp = kmalloc(rp_len, GFP_ATOMIC);
333 if (!rp) {
Jesper Juhlb2c60d42011-01-14 00:18:49 +0100334 read_unlock(&hci_dev_list_lock);
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200335 return -ENOMEM;
Jesper Juhlb2c60d42011-01-14 00:18:49 +0100336 }
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200337
Johan Hedberg476e44c2012-10-19 20:10:46 +0300338 count = 0;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +0200339 list_for_each_entry(d, &hci_dev_list, list) {
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200340 if (test_bit(HCI_SETUP, &d->dev_flags))
Johan Hedbergab81cbf2010-12-15 13:53:18 +0200341 continue;
342
Marcel Holtmann0736cfa2013-08-26 21:40:51 -0700343 if (test_bit(HCI_USER_CHANNEL, &d->dev_flags))
344 continue;
345
Marcel Holtmann1514b892013-10-06 08:25:01 -0700346 if (d->dev_type == HCI_BREDR) {
347 rp->index[count++] = cpu_to_le16(d->id);
348 BT_DBG("Added hci%u", d->id);
349 }
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200350 }
351
Johan Hedberg476e44c2012-10-19 20:10:46 +0300352 rp->num_controllers = cpu_to_le16(count);
353 rp_len = sizeof(*rp) + (2 * count);
354
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200355 read_unlock(&hci_dev_list_lock);
356
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200357 err = cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_INDEX_LIST, 0, rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300358 rp_len);
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200359
Johan Hedberga38528f2011-01-22 06:46:43 +0200360 kfree(rp);
361
362 return err;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200363}
364
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200365static u32 get_supported_settings(struct hci_dev *hdev)
Johan Hedberg03811012010-12-08 00:21:06 +0200366{
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200367 u32 settings = 0;
Johan Hedberg03811012010-12-08 00:21:06 +0200368
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200369 settings |= MGMT_SETTING_POWERED;
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200370 settings |= MGMT_SETTING_PAIRABLE;
Marcel Holtmannb1de97d2014-01-31 11:55:21 -0800371 settings |= MGMT_SETTING_DEBUG_KEYS;
Johan Hedberg03811012010-12-08 00:21:06 +0200372
Andre Guedesed3fa312012-07-24 15:03:46 -0300373 if (lmp_bredr_capable(hdev)) {
Johan Hedberg33c525c2012-10-24 21:11:58 +0300374 settings |= MGMT_SETTING_CONNECTABLE;
Johan Hedberg1a47aee2013-03-15 17:07:06 -0500375 if (hdev->hci_ver >= BLUETOOTH_VER_1_2)
376 settings |= MGMT_SETTING_FAST_CONNECTABLE;
Johan Hedberg33c525c2012-10-24 21:11:58 +0300377 settings |= MGMT_SETTING_DISCOVERABLE;
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200378 settings |= MGMT_SETTING_BREDR;
379 settings |= MGMT_SETTING_LINK_SECURITY;
Marcel Holtmanna82974c2013-10-11 09:48:47 -0700380
381 if (lmp_ssp_capable(hdev)) {
382 settings |= MGMT_SETTING_SSP;
383 settings |= MGMT_SETTING_HS;
384 }
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800385
Marcel Holtmann5afeac142014-01-10 02:07:27 -0800386 if (lmp_sc_capable(hdev) ||
387 test_bit(HCI_FORCE_SC, &hdev->dev_flags))
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800388 settings |= MGMT_SETTING_SECURE_CONN;
Marcel Holtmann848566b2013-10-01 22:59:22 -0700389 }
Marcel Holtmannd7b7e792012-02-20 21:47:49 +0100390
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300391 if (lmp_le_capable(hdev)) {
Marcel Holtmann9d428202012-05-03 07:12:31 +0200392 settings |= MGMT_SETTING_LE;
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300393 settings |= MGMT_SETTING_ADVERTISING;
Johan Hedberg0f4bd942014-02-22 19:06:35 +0200394 settings |= MGMT_SETTING_PRIVACY;
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300395 }
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200396
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200397 return settings;
398}
Johan Hedbergebc99fe2011-01-04 11:54:26 +0200399
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200400static u32 get_current_settings(struct hci_dev *hdev)
401{
402 u32 settings = 0;
Johan Hedbergdc4fe302011-03-16 14:29:36 +0200403
Johan Hedbergf1f0eb02012-02-21 17:15:41 +0200404 if (hdev_is_powered(hdev))
Marcel Holtmannf0d4b782012-02-21 12:14:25 +0100405 settings |= MGMT_SETTING_POWERED;
406
Johan Hedberg5e5282b2012-02-21 16:01:30 +0200407 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200408 settings |= MGMT_SETTING_CONNECTABLE;
409
Johan Hedberg1a4d3c42013-03-15 17:07:08 -0500410 if (test_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags))
411 settings |= MGMT_SETTING_FAST_CONNECTABLE;
412
Johan Hedberg5e5282b2012-02-21 16:01:30 +0200413 if (test_bit(HCI_DISCOVERABLE, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200414 settings |= MGMT_SETTING_DISCOVERABLE;
415
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200416 if (test_bit(HCI_PAIRABLE, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200417 settings |= MGMT_SETTING_PAIRABLE;
418
Johan Hedberg56f87902013-10-02 13:43:13 +0300419 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200420 settings |= MGMT_SETTING_BREDR;
421
Johan Hedberg06199cf2012-02-22 16:37:11 +0200422 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200423 settings |= MGMT_SETTING_LE;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200424
Johan Hedberg47990ea2012-02-22 11:58:37 +0200425 if (test_bit(HCI_LINK_SECURITY, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200426 settings |= MGMT_SETTING_LINK_SECURITY;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200427
Johan Hedberg84bde9d2012-01-25 14:21:06 +0200428 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200429 settings |= MGMT_SETTING_SSP;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200430
Johan Hedberg6d80dfd2012-02-20 23:50:38 +0200431 if (test_bit(HCI_HS_ENABLED, &hdev->dev_flags))
432 settings |= MGMT_SETTING_HS;
433
Johan Hedbergf3d3444a2013-10-05 12:01:04 +0200434 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300435 settings |= MGMT_SETTING_ADVERTISING;
436
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800437 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags))
438 settings |= MGMT_SETTING_SECURE_CONN;
439
Marcel Holtmannb1de97d2014-01-31 11:55:21 -0800440 if (test_bit(HCI_DEBUG_KEYS, &hdev->dev_flags))
441 settings |= MGMT_SETTING_DEBUG_KEYS;
442
Johan Hedberg0f4bd942014-02-22 19:06:35 +0200443 if (test_bit(HCI_PRIVACY, &hdev->dev_flags))
444 settings |= MGMT_SETTING_PRIVACY;
445
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200446 return settings;
Johan Hedbergc542a062011-01-26 13:11:03 +0200447}
448
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300449#define PNP_INFO_SVCLASS_ID 0x1200
450
Johan Hedberg213202e2013-01-27 00:31:33 +0200451static u8 *create_uuid16_list(struct hci_dev *hdev, u8 *data, ptrdiff_t len)
452{
453 u8 *ptr = data, *uuids_start = NULL;
454 struct bt_uuid *uuid;
455
456 if (len < 4)
457 return ptr;
458
459 list_for_each_entry(uuid, &hdev->uuids, list) {
460 u16 uuid16;
461
462 if (uuid->size != 16)
463 continue;
464
465 uuid16 = get_unaligned_le16(&uuid->uuid[12]);
466 if (uuid16 < 0x1100)
467 continue;
468
469 if (uuid16 == PNP_INFO_SVCLASS_ID)
470 continue;
471
472 if (!uuids_start) {
473 uuids_start = ptr;
474 uuids_start[0] = 1;
475 uuids_start[1] = EIR_UUID16_ALL;
476 ptr += 2;
477 }
478
479 /* Stop if not enough space to put next UUID */
480 if ((ptr - data) + sizeof(u16) > len) {
481 uuids_start[1] = EIR_UUID16_SOME;
482 break;
483 }
484
485 *ptr++ = (uuid16 & 0x00ff);
486 *ptr++ = (uuid16 & 0xff00) >> 8;
487 uuids_start[0] += sizeof(uuid16);
488 }
489
490 return ptr;
491}
492
Johan Hedbergcdf19632013-01-27 00:31:34 +0200493static u8 *create_uuid32_list(struct hci_dev *hdev, u8 *data, ptrdiff_t len)
494{
495 u8 *ptr = data, *uuids_start = NULL;
496 struct bt_uuid *uuid;
497
498 if (len < 6)
499 return ptr;
500
501 list_for_each_entry(uuid, &hdev->uuids, list) {
502 if (uuid->size != 32)
503 continue;
504
505 if (!uuids_start) {
506 uuids_start = ptr;
507 uuids_start[0] = 1;
508 uuids_start[1] = EIR_UUID32_ALL;
509 ptr += 2;
510 }
511
512 /* Stop if not enough space to put next UUID */
513 if ((ptr - data) + sizeof(u32) > len) {
514 uuids_start[1] = EIR_UUID32_SOME;
515 break;
516 }
517
518 memcpy(ptr, &uuid->uuid[12], sizeof(u32));
519 ptr += sizeof(u32);
520 uuids_start[0] += sizeof(u32);
521 }
522
523 return ptr;
524}
525
Johan Hedbergc00d5752013-01-27 00:31:35 +0200526static u8 *create_uuid128_list(struct hci_dev *hdev, u8 *data, ptrdiff_t len)
527{
528 u8 *ptr = data, *uuids_start = NULL;
529 struct bt_uuid *uuid;
530
531 if (len < 18)
532 return ptr;
533
534 list_for_each_entry(uuid, &hdev->uuids, list) {
535 if (uuid->size != 128)
536 continue;
537
538 if (!uuids_start) {
539 uuids_start = ptr;
540 uuids_start[0] = 1;
541 uuids_start[1] = EIR_UUID128_ALL;
542 ptr += 2;
543 }
544
545 /* Stop if not enough space to put next UUID */
546 if ((ptr - data) + 16 > len) {
547 uuids_start[1] = EIR_UUID128_SOME;
548 break;
549 }
550
551 memcpy(ptr, uuid->uuid, 16);
552 ptr += 16;
553 uuids_start[0] += 16;
554 }
555
556 return ptr;
557}
558
Johan Hedbergeb2a8d22013-10-19 23:38:20 +0300559static struct pending_cmd *mgmt_pending_find(u16 opcode, struct hci_dev *hdev)
560{
561 struct pending_cmd *cmd;
562
563 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
564 if (cmd->opcode == opcode)
565 return cmd;
566 }
567
568 return NULL;
569}
570
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700571static u8 create_scan_rsp_data(struct hci_dev *hdev, u8 *ptr)
572{
Marcel Holtmann7a5f4992013-10-16 00:16:49 -0700573 u8 ad_len = 0;
574 size_t name_len;
575
576 name_len = strlen(hdev->dev_name);
577 if (name_len > 0) {
578 size_t max_len = HCI_MAX_AD_LENGTH - ad_len - 2;
579
580 if (name_len > max_len) {
581 name_len = max_len;
582 ptr[1] = EIR_NAME_SHORT;
583 } else
584 ptr[1] = EIR_NAME_COMPLETE;
585
586 ptr[0] = name_len + 1;
587
588 memcpy(ptr + 2, hdev->dev_name, name_len);
589
590 ad_len += (name_len + 2);
591 ptr += (name_len + 2);
592 }
593
594 return ad_len;
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700595}
596
597static void update_scan_rsp_data(struct hci_request *req)
598{
599 struct hci_dev *hdev = req->hdev;
600 struct hci_cp_le_set_scan_rsp_data cp;
601 u8 len;
602
Johan Hedberg7751ef12013-10-19 23:38:15 +0300603 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700604 return;
605
606 memset(&cp, 0, sizeof(cp));
607
608 len = create_scan_rsp_data(hdev, cp.data);
609
Johan Hedbergeb438b52013-10-16 15:31:07 +0300610 if (hdev->scan_rsp_data_len == len &&
611 memcmp(cp.data, hdev->scan_rsp_data, len) == 0)
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700612 return;
613
Johan Hedbergeb438b52013-10-16 15:31:07 +0300614 memcpy(hdev->scan_rsp_data, cp.data, sizeof(cp.data));
615 hdev->scan_rsp_data_len = len;
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700616
617 cp.length = len;
618
619 hci_req_add(req, HCI_OP_LE_SET_SCAN_RSP_DATA, sizeof(cp), &cp);
620}
621
Johan Hedberg9a43e252013-10-20 19:00:07 +0300622static u8 get_adv_discov_flags(struct hci_dev *hdev)
623{
624 struct pending_cmd *cmd;
625
626 /* If there's a pending mgmt command the flags will not yet have
627 * their final values, so check for this first.
628 */
629 cmd = mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev);
630 if (cmd) {
631 struct mgmt_mode *cp = cmd->param;
632 if (cp->val == 0x01)
633 return LE_AD_GENERAL;
634 else if (cp->val == 0x02)
635 return LE_AD_LIMITED;
636 } else {
637 if (test_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags))
638 return LE_AD_LIMITED;
639 else if (test_bit(HCI_DISCOVERABLE, &hdev->dev_flags))
640 return LE_AD_GENERAL;
641 }
642
643 return 0;
644}
645
Marcel Holtmann46cad2e2013-10-16 00:16:46 -0700646static u8 create_adv_data(struct hci_dev *hdev, u8 *ptr)
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700647{
648 u8 ad_len = 0, flags = 0;
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700649
Johan Hedberg9a43e252013-10-20 19:00:07 +0300650 flags |= get_adv_discov_flags(hdev);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700651
Johan Hedberge8340042014-01-30 11:16:50 -0800652 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700653 flags |= LE_AD_NO_BREDR;
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700654
655 if (flags) {
656 BT_DBG("adv flags 0x%02x", flags);
657
658 ptr[0] = 2;
659 ptr[1] = EIR_FLAGS;
660 ptr[2] = flags;
661
662 ad_len += 3;
663 ptr += 3;
664 }
665
666 if (hdev->adv_tx_power != HCI_TX_POWER_INVALID) {
667 ptr[0] = 2;
668 ptr[1] = EIR_TX_POWER;
669 ptr[2] = (u8) hdev->adv_tx_power;
670
671 ad_len += 3;
672 ptr += 3;
673 }
674
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700675 return ad_len;
676}
677
Marcel Holtmann5947f4b2013-10-16 00:16:50 -0700678static void update_adv_data(struct hci_request *req)
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700679{
680 struct hci_dev *hdev = req->hdev;
681 struct hci_cp_le_set_adv_data cp;
682 u8 len;
683
Johan Hedberg10994ce2013-10-19 23:38:16 +0300684 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700685 return;
686
687 memset(&cp, 0, sizeof(cp));
688
Marcel Holtmann46cad2e2013-10-16 00:16:46 -0700689 len = create_adv_data(hdev, cp.data);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700690
691 if (hdev->adv_data_len == len &&
692 memcmp(cp.data, hdev->adv_data, len) == 0)
693 return;
694
695 memcpy(hdev->adv_data, cp.data, sizeof(cp.data));
696 hdev->adv_data_len = len;
697
698 cp.length = len;
699
700 hci_req_add(req, HCI_OP_LE_SET_ADV_DATA, sizeof(cp), &cp);
701}
702
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300703static void create_eir(struct hci_dev *hdev, u8 *data)
704{
705 u8 *ptr = data;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300706 size_t name_len;
707
708 name_len = strlen(hdev->dev_name);
709
710 if (name_len > 0) {
711 /* EIR Data type */
712 if (name_len > 48) {
713 name_len = 48;
714 ptr[1] = EIR_NAME_SHORT;
715 } else
716 ptr[1] = EIR_NAME_COMPLETE;
717
718 /* EIR Data length */
719 ptr[0] = name_len + 1;
720
721 memcpy(ptr + 2, hdev->dev_name, name_len);
722
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300723 ptr += (name_len + 2);
724 }
725
Johan Hedbergbbaf4442012-11-08 01:22:59 +0100726 if (hdev->inq_tx_power != HCI_TX_POWER_INVALID) {
Marcel Holtmann91c4e9b2012-03-11 19:27:21 -0700727 ptr[0] = 2;
728 ptr[1] = EIR_TX_POWER;
729 ptr[2] = (u8) hdev->inq_tx_power;
730
Marcel Holtmann91c4e9b2012-03-11 19:27:21 -0700731 ptr += 3;
732 }
733
Marcel Holtmann2b9be132012-03-11 19:32:12 -0700734 if (hdev->devid_source > 0) {
735 ptr[0] = 9;
736 ptr[1] = EIR_DEVICE_ID;
737
738 put_unaligned_le16(hdev->devid_source, ptr + 2);
739 put_unaligned_le16(hdev->devid_vendor, ptr + 4);
740 put_unaligned_le16(hdev->devid_product, ptr + 6);
741 put_unaligned_le16(hdev->devid_version, ptr + 8);
742
Marcel Holtmann2b9be132012-03-11 19:32:12 -0700743 ptr += 10;
744 }
745
Johan Hedberg213202e2013-01-27 00:31:33 +0200746 ptr = create_uuid16_list(hdev, ptr, HCI_MAX_EIR_LENGTH - (ptr - data));
Johan Hedbergcdf19632013-01-27 00:31:34 +0200747 ptr = create_uuid32_list(hdev, ptr, HCI_MAX_EIR_LENGTH - (ptr - data));
Johan Hedbergc00d5752013-01-27 00:31:35 +0200748 ptr = create_uuid128_list(hdev, ptr, HCI_MAX_EIR_LENGTH - (ptr - data));
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300749}
750
Johan Hedberg890ea892013-03-15 17:06:52 -0500751static void update_eir(struct hci_request *req)
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300752{
Johan Hedberg890ea892013-03-15 17:06:52 -0500753 struct hci_dev *hdev = req->hdev;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300754 struct hci_cp_write_eir cp;
755
Johan Hedberg504c8dc2012-02-23 13:30:41 +0200756 if (!hdev_is_powered(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -0500757 return;
Johan Hedberg7770c4a2012-02-22 22:06:38 +0200758
Johan Hedberg976eb202012-10-24 21:12:01 +0300759 if (!lmp_ext_inq_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -0500760 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300761
Johan Hedberg84bde9d2012-01-25 14:21:06 +0200762 if (!test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -0500763 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300764
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200765 if (test_bit(HCI_SERVICE_CACHE, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -0500766 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300767
768 memset(&cp, 0, sizeof(cp));
769
770 create_eir(hdev, cp.data);
771
772 if (memcmp(cp.data, hdev->eir, sizeof(cp.data)) == 0)
Johan Hedberg890ea892013-03-15 17:06:52 -0500773 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300774
775 memcpy(hdev->eir, cp.data, sizeof(cp.data));
776
Johan Hedberg890ea892013-03-15 17:06:52 -0500777 hci_req_add(req, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300778}
779
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200780static u8 get_service_classes(struct hci_dev *hdev)
781{
Gustavo F. Padovan12dc0742011-10-14 19:32:56 -0300782 struct bt_uuid *uuid;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200783 u8 val = 0;
784
Gustavo F. Padovan12dc0742011-10-14 19:32:56 -0300785 list_for_each_entry(uuid, &hdev->uuids, list)
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200786 val |= uuid->svc_hint;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200787
788 return val;
789}
790
Johan Hedberg890ea892013-03-15 17:06:52 -0500791static void update_class(struct hci_request *req)
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200792{
Johan Hedberg890ea892013-03-15 17:06:52 -0500793 struct hci_dev *hdev = req->hdev;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200794 u8 cod[3];
795
796 BT_DBG("%s", hdev->name);
797
Johan Hedberg504c8dc2012-02-23 13:30:41 +0200798 if (!hdev_is_powered(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -0500799 return;
Johan Hedberg7770c4a2012-02-22 22:06:38 +0200800
Johan Hedbergf87ea1d2013-10-19 23:38:17 +0300801 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
802 return;
803
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200804 if (test_bit(HCI_SERVICE_CACHE, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -0500805 return;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200806
807 cod[0] = hdev->minor_class;
808 cod[1] = hdev->major_class;
809 cod[2] = get_service_classes(hdev);
810
Marcel Holtmann6acd7db2013-10-15 06:33:53 -0700811 if (test_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags))
812 cod[1] |= 0x20;
813
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200814 if (memcmp(cod, hdev->dev_class, 3) == 0)
Johan Hedberg890ea892013-03-15 17:06:52 -0500815 return;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200816
Johan Hedberg890ea892013-03-15 17:06:52 -0500817 hci_req_add(req, HCI_OP_WRITE_CLASS_OF_DEV, sizeof(cod), cod);
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200818}
819
Johan Hedberga4858cb2014-02-25 19:56:31 +0200820static bool get_connectable(struct hci_dev *hdev)
Johan Hedberg199a2fb2014-02-22 19:06:33 +0200821{
822 struct pending_cmd *cmd;
Johan Hedberg199a2fb2014-02-22 19:06:33 +0200823
824 /* If there's a pending mgmt command the flag will not yet have
825 * it's final value, so check for this first.
826 */
827 cmd = mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev);
828 if (cmd) {
829 struct mgmt_mode *cp = cmd->param;
Johan Hedberga4858cb2014-02-25 19:56:31 +0200830 return cp->val;
Johan Hedberg199a2fb2014-02-22 19:06:33 +0200831 }
832
Johan Hedberga4858cb2014-02-25 19:56:31 +0200833 return test_bit(HCI_CONNECTABLE, &hdev->dev_flags);
Johan Hedberg199a2fb2014-02-22 19:06:33 +0200834}
835
836static void enable_advertising(struct hci_request *req)
837{
838 struct hci_dev *hdev = req->hdev;
839 struct hci_cp_le_set_adv_param cp;
Johan Hedberg8f2a0602014-02-23 19:42:23 +0200840 u8 own_addr_type, enable = 0x01;
Johan Hedberga4858cb2014-02-25 19:56:31 +0200841 bool connectable;
Johan Hedberg199a2fb2014-02-22 19:06:33 +0200842
Johan Hedberga4858cb2014-02-25 19:56:31 +0200843 connectable = get_connectable(hdev);
Johan Hedberg8f2a0602014-02-23 19:42:23 +0200844
Johan Hedberga4858cb2014-02-25 19:56:31 +0200845 /* Set require_privacy to true only when non-connectable
846 * advertising is used. In that case it is fine to use a
847 * non-resolvable private address.
848 */
849 if (hci_update_random_address(req, !connectable, &own_addr_type) < 0)
Johan Hedberg8f2a0602014-02-23 19:42:23 +0200850 return;
851
Marcel Holtmann41c90c12014-02-23 20:25:55 -0800852 memset(&cp, 0, sizeof(cp));
Johan Hedberg199a2fb2014-02-22 19:06:33 +0200853 cp.min_interval = __constant_cpu_to_le16(0x0800);
854 cp.max_interval = __constant_cpu_to_le16(0x0800);
Johan Hedberga4858cb2014-02-25 19:56:31 +0200855 cp.type = connectable ? LE_ADV_IND : LE_ADV_NONCONN_IND;
Johan Hedberg8f2a0602014-02-23 19:42:23 +0200856 cp.own_address_type = own_addr_type;
Johan Hedberg199a2fb2014-02-22 19:06:33 +0200857 cp.channel_map = hdev->le_adv_channel_map;
858
859 hci_req_add(req, HCI_OP_LE_SET_ADV_PARAM, sizeof(cp), &cp);
860
861 hci_req_add(req, HCI_OP_LE_SET_ADV_ENABLE, sizeof(enable), &enable);
862}
863
864static void disable_advertising(struct hci_request *req)
865{
866 u8 enable = 0x00;
867
868 hci_req_add(req, HCI_OP_LE_SET_ADV_ENABLE, sizeof(enable), &enable);
869}
870
Johan Hedberg7d785252011-12-15 00:47:39 +0200871static void service_cache_off(struct work_struct *work)
872{
873 struct hci_dev *hdev = container_of(work, struct hci_dev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300874 service_cache.work);
Johan Hedberg890ea892013-03-15 17:06:52 -0500875 struct hci_request req;
Johan Hedberg7d785252011-12-15 00:47:39 +0200876
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200877 if (!test_and_clear_bit(HCI_SERVICE_CACHE, &hdev->dev_flags))
Johan Hedberg7d785252011-12-15 00:47:39 +0200878 return;
879
Johan Hedberg890ea892013-03-15 17:06:52 -0500880 hci_req_init(&req, hdev);
881
Johan Hedberg7d785252011-12-15 00:47:39 +0200882 hci_dev_lock(hdev);
883
Johan Hedberg890ea892013-03-15 17:06:52 -0500884 update_eir(&req);
885 update_class(&req);
Johan Hedberg7d785252011-12-15 00:47:39 +0200886
887 hci_dev_unlock(hdev);
Johan Hedberg890ea892013-03-15 17:06:52 -0500888
889 hci_req_run(&req, NULL);
Johan Hedberg7d785252011-12-15 00:47:39 +0200890}
891
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200892static void rpa_expired(struct work_struct *work)
893{
894 struct hci_dev *hdev = container_of(work, struct hci_dev,
895 rpa_expired.work);
896 struct hci_request req;
897
898 BT_DBG("");
899
900 set_bit(HCI_RPA_EXPIRED, &hdev->dev_flags);
901
902 if (!test_bit(HCI_ADVERTISING, &hdev->dev_flags) ||
903 hci_conn_num(hdev, LE_LINK) > 0)
904 return;
905
906 /* The generation of a new RPA and programming it into the
907 * controller happens in the enable_advertising() function.
908 */
909
910 hci_req_init(&req, hdev);
911
912 disable_advertising(&req);
913 enable_advertising(&req);
914
915 hci_req_run(&req, NULL);
916}
917
Johan Hedberg6a919082012-02-28 06:17:26 +0200918static void mgmt_init_hdev(struct sock *sk, struct hci_dev *hdev)
Johan Hedberg7d785252011-12-15 00:47:39 +0200919{
Johan Hedberg4f87da82012-03-02 19:55:56 +0200920 if (test_and_set_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg6a919082012-02-28 06:17:26 +0200921 return;
922
Johan Hedberg4f87da82012-03-02 19:55:56 +0200923 INIT_DELAYED_WORK(&hdev->service_cache, service_cache_off);
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200924 INIT_DELAYED_WORK(&hdev->rpa_expired, rpa_expired);
Johan Hedberg7d785252011-12-15 00:47:39 +0200925
Johan Hedberg4f87da82012-03-02 19:55:56 +0200926 /* Non-mgmt controlled devices get this bit set
927 * implicitly so that pairing works for them, however
928 * for mgmt we require user-space to explicitly enable
929 * it
930 */
931 clear_bit(HCI_PAIRABLE, &hdev->dev_flags);
Johan Hedberg7d785252011-12-15 00:47:39 +0200932}
933
Johan Hedberg0f4e68c2012-02-28 17:18:30 +0200934static int read_controller_info(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300935 void *data, u16 data_len)
Johan Hedberg03811012010-12-08 00:21:06 +0200936{
937 struct mgmt_rp_read_info rp;
Johan Hedberg03811012010-12-08 00:21:06 +0200938
Johan Hedbergbdb6d972012-02-28 06:13:32 +0200939 BT_DBG("sock %p %s", sk, hdev->name);
Johan Hedberg03811012010-12-08 00:21:06 +0200940
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300941 hci_dev_lock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200942
Johan Hedberg03811012010-12-08 00:21:06 +0200943 memset(&rp, 0, sizeof(rp));
944
Johan Hedberg03811012010-12-08 00:21:06 +0200945 bacpy(&rp.bdaddr, &hdev->bdaddr);
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200946
947 rp.version = hdev->hci_ver;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200948 rp.manufacturer = cpu_to_le16(hdev->manufacturer);
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200949
950 rp.supported_settings = cpu_to_le32(get_supported_settings(hdev));
951 rp.current_settings = cpu_to_le32(get_current_settings(hdev));
952
953 memcpy(rp.dev_class, hdev->dev_class, 3);
Johan Hedberg03811012010-12-08 00:21:06 +0200954
955 memcpy(rp.name, hdev->dev_name, sizeof(hdev->dev_name));
Johan Hedberg27fcc362012-02-22 21:46:22 +0200956 memcpy(rp.short_name, hdev->short_name, sizeof(hdev->short_name));
Johan Hedberg03811012010-12-08 00:21:06 +0200957
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300958 hci_dev_unlock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200959
Johan Hedbergbdb6d972012-02-28 06:13:32 +0200960 return cmd_complete(sk, hdev->id, MGMT_OP_READ_INFO, 0, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300961 sizeof(rp));
Johan Hedberg03811012010-12-08 00:21:06 +0200962}
963
964static void mgmt_pending_free(struct pending_cmd *cmd)
965{
966 sock_put(cmd->sk);
967 kfree(cmd->param);
968 kfree(cmd);
969}
970
971static struct pending_cmd *mgmt_pending_add(struct sock *sk, u16 opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300972 struct hci_dev *hdev, void *data,
973 u16 len)
Johan Hedberg03811012010-12-08 00:21:06 +0200974{
975 struct pending_cmd *cmd;
976
Andre Guedes12b94562012-06-07 19:05:45 -0300977 cmd = kmalloc(sizeof(*cmd), GFP_KERNEL);
Johan Hedberg03811012010-12-08 00:21:06 +0200978 if (!cmd)
979 return NULL;
980
981 cmd->opcode = opcode;
982 cmd->index = hdev->id;
983
Andre Guedes12b94562012-06-07 19:05:45 -0300984 cmd->param = kmalloc(len, GFP_KERNEL);
Johan Hedberg03811012010-12-08 00:21:06 +0200985 if (!cmd->param) {
986 kfree(cmd);
987 return NULL;
988 }
989
990 if (data)
991 memcpy(cmd->param, data, len);
992
993 cmd->sk = sk;
994 sock_hold(sk);
995
996 list_add(&cmd->list, &hdev->mgmt_pending);
997
998 return cmd;
999}
1000
1001static void mgmt_pending_foreach(u16 opcode, struct hci_dev *hdev,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03001002 void (*cb)(struct pending_cmd *cmd,
1003 void *data),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001004 void *data)
Johan Hedberg03811012010-12-08 00:21:06 +02001005{
Andre Guedesa3d09352013-02-01 11:21:30 -03001006 struct pending_cmd *cmd, *tmp;
Johan Hedberg03811012010-12-08 00:21:06 +02001007
Andre Guedesa3d09352013-02-01 11:21:30 -03001008 list_for_each_entry_safe(cmd, tmp, &hdev->mgmt_pending, list) {
Johan Hedberg03811012010-12-08 00:21:06 +02001009 if (opcode > 0 && cmd->opcode != opcode)
1010 continue;
1011
1012 cb(cmd, data);
1013 }
1014}
1015
Johan Hedberg03811012010-12-08 00:21:06 +02001016static void mgmt_pending_remove(struct pending_cmd *cmd)
1017{
1018 list_del(&cmd->list);
1019 mgmt_pending_free(cmd);
1020}
1021
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001022static int send_settings_rsp(struct sock *sk, u16 opcode, struct hci_dev *hdev)
Johan Hedberg86805702011-11-11 16:18:52 +02001023{
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001024 __le32 settings = cpu_to_le32(get_current_settings(hdev));
Johan Hedberg86805702011-11-11 16:18:52 +02001025
Johan Hedbergaee9b2182012-02-18 15:07:59 +02001026 return cmd_complete(sk, hdev->id, opcode, 0, &settings,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001027 sizeof(settings));
Johan Hedberg86805702011-11-11 16:18:52 +02001028}
1029
Johan Hedberg8b064a32014-02-24 14:52:22 +02001030static void clean_up_hci_complete(struct hci_dev *hdev, u8 status)
1031{
1032 BT_DBG("%s status 0x%02x", hdev->name, status);
1033
1034 if (hci_conn_count(hdev) == 0)
1035 queue_work(hdev->req_workqueue, &hdev->power_off.work);
1036}
1037
1038static int clean_up_hci_state(struct hci_dev *hdev)
1039{
1040 struct hci_request req;
1041 struct hci_conn *conn;
1042
1043 hci_req_init(&req, hdev);
1044
1045 if (test_bit(HCI_ISCAN, &hdev->flags) ||
1046 test_bit(HCI_PSCAN, &hdev->flags)) {
1047 u8 scan = 0x00;
1048 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
1049 }
1050
1051 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
1052 disable_advertising(&req);
1053
1054 if (test_bit(HCI_LE_SCAN, &hdev->dev_flags)) {
Andre Guedesb1efcc22014-02-26 20:21:40 -03001055 hci_req_add_le_scan_disable(&req);
Johan Hedberg8b064a32014-02-24 14:52:22 +02001056 }
1057
1058 list_for_each_entry(conn, &hdev->conn_hash.list, list) {
1059 struct hci_cp_disconnect dc;
1060
1061 dc.handle = cpu_to_le16(conn->handle);
1062 dc.reason = 0x15; /* Terminated due to Power Off */
1063 hci_req_add(&req, HCI_OP_DISCONNECT, sizeof(dc), &dc);
1064 }
1065
1066 return hci_req_run(&req, clean_up_hci_complete);
1067}
1068
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001069static int set_powered(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001070 u16 len)
Johan Hedberg03811012010-12-08 00:21:06 +02001071{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001072 struct mgmt_mode *cp = data;
Johan Hedberg03811012010-12-08 00:21:06 +02001073 struct pending_cmd *cmd;
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001074 int err;
Johan Hedberg03811012010-12-08 00:21:06 +02001075
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001076 BT_DBG("request for %s", hdev->name);
Johan Hedberg03811012010-12-08 00:21:06 +02001077
Johan Hedberga7e80f22013-01-09 16:05:19 +02001078 if (cp->val != 0x00 && cp->val != 0x01)
1079 return cmd_status(sk, hdev->id, MGMT_OP_SET_POWERED,
1080 MGMT_STATUS_INVALID_PARAMS);
1081
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001082 hci_dev_lock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +02001083
Johan Hedberg87b95ba2013-09-25 13:26:06 +03001084 if (mgmt_pending_find(MGMT_OP_SET_POWERED, hdev)) {
1085 err = cmd_status(sk, hdev->id, MGMT_OP_SET_POWERED,
1086 MGMT_STATUS_BUSY);
1087 goto failed;
1088 }
1089
Marcel Holtmannf0d4b782012-02-21 12:14:25 +01001090 if (test_and_clear_bit(HCI_AUTO_OFF, &hdev->dev_flags)) {
1091 cancel_delayed_work(&hdev->power_off);
1092
1093 if (cp->val) {
Johan Hedberga1d70452013-01-09 15:29:40 +02001094 mgmt_pending_add(sk, MGMT_OP_SET_POWERED, hdev,
1095 data, len);
1096 err = mgmt_powered(hdev, 1);
Marcel Holtmannf0d4b782012-02-21 12:14:25 +01001097 goto failed;
1098 }
1099 }
1100
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001101 if (!!cp->val == hdev_is_powered(hdev)) {
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001102 err = send_settings_rsp(sk, MGMT_OP_SET_POWERED, hdev);
Johan Hedberg03811012010-12-08 00:21:06 +02001103 goto failed;
1104 }
1105
Johan Hedberg03811012010-12-08 00:21:06 +02001106 cmd = mgmt_pending_add(sk, MGMT_OP_SET_POWERED, hdev, data, len);
1107 if (!cmd) {
1108 err = -ENOMEM;
1109 goto failed;
1110 }
1111
Johan Hedberg8b064a32014-02-24 14:52:22 +02001112 if (cp->val) {
Johan Hedberg19202572013-01-14 22:33:51 +02001113 queue_work(hdev->req_workqueue, &hdev->power_on);
Johan Hedberg8b064a32014-02-24 14:52:22 +02001114 err = 0;
1115 } else {
1116 /* Disconnect connections, stop scans, etc */
1117 err = clean_up_hci_state(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +02001118
Johan Hedberg8b064a32014-02-24 14:52:22 +02001119 /* ENODATA means there were no HCI commands queued */
1120 if (err == -ENODATA) {
1121 queue_work(hdev->req_workqueue, &hdev->power_off.work);
1122 err = 0;
1123 }
1124 }
Johan Hedberg03811012010-12-08 00:21:06 +02001125
1126failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001127 hci_dev_unlock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +02001128 return err;
1129}
1130
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001131static int mgmt_event(u16 event, struct hci_dev *hdev, void *data, u16 data_len,
1132 struct sock *skip_sk)
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001133{
1134 struct sk_buff *skb;
1135 struct mgmt_hdr *hdr;
1136
Andre Guedes790eff42012-06-07 19:05:46 -03001137 skb = alloc_skb(sizeof(*hdr) + data_len, GFP_KERNEL);
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001138 if (!skb)
1139 return -ENOMEM;
1140
1141 hdr = (void *) skb_put(skb, sizeof(*hdr));
1142 hdr->opcode = cpu_to_le16(event);
1143 if (hdev)
1144 hdr->index = cpu_to_le16(hdev->id);
1145 else
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05301146 hdr->index = __constant_cpu_to_le16(MGMT_INDEX_NONE);
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001147 hdr->len = cpu_to_le16(data_len);
1148
1149 if (data)
1150 memcpy(skb_put(skb, data_len), data, data_len);
1151
Marcel Holtmann97e0bde2012-02-22 13:49:28 +01001152 /* Time stamp */
1153 __net_timestamp(skb);
1154
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001155 hci_send_to_control(skb, skip_sk);
1156 kfree_skb(skb);
1157
1158 return 0;
1159}
1160
1161static int new_settings(struct hci_dev *hdev, struct sock *skip)
1162{
1163 __le32 ev;
1164
1165 ev = cpu_to_le32(get_current_settings(hdev));
1166
1167 return mgmt_event(MGMT_EV_NEW_SETTINGS, hdev, &ev, sizeof(ev), skip);
1168}
1169
Johan Hedbergbd99abd2013-09-25 13:26:07 +03001170struct cmd_lookup {
1171 struct sock *sk;
1172 struct hci_dev *hdev;
1173 u8 mgmt_status;
1174};
1175
1176static void settings_rsp(struct pending_cmd *cmd, void *data)
1177{
1178 struct cmd_lookup *match = data;
1179
1180 send_settings_rsp(cmd->sk, cmd->opcode, match->hdev);
1181
1182 list_del(&cmd->list);
1183
1184 if (match->sk == NULL) {
1185 match->sk = cmd->sk;
1186 sock_hold(match->sk);
1187 }
1188
1189 mgmt_pending_free(cmd);
1190}
1191
1192static void cmd_status_rsp(struct pending_cmd *cmd, void *data)
1193{
1194 u8 *status = data;
1195
1196 cmd_status(cmd->sk, cmd->index, cmd->opcode, *status);
1197 mgmt_pending_remove(cmd);
1198}
1199
Johan Hedberge6fe7982013-10-02 15:45:22 +03001200static u8 mgmt_bredr_support(struct hci_dev *hdev)
1201{
1202 if (!lmp_bredr_capable(hdev))
1203 return MGMT_STATUS_NOT_SUPPORTED;
1204 else if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
1205 return MGMT_STATUS_REJECTED;
1206 else
1207 return MGMT_STATUS_SUCCESS;
1208}
1209
1210static u8 mgmt_le_support(struct hci_dev *hdev)
1211{
1212 if (!lmp_le_capable(hdev))
1213 return MGMT_STATUS_NOT_SUPPORTED;
1214 else if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
1215 return MGMT_STATUS_REJECTED;
1216 else
1217 return MGMT_STATUS_SUCCESS;
1218}
1219
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001220static void set_discoverable_complete(struct hci_dev *hdev, u8 status)
1221{
1222 struct pending_cmd *cmd;
1223 struct mgmt_mode *cp;
Marcel Holtmann970ba522013-10-15 06:33:57 -07001224 struct hci_request req;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001225 bool changed;
1226
1227 BT_DBG("status 0x%02x", status);
1228
1229 hci_dev_lock(hdev);
1230
1231 cmd = mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev);
1232 if (!cmd)
1233 goto unlock;
1234
1235 if (status) {
1236 u8 mgmt_err = mgmt_status(status);
1237 cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001238 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001239 goto remove_cmd;
1240 }
1241
1242 cp = cmd->param;
Marcel Holtmannd4462a02013-10-15 08:11:02 -07001243 if (cp->val) {
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001244 changed = !test_and_set_bit(HCI_DISCOVERABLE,
1245 &hdev->dev_flags);
Marcel Holtmannd4462a02013-10-15 08:11:02 -07001246
1247 if (hdev->discov_timeout > 0) {
1248 int to = msecs_to_jiffies(hdev->discov_timeout * 1000);
1249 queue_delayed_work(hdev->workqueue, &hdev->discov_off,
1250 to);
1251 }
1252 } else {
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001253 changed = test_and_clear_bit(HCI_DISCOVERABLE,
1254 &hdev->dev_flags);
Marcel Holtmannd4462a02013-10-15 08:11:02 -07001255 }
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001256
1257 send_settings_rsp(cmd->sk, MGMT_OP_SET_DISCOVERABLE, hdev);
1258
1259 if (changed)
1260 new_settings(hdev, cmd->sk);
1261
Marcel Holtmann970ba522013-10-15 06:33:57 -07001262 /* When the discoverable mode gets changed, make sure
1263 * that class of device has the limited discoverable
1264 * bit correctly set.
1265 */
1266 hci_req_init(&req, hdev);
1267 update_class(&req);
1268 hci_req_run(&req, NULL);
1269
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001270remove_cmd:
1271 mgmt_pending_remove(cmd);
1272
1273unlock:
1274 hci_dev_unlock(hdev);
1275}
1276
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001277static int set_discoverable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001278 u16 len)
Johan Hedberg03811012010-12-08 00:21:06 +02001279{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001280 struct mgmt_cp_set_discoverable *cp = data;
Johan Hedberg03811012010-12-08 00:21:06 +02001281 struct pending_cmd *cmd;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001282 struct hci_request req;
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001283 u16 timeout;
Johan Hedberg9a43e252013-10-20 19:00:07 +03001284 u8 scan;
Johan Hedberg03811012010-12-08 00:21:06 +02001285 int err;
1286
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001287 BT_DBG("request for %s", hdev->name);
Johan Hedberg03811012010-12-08 00:21:06 +02001288
Johan Hedberg9a43e252013-10-20 19:00:07 +03001289 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags) &&
1290 !test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedberg33c525c2012-10-24 21:11:58 +03001291 return cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Johan Hedberg9a43e252013-10-20 19:00:07 +03001292 MGMT_STATUS_REJECTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001293
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001294 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga7e80f22013-01-09 16:05:19 +02001295 return cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1296 MGMT_STATUS_INVALID_PARAMS);
1297
Marcel Holtmann1f350c82012-03-12 20:31:08 -07001298 timeout = __le16_to_cpu(cp->timeout);
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001299
1300 /* Disabling discoverable requires that no timeout is set,
1301 * and enabling limited discoverable requires a timeout.
1302 */
1303 if ((cp->val == 0x00 && timeout > 0) ||
1304 (cp->val == 0x02 && timeout == 0))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001305 return cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001306 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001307
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001308 hci_dev_lock(hdev);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001309
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001310 if (!hdev_is_powered(hdev) && timeout > 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001311 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001312 MGMT_STATUS_NOT_POWERED);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001313 goto failed;
1314 }
1315
1316 if (mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03001317 mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001318 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001319 MGMT_STATUS_BUSY);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001320 goto failed;
1321 }
1322
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001323 if (!test_bit(HCI_CONNECTABLE, &hdev->dev_flags)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001324 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001325 MGMT_STATUS_REJECTED);
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001326 goto failed;
1327 }
1328
1329 if (!hdev_is_powered(hdev)) {
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001330 bool changed = false;
1331
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001332 /* Setting limited discoverable when powered off is
1333 * not a valid operation since it requires a timeout
1334 * and so no need to check HCI_LIMITED_DISCOVERABLE.
1335 */
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001336 if (!!cp->val != test_bit(HCI_DISCOVERABLE, &hdev->dev_flags)) {
1337 change_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
1338 changed = true;
1339 }
1340
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001341 err = send_settings_rsp(sk, MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001342 if (err < 0)
1343 goto failed;
1344
1345 if (changed)
1346 err = new_settings(hdev, sk);
1347
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001348 goto failed;
1349 }
1350
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001351 /* If the current mode is the same, then just update the timeout
1352 * value with the new value. And if only the timeout gets updated,
1353 * then no need for any HCI transactions.
1354 */
1355 if (!!cp->val == test_bit(HCI_DISCOVERABLE, &hdev->dev_flags) &&
1356 (cp->val == 0x02) == test_bit(HCI_LIMITED_DISCOVERABLE,
1357 &hdev->dev_flags)) {
Marcel Holtmann36261542013-10-15 08:28:51 -07001358 cancel_delayed_work(&hdev->discov_off);
1359 hdev->discov_timeout = timeout;
Marcel Holtmann955638e2012-02-22 18:21:00 +01001360
Marcel Holtmann36261542013-10-15 08:28:51 -07001361 if (cp->val && hdev->discov_timeout > 0) {
1362 int to = msecs_to_jiffies(hdev->discov_timeout * 1000);
Marcel Holtmann955638e2012-02-22 18:21:00 +01001363 queue_delayed_work(hdev->workqueue, &hdev->discov_off,
Marcel Holtmann36261542013-10-15 08:28:51 -07001364 to);
Marcel Holtmann955638e2012-02-22 18:21:00 +01001365 }
1366
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001367 err = send_settings_rsp(sk, MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001368 goto failed;
1369 }
1370
1371 cmd = mgmt_pending_add(sk, MGMT_OP_SET_DISCOVERABLE, hdev, data, len);
1372 if (!cmd) {
1373 err = -ENOMEM;
1374 goto failed;
1375 }
1376
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001377 /* Cancel any potential discoverable timeout that might be
1378 * still active and store new timeout value. The arming of
1379 * the timeout happens in the complete handler.
1380 */
1381 cancel_delayed_work(&hdev->discov_off);
1382 hdev->discov_timeout = timeout;
1383
Johan Hedbergb456f872013-10-19 23:38:22 +03001384 /* Limited discoverable mode */
1385 if (cp->val == 0x02)
1386 set_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1387 else
1388 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1389
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001390 hci_req_init(&req, hdev);
1391
Johan Hedberg9a43e252013-10-20 19:00:07 +03001392 /* The procedure for LE-only controllers is much simpler - just
1393 * update the advertising data.
1394 */
1395 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
1396 goto update_ad;
1397
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001398 scan = SCAN_PAGE;
1399
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001400 if (cp->val) {
1401 struct hci_cp_write_current_iac_lap hci_cp;
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001402
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001403 if (cp->val == 0x02) {
1404 /* Limited discoverable mode */
Marcel Holtmann33337dc2013-10-23 08:28:01 -07001405 hci_cp.num_iac = min_t(u8, hdev->num_iac, 2);
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001406 hci_cp.iac_lap[0] = 0x00; /* LIAC */
1407 hci_cp.iac_lap[1] = 0x8b;
1408 hci_cp.iac_lap[2] = 0x9e;
1409 hci_cp.iac_lap[3] = 0x33; /* GIAC */
1410 hci_cp.iac_lap[4] = 0x8b;
1411 hci_cp.iac_lap[5] = 0x9e;
1412 } else {
1413 /* General discoverable mode */
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001414 hci_cp.num_iac = 1;
1415 hci_cp.iac_lap[0] = 0x33; /* GIAC */
1416 hci_cp.iac_lap[1] = 0x8b;
1417 hci_cp.iac_lap[2] = 0x9e;
1418 }
1419
1420 hci_req_add(&req, HCI_OP_WRITE_CURRENT_IAC_LAP,
1421 (hci_cp.num_iac * 3) + 1, &hci_cp);
1422
1423 scan |= SCAN_INQUIRY;
1424 } else {
1425 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1426 }
1427
1428 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE, sizeof(scan), &scan);
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001429
Johan Hedberg9a43e252013-10-20 19:00:07 +03001430update_ad:
1431 update_adv_data(&req);
1432
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001433 err = hci_req_run(&req, set_discoverable_complete);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001434 if (err < 0)
1435 mgmt_pending_remove(cmd);
1436
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001437failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001438 hci_dev_unlock(hdev);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001439 return err;
1440}
1441
Johan Hedberg406d7802013-03-15 17:07:09 -05001442static void write_fast_connectable(struct hci_request *req, bool enable)
1443{
Johan Hedbergbd98b992013-03-15 17:07:13 -05001444 struct hci_dev *hdev = req->hdev;
Johan Hedberg406d7802013-03-15 17:07:09 -05001445 struct hci_cp_write_page_scan_activity acp;
1446 u8 type;
1447
Johan Hedberg547003b2013-10-21 16:51:53 +03001448 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
1449 return;
1450
Johan Hedberg4c01f8b2013-03-15 17:07:14 -05001451 if (hdev->hci_ver < BLUETOOTH_VER_1_2)
1452 return;
1453
Johan Hedberg406d7802013-03-15 17:07:09 -05001454 if (enable) {
1455 type = PAGE_SCAN_TYPE_INTERLACED;
1456
1457 /* 160 msec page scan interval */
1458 acp.interval = __constant_cpu_to_le16(0x0100);
1459 } else {
1460 type = PAGE_SCAN_TYPE_STANDARD; /* default */
1461
1462 /* default 1.28 sec page scan */
1463 acp.interval = __constant_cpu_to_le16(0x0800);
1464 }
1465
1466 acp.window = __constant_cpu_to_le16(0x0012);
1467
Johan Hedbergbd98b992013-03-15 17:07:13 -05001468 if (__cpu_to_le16(hdev->page_scan_interval) != acp.interval ||
1469 __cpu_to_le16(hdev->page_scan_window) != acp.window)
1470 hci_req_add(req, HCI_OP_WRITE_PAGE_SCAN_ACTIVITY,
1471 sizeof(acp), &acp);
1472
1473 if (hdev->page_scan_type != type)
1474 hci_req_add(req, HCI_OP_WRITE_PAGE_SCAN_TYPE, 1, &type);
Johan Hedberg406d7802013-03-15 17:07:09 -05001475}
1476
Johan Hedberg2b76f452013-03-15 17:07:04 -05001477static void set_connectable_complete(struct hci_dev *hdev, u8 status)
1478{
1479 struct pending_cmd *cmd;
Johan Hedbergd7b856f2013-10-14 16:20:04 +03001480 struct mgmt_mode *cp;
1481 bool changed;
Johan Hedberg2b76f452013-03-15 17:07:04 -05001482
1483 BT_DBG("status 0x%02x", status);
1484
1485 hci_dev_lock(hdev);
1486
1487 cmd = mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev);
1488 if (!cmd)
1489 goto unlock;
1490
Johan Hedberg37438c12013-10-14 16:20:05 +03001491 if (status) {
1492 u8 mgmt_err = mgmt_status(status);
1493 cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
1494 goto remove_cmd;
1495 }
1496
Johan Hedbergd7b856f2013-10-14 16:20:04 +03001497 cp = cmd->param;
1498 if (cp->val)
1499 changed = !test_and_set_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1500 else
1501 changed = test_and_clear_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1502
Johan Hedberg2b76f452013-03-15 17:07:04 -05001503 send_settings_rsp(cmd->sk, MGMT_OP_SET_CONNECTABLE, hdev);
1504
Johan Hedbergd7b856f2013-10-14 16:20:04 +03001505 if (changed)
1506 new_settings(hdev, cmd->sk);
1507
Johan Hedberg37438c12013-10-14 16:20:05 +03001508remove_cmd:
Johan Hedberg2b76f452013-03-15 17:07:04 -05001509 mgmt_pending_remove(cmd);
1510
1511unlock:
1512 hci_dev_unlock(hdev);
1513}
1514
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001515static int set_connectable_update_settings(struct hci_dev *hdev,
1516 struct sock *sk, u8 val)
1517{
1518 bool changed = false;
1519 int err;
1520
1521 if (!!val != test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
1522 changed = true;
1523
1524 if (val) {
1525 set_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1526 } else {
1527 clear_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1528 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
1529 }
1530
1531 err = send_settings_rsp(sk, MGMT_OP_SET_CONNECTABLE, hdev);
1532 if (err < 0)
1533 return err;
1534
1535 if (changed)
1536 return new_settings(hdev, sk);
1537
1538 return 0;
1539}
1540
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001541static int set_connectable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001542 u16 len)
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001543{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001544 struct mgmt_mode *cp = data;
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001545 struct pending_cmd *cmd;
Johan Hedberg2b76f452013-03-15 17:07:04 -05001546 struct hci_request req;
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001547 u8 scan;
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001548 int err;
Johan Hedberg03811012010-12-08 00:21:06 +02001549
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001550 BT_DBG("request for %s", hdev->name);
Johan Hedberge41d8b42010-12-13 21:07:03 +02001551
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001552 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags) &&
1553 !test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedberg33c525c2012-10-24 21:11:58 +03001554 return cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001555 MGMT_STATUS_REJECTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001556
Johan Hedberga7e80f22013-01-09 16:05:19 +02001557 if (cp->val != 0x00 && cp->val != 0x01)
1558 return cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
1559 MGMT_STATUS_INVALID_PARAMS);
1560
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001561 hci_dev_lock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001562
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001563 if (!hdev_is_powered(hdev)) {
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001564 err = set_connectable_update_settings(hdev, sk, cp->val);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001565 goto failed;
1566 }
1567
1568 if (mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03001569 mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001570 err = cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001571 MGMT_STATUS_BUSY);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001572 goto failed;
1573 }
1574
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001575 cmd = mgmt_pending_add(sk, MGMT_OP_SET_CONNECTABLE, hdev, data, len);
1576 if (!cmd) {
1577 err = -ENOMEM;
1578 goto failed;
1579 }
1580
Johan Hedberg2b76f452013-03-15 17:07:04 -05001581 hci_req_init(&req, hdev);
1582
Johan Hedberg9a43e252013-10-20 19:00:07 +03001583 /* If BR/EDR is not enabled and we disable advertising as a
1584 * by-product of disabling connectable, we need to update the
1585 * advertising flags.
1586 */
1587 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
1588 if (!cp->val) {
1589 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1590 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
1591 }
1592 update_adv_data(&req);
1593 } else if (cp->val != test_bit(HCI_PSCAN, &hdev->flags)) {
Johan Hedberg9b742462013-10-14 16:20:03 +03001594 if (cp->val) {
1595 scan = SCAN_PAGE;
1596 } else {
1597 scan = 0;
1598
1599 if (test_bit(HCI_ISCAN, &hdev->flags) &&
Marcel Holtmann8d6083f2013-10-14 16:38:45 -07001600 hdev->discov_timeout > 0)
Johan Hedberg9b742462013-10-14 16:20:03 +03001601 cancel_delayed_work(&hdev->discov_off);
1602 }
1603
1604 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
1605 }
Johan Hedberg2b76f452013-03-15 17:07:04 -05001606
Johan Hedberg4c01f8b2013-03-15 17:07:14 -05001607 /* If we're going from non-connectable to connectable or
1608 * vice-versa when fast connectable is enabled ensure that fast
1609 * connectable gets disabled. write_fast_connectable won't do
1610 * anything if the page scan parameters are already what they
1611 * should be.
1612 */
1613 if (cp->val || test_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags))
Johan Hedberge36a3762013-03-15 17:07:10 -05001614 write_fast_connectable(&req, false);
1615
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001616 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags) &&
1617 hci_conn_num(hdev, LE_LINK) == 0) {
1618 disable_advertising(&req);
1619 enable_advertising(&req);
1620 }
1621
Johan Hedberg2b76f452013-03-15 17:07:04 -05001622 err = hci_req_run(&req, set_connectable_complete);
Johan Hedberg9b742462013-10-14 16:20:03 +03001623 if (err < 0) {
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001624 mgmt_pending_remove(cmd);
Johan Hedberg9b742462013-10-14 16:20:03 +03001625 if (err == -ENODATA)
Johan Hedberga81070b2013-10-19 23:38:19 +03001626 err = set_connectable_update_settings(hdev, sk,
1627 cp->val);
Johan Hedberg9b742462013-10-14 16:20:03 +03001628 goto failed;
1629 }
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001630
1631failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001632 hci_dev_unlock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001633 return err;
1634}
1635
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001636static int set_pairable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001637 u16 len)
Johan Hedberg73f22f62010-12-29 16:00:25 +02001638{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001639 struct mgmt_mode *cp = data;
Marcel Holtmann55594352013-10-06 16:11:57 -07001640 bool changed;
Johan Hedberg73f22f62010-12-29 16:00:25 +02001641 int err;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001642
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001643 BT_DBG("request for %s", hdev->name);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001644
Johan Hedberga7e80f22013-01-09 16:05:19 +02001645 if (cp->val != 0x00 && cp->val != 0x01)
1646 return cmd_status(sk, hdev->id, MGMT_OP_SET_PAIRABLE,
1647 MGMT_STATUS_INVALID_PARAMS);
1648
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001649 hci_dev_lock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001650
1651 if (cp->val)
Marcel Holtmann55594352013-10-06 16:11:57 -07001652 changed = !test_and_set_bit(HCI_PAIRABLE, &hdev->dev_flags);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001653 else
Marcel Holtmann55594352013-10-06 16:11:57 -07001654 changed = test_and_clear_bit(HCI_PAIRABLE, &hdev->dev_flags);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001655
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001656 err = send_settings_rsp(sk, MGMT_OP_SET_PAIRABLE, hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001657 if (err < 0)
Marcel Holtmann55594352013-10-06 16:11:57 -07001658 goto unlock;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001659
Marcel Holtmann55594352013-10-06 16:11:57 -07001660 if (changed)
1661 err = new_settings(hdev, sk);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001662
Marcel Holtmann55594352013-10-06 16:11:57 -07001663unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001664 hci_dev_unlock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001665 return err;
1666}
Johan Hedberg72a734e2010-12-30 00:38:22 +02001667
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001668static int set_link_security(struct sock *sk, struct hci_dev *hdev, void *data,
1669 u16 len)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001670{
1671 struct mgmt_mode *cp = data;
1672 struct pending_cmd *cmd;
Johan Hedberge6fe7982013-10-02 15:45:22 +03001673 u8 val, status;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001674 int err;
1675
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001676 BT_DBG("request for %s", hdev->name);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001677
Johan Hedberge6fe7982013-10-02 15:45:22 +03001678 status = mgmt_bredr_support(hdev);
1679 if (status)
Johan Hedberg33c525c2012-10-24 21:11:58 +03001680 return cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03001681 status);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001682
Johan Hedberga7e80f22013-01-09 16:05:19 +02001683 if (cp->val != 0x00 && cp->val != 0x01)
1684 return cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
1685 MGMT_STATUS_INVALID_PARAMS);
1686
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001687 hci_dev_lock(hdev);
1688
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001689 if (!hdev_is_powered(hdev)) {
Johan Hedberg47990ea2012-02-22 11:58:37 +02001690 bool changed = false;
1691
1692 if (!!cp->val != test_bit(HCI_LINK_SECURITY,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03001693 &hdev->dev_flags)) {
Johan Hedberg47990ea2012-02-22 11:58:37 +02001694 change_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
1695 changed = true;
1696 }
1697
1698 err = send_settings_rsp(sk, MGMT_OP_SET_LINK_SECURITY, hdev);
1699 if (err < 0)
1700 goto failed;
1701
1702 if (changed)
1703 err = new_settings(hdev, sk);
1704
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001705 goto failed;
1706 }
1707
1708 if (mgmt_pending_find(MGMT_OP_SET_LINK_SECURITY, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001709 err = cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001710 MGMT_STATUS_BUSY);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001711 goto failed;
1712 }
1713
1714 val = !!cp->val;
1715
1716 if (test_bit(HCI_AUTH, &hdev->flags) == val) {
1717 err = send_settings_rsp(sk, MGMT_OP_SET_LINK_SECURITY, hdev);
1718 goto failed;
1719 }
1720
1721 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LINK_SECURITY, hdev, data, len);
1722 if (!cmd) {
1723 err = -ENOMEM;
1724 goto failed;
1725 }
1726
1727 err = hci_send_cmd(hdev, HCI_OP_WRITE_AUTH_ENABLE, sizeof(val), &val);
1728 if (err < 0) {
1729 mgmt_pending_remove(cmd);
1730 goto failed;
1731 }
1732
1733failed:
1734 hci_dev_unlock(hdev);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001735 return err;
1736}
1737
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001738static int set_ssp(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001739{
1740 struct mgmt_mode *cp = data;
1741 struct pending_cmd *cmd;
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001742 u8 status;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001743 int err;
1744
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001745 BT_DBG("request for %s", hdev->name);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001746
Marcel Holtmanncdba5282013-10-02 21:31:52 -07001747 status = mgmt_bredr_support(hdev);
1748 if (status)
1749 return cmd_status(sk, hdev->id, MGMT_OP_SET_SSP, status);
1750
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001751 if (!lmp_ssp_capable(hdev))
1752 return cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1753 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001754
Johan Hedberga7e80f22013-01-09 16:05:19 +02001755 if (cp->val != 0x00 && cp->val != 0x01)
1756 return cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1757 MGMT_STATUS_INVALID_PARAMS);
1758
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001759 hci_dev_lock(hdev);
Johan Hedberg6c8f12c2012-02-22 16:35:26 +02001760
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001761 if (!hdev_is_powered(hdev)) {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001762 bool changed;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02001763
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001764 if (cp->val) {
1765 changed = !test_and_set_bit(HCI_SSP_ENABLED,
1766 &hdev->dev_flags);
1767 } else {
1768 changed = test_and_clear_bit(HCI_SSP_ENABLED,
1769 &hdev->dev_flags);
1770 if (!changed)
1771 changed = test_and_clear_bit(HCI_HS_ENABLED,
1772 &hdev->dev_flags);
1773 else
1774 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02001775 }
1776
1777 err = send_settings_rsp(sk, MGMT_OP_SET_SSP, hdev);
1778 if (err < 0)
1779 goto failed;
1780
1781 if (changed)
1782 err = new_settings(hdev, sk);
1783
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001784 goto failed;
1785 }
1786
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001787 if (mgmt_pending_find(MGMT_OP_SET_SSP, hdev) ||
1788 mgmt_pending_find(MGMT_OP_SET_HS, hdev)) {
Szymon Jancd97dcb62012-03-16 16:02:56 +01001789 err = cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1790 MGMT_STATUS_BUSY);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001791 goto failed;
1792 }
1793
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001794 if (!!cp->val == test_bit(HCI_SSP_ENABLED, &hdev->dev_flags)) {
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001795 err = send_settings_rsp(sk, MGMT_OP_SET_SSP, hdev);
1796 goto failed;
1797 }
1798
1799 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SSP, hdev, data, len);
1800 if (!cmd) {
1801 err = -ENOMEM;
1802 goto failed;
1803 }
1804
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001805 err = hci_send_cmd(hdev, HCI_OP_WRITE_SSP_MODE, 1, &cp->val);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001806 if (err < 0) {
1807 mgmt_pending_remove(cmd);
1808 goto failed;
1809 }
1810
1811failed:
1812 hci_dev_unlock(hdev);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001813 return err;
1814}
1815
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001816static int set_hs(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001817{
1818 struct mgmt_mode *cp = data;
Marcel Holtmannee392692013-10-01 22:59:23 -07001819 bool changed;
Johan Hedberge6fe7982013-10-02 15:45:22 +03001820 u8 status;
Marcel Holtmannee392692013-10-01 22:59:23 -07001821 int err;
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001822
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001823 BT_DBG("request for %s", hdev->name);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001824
Johan Hedberge6fe7982013-10-02 15:45:22 +03001825 status = mgmt_bredr_support(hdev);
1826 if (status)
1827 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS, status);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001828
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001829 if (!lmp_ssp_capable(hdev))
1830 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1831 MGMT_STATUS_NOT_SUPPORTED);
1832
1833 if (!test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
1834 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1835 MGMT_STATUS_REJECTED);
1836
Johan Hedberga7e80f22013-01-09 16:05:19 +02001837 if (cp->val != 0x00 && cp->val != 0x01)
1838 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1839 MGMT_STATUS_INVALID_PARAMS);
1840
Marcel Holtmannee392692013-10-01 22:59:23 -07001841 hci_dev_lock(hdev);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001842
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001843 if (cp->val) {
Marcel Holtmannee392692013-10-01 22:59:23 -07001844 changed = !test_and_set_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001845 } else {
1846 if (hdev_is_powered(hdev)) {
1847 err = cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1848 MGMT_STATUS_REJECTED);
1849 goto unlock;
1850 }
1851
Marcel Holtmannee392692013-10-01 22:59:23 -07001852 changed = test_and_clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001853 }
Marcel Holtmannee392692013-10-01 22:59:23 -07001854
1855 err = send_settings_rsp(sk, MGMT_OP_SET_HS, hdev);
1856 if (err < 0)
1857 goto unlock;
1858
1859 if (changed)
1860 err = new_settings(hdev, sk);
1861
1862unlock:
1863 hci_dev_unlock(hdev);
1864 return err;
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001865}
1866
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001867static void le_enable_complete(struct hci_dev *hdev, u8 status)
1868{
1869 struct cmd_lookup match = { NULL, hdev };
1870
1871 if (status) {
1872 u8 mgmt_err = mgmt_status(status);
1873
1874 mgmt_pending_foreach(MGMT_OP_SET_LE, hdev, cmd_status_rsp,
1875 &mgmt_err);
1876 return;
1877 }
1878
1879 mgmt_pending_foreach(MGMT_OP_SET_LE, hdev, settings_rsp, &match);
1880
1881 new_settings(hdev, match.sk);
1882
1883 if (match.sk)
1884 sock_put(match.sk);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001885
1886 /* Make sure the controller has a good default for
1887 * advertising data. Restrict the update to when LE
1888 * has actually been enabled. During power on, the
1889 * update in powered_update_hci will take care of it.
1890 */
1891 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
1892 struct hci_request req;
1893
1894 hci_dev_lock(hdev);
1895
1896 hci_req_init(&req, hdev);
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07001897 update_adv_data(&req);
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07001898 update_scan_rsp_data(&req);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001899 hci_req_run(&req, NULL);
1900
1901 hci_dev_unlock(hdev);
1902 }
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001903}
1904
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001905static int set_le(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg06199cf2012-02-22 16:37:11 +02001906{
1907 struct mgmt_mode *cp = data;
1908 struct hci_cp_write_le_host_supported hci_cp;
1909 struct pending_cmd *cmd;
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001910 struct hci_request req;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001911 int err;
Johan Hedberg0b60eba2012-02-28 00:57:24 +02001912 u8 val, enabled;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001913
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001914 BT_DBG("request for %s", hdev->name);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001915
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001916 if (!lmp_le_capable(hdev))
1917 return cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1918 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg1de028c2012-02-29 19:55:35 -08001919
Johan Hedberga7e80f22013-01-09 16:05:19 +02001920 if (cp->val != 0x00 && cp->val != 0x01)
1921 return cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1922 MGMT_STATUS_INVALID_PARAMS);
1923
Johan Hedbergc73eee92013-04-19 18:35:21 +03001924 /* LE-only devices do not allow toggling LE on/off */
Johan Hedberg56f87902013-10-02 13:43:13 +03001925 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedbergc73eee92013-04-19 18:35:21 +03001926 return cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1927 MGMT_STATUS_REJECTED);
1928
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001929 hci_dev_lock(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001930
1931 val = !!cp->val;
Gustavo Padovanffa88e02012-11-23 16:50:51 -02001932 enabled = lmp_host_le_capable(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001933
Johan Hedberg0b60eba2012-02-28 00:57:24 +02001934 if (!hdev_is_powered(hdev) || val == enabled) {
Johan Hedberg06199cf2012-02-22 16:37:11 +02001935 bool changed = false;
1936
1937 if (val != test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
1938 change_bit(HCI_LE_ENABLED, &hdev->dev_flags);
1939 changed = true;
1940 }
1941
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02001942 if (!val && test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
1943 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedbergeeca6f82013-09-25 13:26:09 +03001944 changed = true;
1945 }
1946
Johan Hedberg06199cf2012-02-22 16:37:11 +02001947 err = send_settings_rsp(sk, MGMT_OP_SET_LE, hdev);
1948 if (err < 0)
Johan Hedberg1de028c2012-02-29 19:55:35 -08001949 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001950
1951 if (changed)
1952 err = new_settings(hdev, sk);
1953
Johan Hedberg1de028c2012-02-29 19:55:35 -08001954 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001955 }
1956
Johan Hedberg4375f102013-09-25 13:26:10 +03001957 if (mgmt_pending_find(MGMT_OP_SET_LE, hdev) ||
1958 mgmt_pending_find(MGMT_OP_SET_ADVERTISING, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001959 err = cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001960 MGMT_STATUS_BUSY);
Johan Hedberg1de028c2012-02-29 19:55:35 -08001961 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001962 }
1963
1964 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LE, hdev, data, len);
1965 if (!cmd) {
1966 err = -ENOMEM;
Johan Hedberg1de028c2012-02-29 19:55:35 -08001967 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001968 }
1969
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001970 hci_req_init(&req, hdev);
1971
Johan Hedberg06199cf2012-02-22 16:37:11 +02001972 memset(&hci_cp, 0, sizeof(hci_cp));
1973
1974 if (val) {
1975 hci_cp.le = val;
Gustavo Padovanffa88e02012-11-23 16:50:51 -02001976 hci_cp.simul = lmp_le_br_capable(hdev);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001977 } else {
1978 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
1979 disable_advertising(&req);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001980 }
1981
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001982 hci_req_add(&req, HCI_OP_WRITE_LE_HOST_SUPPORTED, sizeof(hci_cp),
1983 &hci_cp);
1984
1985 err = hci_req_run(&req, le_enable_complete);
Syam Sidhardhan0c01bc42012-04-12 20:33:21 +05301986 if (err < 0)
Johan Hedberg06199cf2012-02-22 16:37:11 +02001987 mgmt_pending_remove(cmd);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001988
Johan Hedberg1de028c2012-02-29 19:55:35 -08001989unlock:
1990 hci_dev_unlock(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001991 return err;
1992}
1993
Johan Hedberg0cab9c82013-03-15 17:06:54 -05001994/* This is a helper function to test for pending mgmt commands that can
1995 * cause CoD or EIR HCI commands. We can only allow one such pending
1996 * mgmt command at a time since otherwise we cannot easily track what
1997 * the current values are, will be, and based on that calculate if a new
1998 * HCI command needs to be sent and if yes with what value.
1999 */
2000static bool pending_eir_or_class(struct hci_dev *hdev)
2001{
2002 struct pending_cmd *cmd;
2003
2004 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
2005 switch (cmd->opcode) {
2006 case MGMT_OP_ADD_UUID:
2007 case MGMT_OP_REMOVE_UUID:
2008 case MGMT_OP_SET_DEV_CLASS:
2009 case MGMT_OP_SET_POWERED:
2010 return true;
2011 }
2012 }
2013
2014 return false;
2015}
2016
Johan Hedberg83be8ec2013-01-27 00:31:29 +02002017static const u8 bluetooth_base_uuid[] = {
2018 0xfb, 0x34, 0x9b, 0x5f, 0x80, 0x00, 0x00, 0x80,
2019 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
2020};
2021
2022static u8 get_uuid_size(const u8 *uuid)
2023{
2024 u32 val;
2025
2026 if (memcmp(uuid, bluetooth_base_uuid, 12))
2027 return 128;
2028
2029 val = get_unaligned_le32(&uuid[12]);
2030 if (val > 0xffff)
2031 return 32;
2032
2033 return 16;
2034}
2035
Johan Hedberg92da6092013-03-15 17:06:55 -05002036static void mgmt_class_complete(struct hci_dev *hdev, u16 mgmt_op, u8 status)
2037{
2038 struct pending_cmd *cmd;
2039
2040 hci_dev_lock(hdev);
2041
2042 cmd = mgmt_pending_find(mgmt_op, hdev);
2043 if (!cmd)
2044 goto unlock;
2045
2046 cmd_complete(cmd->sk, cmd->index, cmd->opcode, mgmt_status(status),
2047 hdev->dev_class, 3);
2048
2049 mgmt_pending_remove(cmd);
2050
2051unlock:
2052 hci_dev_unlock(hdev);
2053}
2054
2055static void add_uuid_complete(struct hci_dev *hdev, u8 status)
2056{
2057 BT_DBG("status 0x%02x", status);
2058
2059 mgmt_class_complete(hdev, MGMT_OP_ADD_UUID, status);
2060}
2061
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002062static int add_uuid(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002063{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002064 struct mgmt_cp_add_uuid *cp = data;
Johan Hedberg90e70452012-02-23 23:09:40 +02002065 struct pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05002066 struct hci_request req;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002067 struct bt_uuid *uuid;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002068 int err;
2069
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002070 BT_DBG("request for %s", hdev->name);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002071
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002072 hci_dev_lock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002073
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002074 if (pending_eir_or_class(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002075 err = cmd_status(sk, hdev->id, MGMT_OP_ADD_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002076 MGMT_STATUS_BUSY);
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02002077 goto failed;
2078 }
2079
Andre Guedes92c4c202012-06-07 19:05:44 -03002080 uuid = kmalloc(sizeof(*uuid), GFP_KERNEL);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002081 if (!uuid) {
2082 err = -ENOMEM;
2083 goto failed;
2084 }
2085
2086 memcpy(uuid->uuid, cp->uuid, 16);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002087 uuid->svc_hint = cp->svc_hint;
Johan Hedberg83be8ec2013-01-27 00:31:29 +02002088 uuid->size = get_uuid_size(cp->uuid);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002089
Johan Hedbergde66aa62013-01-27 00:31:27 +02002090 list_add_tail(&uuid->list, &hdev->uuids);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002091
Johan Hedberg890ea892013-03-15 17:06:52 -05002092 hci_req_init(&req, hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002093
Johan Hedberg890ea892013-03-15 17:06:52 -05002094 update_class(&req);
2095 update_eir(&req);
2096
Johan Hedberg92da6092013-03-15 17:06:55 -05002097 err = hci_req_run(&req, add_uuid_complete);
2098 if (err < 0) {
2099 if (err != -ENODATA)
2100 goto failed;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +03002101
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002102 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_UUID, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002103 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002104 goto failed;
2105 }
2106
2107 cmd = mgmt_pending_add(sk, MGMT_OP_ADD_UUID, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002108 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002109 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002110 goto failed;
2111 }
2112
2113 err = 0;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002114
2115failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002116 hci_dev_unlock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002117 return err;
2118}
2119
Johan Hedberg24b78d02012-02-23 23:24:30 +02002120static bool enable_service_cache(struct hci_dev *hdev)
2121{
2122 if (!hdev_is_powered(hdev))
2123 return false;
2124
2125 if (!test_and_set_bit(HCI_SERVICE_CACHE, &hdev->dev_flags)) {
Johan Hedberg46818ed2013-01-14 22:33:52 +02002126 queue_delayed_work(hdev->workqueue, &hdev->service_cache,
2127 CACHE_TIMEOUT);
Johan Hedberg24b78d02012-02-23 23:24:30 +02002128 return true;
2129 }
2130
2131 return false;
2132}
2133
Johan Hedberg92da6092013-03-15 17:06:55 -05002134static void remove_uuid_complete(struct hci_dev *hdev, u8 status)
2135{
2136 BT_DBG("status 0x%02x", status);
2137
2138 mgmt_class_complete(hdev, MGMT_OP_REMOVE_UUID, status);
2139}
2140
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002141static int remove_uuid(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002142 u16 len)
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002143{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002144 struct mgmt_cp_remove_uuid *cp = data;
Johan Hedberg90e70452012-02-23 23:09:40 +02002145 struct pending_cmd *cmd;
Johan Hedberg056341c2013-01-27 00:31:30 +02002146 struct bt_uuid *match, *tmp;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002147 u8 bt_uuid_any[] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
Johan Hedberg890ea892013-03-15 17:06:52 -05002148 struct hci_request req;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002149 int err, found;
2150
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002151 BT_DBG("request for %s", hdev->name);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002152
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002153 hci_dev_lock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002154
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002155 if (pending_eir_or_class(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002156 err = cmd_status(sk, hdev->id, MGMT_OP_REMOVE_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002157 MGMT_STATUS_BUSY);
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02002158 goto unlock;
2159 }
2160
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002161 if (memcmp(cp->uuid, bt_uuid_any, 16) == 0) {
Johan Hedberg35f74982014-02-18 17:14:32 +02002162 hci_uuids_clear(hdev);
Johan Hedberg4004b6d2012-02-23 21:30:12 +02002163
Johan Hedberg24b78d02012-02-23 23:24:30 +02002164 if (enable_service_cache(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002165 err = cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002166 0, hdev->dev_class, 3);
Johan Hedberg24b78d02012-02-23 23:24:30 +02002167 goto unlock;
2168 }
Johan Hedberg4004b6d2012-02-23 21:30:12 +02002169
Johan Hedberg9246a862012-02-23 21:33:16 +02002170 goto update_class;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002171 }
2172
2173 found = 0;
2174
Johan Hedberg056341c2013-01-27 00:31:30 +02002175 list_for_each_entry_safe(match, tmp, &hdev->uuids, list) {
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002176 if (memcmp(match->uuid, cp->uuid, 16) != 0)
2177 continue;
2178
2179 list_del(&match->list);
Johan Hedberg482049f2012-11-08 10:25:26 +01002180 kfree(match);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002181 found++;
2182 }
2183
2184 if (found == 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002185 err = cmd_status(sk, hdev->id, MGMT_OP_REMOVE_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002186 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002187 goto unlock;
2188 }
2189
Johan Hedberg9246a862012-02-23 21:33:16 +02002190update_class:
Johan Hedberg890ea892013-03-15 17:06:52 -05002191 hci_req_init(&req, hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002192
Johan Hedberg890ea892013-03-15 17:06:52 -05002193 update_class(&req);
2194 update_eir(&req);
2195
Johan Hedberg92da6092013-03-15 17:06:55 -05002196 err = hci_req_run(&req, remove_uuid_complete);
2197 if (err < 0) {
2198 if (err != -ENODATA)
2199 goto unlock;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +03002200
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002201 err = cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_UUID, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002202 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002203 goto unlock;
2204 }
2205
2206 cmd = mgmt_pending_add(sk, MGMT_OP_REMOVE_UUID, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002207 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002208 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002209 goto unlock;
2210 }
2211
2212 err = 0;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002213
2214unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002215 hci_dev_unlock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002216 return err;
2217}
2218
Johan Hedberg92da6092013-03-15 17:06:55 -05002219static void set_class_complete(struct hci_dev *hdev, u8 status)
2220{
2221 BT_DBG("status 0x%02x", status);
2222
2223 mgmt_class_complete(hdev, MGMT_OP_SET_DEV_CLASS, status);
2224}
2225
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002226static int set_dev_class(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002227 u16 len)
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002228{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002229 struct mgmt_cp_set_dev_class *cp = data;
Johan Hedberg90e70452012-02-23 23:09:40 +02002230 struct pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05002231 struct hci_request req;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002232 int err;
2233
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002234 BT_DBG("request for %s", hdev->name);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002235
Marcel Holtmann6203fc92013-10-02 23:37:29 -07002236 if (!lmp_bredr_capable(hdev))
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02002237 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2238 MGMT_STATUS_NOT_SUPPORTED);
2239
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002240 hci_dev_lock(hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002241
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002242 if (pending_eir_or_class(hdev)) {
2243 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2244 MGMT_STATUS_BUSY);
2245 goto unlock;
2246 }
2247
2248 if ((cp->minor & 0x03) != 0 || (cp->major & 0xe0) != 0) {
2249 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2250 MGMT_STATUS_INVALID_PARAMS);
2251 goto unlock;
2252 }
2253
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002254 hdev->major_class = cp->major;
2255 hdev->minor_class = cp->minor;
2256
Johan Hedberg932f5ff2012-02-22 22:11:32 +02002257 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002258 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_DEV_CLASS, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002259 hdev->dev_class, 3);
Johan Hedberg932f5ff2012-02-22 22:11:32 +02002260 goto unlock;
2261 }
2262
Johan Hedberg890ea892013-03-15 17:06:52 -05002263 hci_req_init(&req, hdev);
2264
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02002265 if (test_and_clear_bit(HCI_SERVICE_CACHE, &hdev->dev_flags)) {
Johan Hedberg7d785252011-12-15 00:47:39 +02002266 hci_dev_unlock(hdev);
2267 cancel_delayed_work_sync(&hdev->service_cache);
2268 hci_dev_lock(hdev);
Johan Hedberg890ea892013-03-15 17:06:52 -05002269 update_eir(&req);
Johan Hedberg7d785252011-12-15 00:47:39 +02002270 }
Johan Hedberg14c0b602011-12-15 00:47:37 +02002271
Johan Hedberg890ea892013-03-15 17:06:52 -05002272 update_class(&req);
2273
Johan Hedberg92da6092013-03-15 17:06:55 -05002274 err = hci_req_run(&req, set_class_complete);
2275 if (err < 0) {
2276 if (err != -ENODATA)
2277 goto unlock;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002278
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002279 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_DEV_CLASS, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002280 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002281 goto unlock;
2282 }
2283
2284 cmd = mgmt_pending_add(sk, MGMT_OP_SET_DEV_CLASS, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002285 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002286 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002287 goto unlock;
2288 }
2289
2290 err = 0;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002291
Johan Hedbergb5235a62012-02-21 14:32:24 +02002292unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002293 hci_dev_unlock(hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002294 return err;
2295}
2296
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002297static int load_link_keys(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002298 u16 len)
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002299{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002300 struct mgmt_cp_load_link_keys *cp = data;
Szymon Janc4e51eae2011-02-25 19:05:48 +01002301 u16 key_count, expected_len;
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002302 bool changed;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002303 int i;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002304
Marcel Holtmann9060d5c2013-10-02 21:16:07 -07002305 BT_DBG("request for %s", hdev->name);
2306
2307 if (!lmp_bredr_capable(hdev))
2308 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2309 MGMT_STATUS_NOT_SUPPORTED);
2310
Marcel Holtmann1f350c82012-03-12 20:31:08 -07002311 key_count = __le16_to_cpu(cp->key_count);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002312
Johan Hedberg86742e12011-11-07 23:13:38 +02002313 expected_len = sizeof(*cp) + key_count *
2314 sizeof(struct mgmt_link_key_info);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002315 if (expected_len != len) {
Johan Hedberg86742e12011-11-07 23:13:38 +02002316 BT_ERR("load_link_keys: expected %u bytes, got %u bytes",
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002317 len, expected_len);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002318 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002319 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002320 }
2321
Johan Hedberg4ae14302013-01-20 14:27:13 +02002322 if (cp->debug_keys != 0x00 && cp->debug_keys != 0x01)
2323 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2324 MGMT_STATUS_INVALID_PARAMS);
2325
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002326 BT_DBG("%s debug_keys %u key_count %u", hdev->name, cp->debug_keys,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002327 key_count);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002328
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002329 for (i = 0; i < key_count; i++) {
2330 struct mgmt_link_key_info *key = &cp->keys[i];
2331
Marcel Holtmann8e991132014-01-10 02:07:25 -08002332 if (key->addr.type != BDADDR_BREDR || key->type > 0x08)
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002333 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2334 MGMT_STATUS_INVALID_PARAMS);
2335 }
2336
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002337 hci_dev_lock(hdev);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002338
2339 hci_link_keys_clear(hdev);
2340
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002341 if (cp->debug_keys)
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002342 changed = !test_and_set_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002343 else
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002344 changed = test_and_clear_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
2345
2346 if (changed)
2347 new_settings(hdev, NULL);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002348
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002349 for (i = 0; i < key_count; i++) {
Johan Hedberg86742e12011-11-07 23:13:38 +02002350 struct mgmt_link_key_info *key = &cp->keys[i];
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002351
Johan Hedbergd753fdc2012-02-17 14:06:34 +02002352 hci_add_link_key(hdev, NULL, 0, &key->addr.bdaddr, key->val,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002353 key->type, key->pin_len);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002354 }
2355
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002356 cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS, 0, NULL, 0);
Johan Hedberg0e5f8752011-11-11 16:18:54 +02002357
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002358 hci_dev_unlock(hdev);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002359
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002360 return 0;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002361}
2362
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002363static int device_unpaired(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002364 u8 addr_type, struct sock *skip_sk)
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002365{
2366 struct mgmt_ev_device_unpaired ev;
2367
2368 bacpy(&ev.addr.bdaddr, bdaddr);
2369 ev.addr.type = addr_type;
2370
2371 return mgmt_event(MGMT_EV_DEVICE_UNPAIRED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002372 skip_sk);
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002373}
2374
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002375static int unpair_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002376 u16 len)
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002377{
Johan Hedberg124f6e32012-02-09 13:50:12 +02002378 struct mgmt_cp_unpair_device *cp = data;
2379 struct mgmt_rp_unpair_device rp;
Johan Hedberga8a1d192011-11-10 15:54:38 +02002380 struct hci_cp_disconnect dc;
2381 struct pending_cmd *cmd;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002382 struct hci_conn *conn;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002383 int err;
2384
Johan Hedberga8a1d192011-11-10 15:54:38 +02002385 memset(&rp, 0, sizeof(rp));
Johan Hedberg124f6e32012-02-09 13:50:12 +02002386 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2387 rp.addr.type = cp->addr.type;
Johan Hedberga8a1d192011-11-10 15:54:38 +02002388
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002389 if (!bdaddr_type_is_valid(cp->addr.type))
2390 return cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2391 MGMT_STATUS_INVALID_PARAMS,
2392 &rp, sizeof(rp));
2393
Johan Hedberg118da702013-01-20 14:27:20 +02002394 if (cp->disconnect != 0x00 && cp->disconnect != 0x01)
2395 return cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2396 MGMT_STATUS_INVALID_PARAMS,
2397 &rp, sizeof(rp));
2398
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002399 hci_dev_lock(hdev);
2400
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002401 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002402 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002403 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002404 goto unlock;
2405 }
2406
Johan Hedberge0b2b272014-02-18 17:14:31 +02002407 if (cp->addr.type == BDADDR_BREDR) {
Johan Hedberg124f6e32012-02-09 13:50:12 +02002408 err = hci_remove_link_key(hdev, &cp->addr.bdaddr);
Johan Hedberge0b2b272014-02-18 17:14:31 +02002409 } else {
2410 u8 addr_type;
2411
2412 if (cp->addr.type == BDADDR_LE_PUBLIC)
2413 addr_type = ADDR_LE_DEV_PUBLIC;
2414 else
2415 addr_type = ADDR_LE_DEV_RANDOM;
2416
Johan Hedberga7ec7332014-02-18 17:14:35 +02002417 hci_remove_irk(hdev, &cp->addr.bdaddr, addr_type);
2418
Johan Hedberge0b2b272014-02-18 17:14:31 +02002419 err = hci_remove_ltk(hdev, &cp->addr.bdaddr, addr_type);
2420 }
Vinicius Costa Gomesb0dbfb42012-02-02 21:08:03 -03002421
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002422 if (err < 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002423 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002424 MGMT_STATUS_NOT_PAIRED, &rp, sizeof(rp));
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002425 goto unlock;
2426 }
2427
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002428 if (cp->disconnect) {
Andre Guedes591f47f2012-04-24 21:02:49 -03002429 if (cp->addr.type == BDADDR_BREDR)
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002430 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002431 &cp->addr.bdaddr);
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002432 else
2433 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002434 &cp->addr.bdaddr);
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002435 } else {
2436 conn = NULL;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002437 }
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002438
Johan Hedberga8a1d192011-11-10 15:54:38 +02002439 if (!conn) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002440 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002441 &rp, sizeof(rp));
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002442 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, sk);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002443 goto unlock;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002444 }
2445
Johan Hedberg124f6e32012-02-09 13:50:12 +02002446 cmd = mgmt_pending_add(sk, MGMT_OP_UNPAIR_DEVICE, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002447 sizeof(*cp));
Johan Hedberga8a1d192011-11-10 15:54:38 +02002448 if (!cmd) {
2449 err = -ENOMEM;
2450 goto unlock;
2451 }
2452
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002453 dc.handle = cpu_to_le16(conn->handle);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002454 dc.reason = 0x13; /* Remote User Terminated Connection */
2455 err = hci_send_cmd(hdev, HCI_OP_DISCONNECT, sizeof(dc), &dc);
2456 if (err < 0)
2457 mgmt_pending_remove(cmd);
2458
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002459unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002460 hci_dev_unlock(hdev);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002461 return err;
2462}
2463
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002464static int disconnect(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002465 u16 len)
Johan Hedberg8962ee72011-01-20 12:40:27 +02002466{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002467 struct mgmt_cp_disconnect *cp = data;
Johan Hedberg06a63b12013-01-20 14:27:21 +02002468 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002469 struct hci_cp_disconnect dc;
Johan Hedberg366a0332011-02-19 12:05:55 -03002470 struct pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002471 struct hci_conn *conn;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002472 int err;
2473
2474 BT_DBG("");
2475
Johan Hedberg06a63b12013-01-20 14:27:21 +02002476 memset(&rp, 0, sizeof(rp));
2477 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2478 rp.addr.type = cp->addr.type;
2479
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002480 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg06a63b12013-01-20 14:27:21 +02002481 return cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2482 MGMT_STATUS_INVALID_PARAMS,
2483 &rp, sizeof(rp));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002484
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002485 hci_dev_lock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002486
2487 if (!test_bit(HCI_UP, &hdev->flags)) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002488 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2489 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002490 goto failed;
2491 }
2492
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002493 if (mgmt_pending_find(MGMT_OP_DISCONNECT, hdev)) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002494 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2495 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002496 goto failed;
2497 }
2498
Andre Guedes591f47f2012-04-24 21:02:49 -03002499 if (cp->addr.type == BDADDR_BREDR)
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03002500 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
2501 &cp->addr.bdaddr);
Johan Hedberg88c3df12012-02-09 14:27:38 +02002502 else
2503 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &cp->addr.bdaddr);
Vinicius Costa Gomes365227e2011-05-06 18:41:44 -03002504
Vishal Agarwalf9607272012-06-13 05:32:43 +05302505 if (!conn || conn->state == BT_OPEN || conn->state == BT_CLOSED) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002506 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2507 MGMT_STATUS_NOT_CONNECTED, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002508 goto failed;
2509 }
2510
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002511 cmd = mgmt_pending_add(sk, MGMT_OP_DISCONNECT, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002512 if (!cmd) {
2513 err = -ENOMEM;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002514 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002515 }
Johan Hedberg8962ee72011-01-20 12:40:27 +02002516
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002517 dc.handle = cpu_to_le16(conn->handle);
Andre Guedes3701f942012-06-11 18:41:12 -03002518 dc.reason = HCI_ERROR_REMOTE_USER_TERM;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002519
2520 err = hci_send_cmd(hdev, HCI_OP_DISCONNECT, sizeof(dc), &dc);
2521 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002522 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002523
2524failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002525 hci_dev_unlock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002526 return err;
2527}
2528
Andre Guedes57c14772012-04-24 21:02:50 -03002529static u8 link_to_bdaddr(u8 link_type, u8 addr_type)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002530{
2531 switch (link_type) {
2532 case LE_LINK:
Johan Hedberg48264f02011-11-09 13:58:58 +02002533 switch (addr_type) {
2534 case ADDR_LE_DEV_PUBLIC:
Andre Guedes591f47f2012-04-24 21:02:49 -03002535 return BDADDR_LE_PUBLIC;
Andre Guedes0ed09142012-04-03 08:46:54 -03002536
Johan Hedberg48264f02011-11-09 13:58:58 +02002537 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002538 /* Fallback to LE Random address type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002539 return BDADDR_LE_RANDOM;
Johan Hedberg48264f02011-11-09 13:58:58 +02002540 }
Andre Guedes0ed09142012-04-03 08:46:54 -03002541
Johan Hedberg4c659c32011-11-07 23:13:39 +02002542 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002543 /* Fallback to BR/EDR type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002544 return BDADDR_BREDR;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002545 }
2546}
2547
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002548static int get_connections(struct sock *sk, struct hci_dev *hdev, void *data,
2549 u16 data_len)
Johan Hedberg2784eb42011-01-21 13:56:35 +02002550{
Johan Hedberg2784eb42011-01-21 13:56:35 +02002551 struct mgmt_rp_get_connections *rp;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +02002552 struct hci_conn *c;
Johan Hedberga38528f2011-01-22 06:46:43 +02002553 size_t rp_len;
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002554 int err;
2555 u16 i;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002556
2557 BT_DBG("");
2558
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002559 hci_dev_lock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002560
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002561 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002562 err = cmd_status(sk, hdev->id, MGMT_OP_GET_CONNECTIONS,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002563 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002564 goto unlock;
2565 }
2566
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002567 i = 0;
Johan Hedbergb644ba32012-01-17 21:48:47 +02002568 list_for_each_entry(c, &hdev->conn_hash.list, list) {
2569 if (test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002570 i++;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002571 }
2572
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002573 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Andre Guedes92c4c202012-06-07 19:05:44 -03002574 rp = kmalloc(rp_len, GFP_KERNEL);
Johan Hedberga38528f2011-01-22 06:46:43 +02002575 if (!rp) {
Johan Hedberg2784eb42011-01-21 13:56:35 +02002576 err = -ENOMEM;
2577 goto unlock;
2578 }
2579
Johan Hedberg2784eb42011-01-21 13:56:35 +02002580 i = 0;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002581 list_for_each_entry(c, &hdev->conn_hash.list, list) {
Johan Hedbergb644ba32012-01-17 21:48:47 +02002582 if (!test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
2583 continue;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002584 bacpy(&rp->addr[i].bdaddr, &c->dst);
Andre Guedes57c14772012-04-24 21:02:50 -03002585 rp->addr[i].type = link_to_bdaddr(c->type, c->dst_type);
Andre Guedes0ed09142012-04-03 08:46:54 -03002586 if (c->type == SCO_LINK || c->type == ESCO_LINK)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002587 continue;
2588 i++;
2589 }
2590
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002591 rp->conn_count = cpu_to_le16(i);
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002592
Johan Hedberg4c659c32011-11-07 23:13:39 +02002593 /* Recalculate length in case of filtered SCO connections, etc */
2594 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Johan Hedberg2784eb42011-01-21 13:56:35 +02002595
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002596 err = cmd_complete(sk, hdev->id, MGMT_OP_GET_CONNECTIONS, 0, rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002597 rp_len);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002598
Johan Hedberga38528f2011-01-22 06:46:43 +02002599 kfree(rp);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002600
2601unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002602 hci_dev_unlock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002603 return err;
2604}
2605
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002606static int send_pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002607 struct mgmt_cp_pin_code_neg_reply *cp)
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002608{
2609 struct pending_cmd *cmd;
2610 int err;
2611
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002612 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_NEG_REPLY, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002613 sizeof(*cp));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002614 if (!cmd)
2615 return -ENOMEM;
2616
Johan Hedbergd8457692012-02-17 14:24:57 +02002617 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_NEG_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002618 sizeof(cp->addr.bdaddr), &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002619 if (err < 0)
2620 mgmt_pending_remove(cmd);
2621
2622 return err;
2623}
2624
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002625static int pin_code_reply(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002626 u16 len)
Johan Hedberg980e1a52011-01-22 06:10:07 +02002627{
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002628 struct hci_conn *conn;
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002629 struct mgmt_cp_pin_code_reply *cp = data;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002630 struct hci_cp_pin_code_reply reply;
Johan Hedberg366a0332011-02-19 12:05:55 -03002631 struct pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002632 int err;
2633
2634 BT_DBG("");
2635
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002636 hci_dev_lock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002637
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002638 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002639 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002640 MGMT_STATUS_NOT_POWERED);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002641 goto failed;
2642 }
2643
Johan Hedbergd8457692012-02-17 14:24:57 +02002644 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002645 if (!conn) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002646 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002647 MGMT_STATUS_NOT_CONNECTED);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002648 goto failed;
2649 }
2650
2651 if (conn->pending_sec_level == BT_SECURITY_HIGH && cp->pin_len != 16) {
Johan Hedbergd8457692012-02-17 14:24:57 +02002652 struct mgmt_cp_pin_code_neg_reply ncp;
2653
2654 memcpy(&ncp.addr, &cp->addr, sizeof(ncp.addr));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002655
2656 BT_ERR("PIN code is not 16 bytes long");
2657
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002658 err = send_pin_code_neg_reply(sk, hdev, &ncp);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002659 if (err >= 0)
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002660 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002661 MGMT_STATUS_INVALID_PARAMS);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002662
2663 goto failed;
2664 }
2665
Gustavo F. Padovan00abfe42012-03-01 00:37:10 -03002666 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_REPLY, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002667 if (!cmd) {
2668 err = -ENOMEM;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002669 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002670 }
Johan Hedberg980e1a52011-01-22 06:10:07 +02002671
Johan Hedbergd8457692012-02-17 14:24:57 +02002672 bacpy(&reply.bdaddr, &cp->addr.bdaddr);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002673 reply.pin_len = cp->pin_len;
Waldemar Rymarkiewicz24718ca2011-06-01 17:28:47 +02002674 memcpy(reply.pin_code, cp->pin_code, sizeof(reply.pin_code));
Johan Hedberg980e1a52011-01-22 06:10:07 +02002675
2676 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_REPLY, sizeof(reply), &reply);
2677 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002678 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002679
2680failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002681 hci_dev_unlock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002682 return err;
2683}
2684
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002685static int set_io_capability(struct sock *sk, struct hci_dev *hdev, void *data,
2686 u16 len)
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002687{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002688 struct mgmt_cp_set_io_capability *cp = data;
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002689
2690 BT_DBG("");
2691
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002692 hci_dev_lock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002693
2694 hdev->io_capability = cp->io_capability;
2695
2696 BT_DBG("%s IO capability set to 0x%02x", hdev->name,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002697 hdev->io_capability);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002698
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002699 hci_dev_unlock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002700
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002701 return cmd_complete(sk, hdev->id, MGMT_OP_SET_IO_CAPABILITY, 0, NULL,
2702 0);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002703}
2704
Gustavo Padovan6039aa732012-05-23 04:04:18 -03002705static struct pending_cmd *find_pairing(struct hci_conn *conn)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002706{
2707 struct hci_dev *hdev = conn->hdev;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +02002708 struct pending_cmd *cmd;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002709
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002710 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
Johan Hedberge9a416b2011-02-19 12:05:56 -03002711 if (cmd->opcode != MGMT_OP_PAIR_DEVICE)
2712 continue;
2713
Johan Hedberge9a416b2011-02-19 12:05:56 -03002714 if (cmd->user_data != conn)
2715 continue;
2716
2717 return cmd;
2718 }
2719
2720 return NULL;
2721}
2722
2723static void pairing_complete(struct pending_cmd *cmd, u8 status)
2724{
2725 struct mgmt_rp_pair_device rp;
2726 struct hci_conn *conn = cmd->user_data;
2727
Johan Hedbergba4e5642011-11-11 00:07:34 +02002728 bacpy(&rp.addr.bdaddr, &conn->dst);
Andre Guedes57c14772012-04-24 21:02:50 -03002729 rp.addr.type = link_to_bdaddr(conn->type, conn->dst_type);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002730
Johan Hedbergaee9b2182012-02-18 15:07:59 +02002731 cmd_complete(cmd->sk, cmd->index, MGMT_OP_PAIR_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002732 &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002733
2734 /* So we don't get further callbacks for this connection */
2735 conn->connect_cfm_cb = NULL;
2736 conn->security_cfm_cb = NULL;
2737 conn->disconn_cfm_cb = NULL;
2738
David Herrmann76a68ba2013-04-06 20:28:37 +02002739 hci_conn_drop(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002740
Johan Hedberga664b5b2011-02-19 12:06:02 -03002741 mgmt_pending_remove(cmd);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002742}
2743
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002744void mgmt_smp_complete(struct hci_conn *conn, bool complete)
2745{
2746 u8 status = complete ? MGMT_STATUS_SUCCESS : MGMT_STATUS_FAILED;
2747 struct pending_cmd *cmd;
2748
2749 cmd = find_pairing(conn);
2750 if (cmd)
2751 pairing_complete(cmd, status);
2752}
2753
Johan Hedberge9a416b2011-02-19 12:05:56 -03002754static void pairing_complete_cb(struct hci_conn *conn, u8 status)
2755{
2756 struct pending_cmd *cmd;
2757
2758 BT_DBG("status %u", status);
2759
Johan Hedberg56e5cb82011-11-08 20:40:16 +02002760 cmd = find_pairing(conn);
2761 if (!cmd)
2762 BT_DBG("Unable to find a pending command");
2763 else
Johan Hedberge2113262012-02-18 15:20:03 +02002764 pairing_complete(cmd, mgmt_status(status));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002765}
2766
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002767static void le_pairing_complete_cb(struct hci_conn *conn, u8 status)
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302768{
2769 struct pending_cmd *cmd;
2770
2771 BT_DBG("status %u", status);
2772
2773 if (!status)
2774 return;
2775
2776 cmd = find_pairing(conn);
2777 if (!cmd)
2778 BT_DBG("Unable to find a pending command");
2779 else
2780 pairing_complete(cmd, mgmt_status(status));
2781}
2782
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002783static int pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002784 u16 len)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002785{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002786 struct mgmt_cp_pair_device *cp = data;
Johan Hedberg1425acb2011-11-11 00:07:35 +02002787 struct mgmt_rp_pair_device rp;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002788 struct pending_cmd *cmd;
2789 u8 sec_level, auth_type;
2790 struct hci_conn *conn;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002791 int err;
2792
2793 BT_DBG("");
2794
Szymon Jancf950a30e2013-01-18 12:48:07 +01002795 memset(&rp, 0, sizeof(rp));
2796 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2797 rp.addr.type = cp->addr.type;
2798
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002799 if (!bdaddr_type_is_valid(cp->addr.type))
2800 return cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2801 MGMT_STATUS_INVALID_PARAMS,
2802 &rp, sizeof(rp));
2803
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002804 hci_dev_lock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002805
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002806 if (!hdev_is_powered(hdev)) {
Szymon Jancf950a30e2013-01-18 12:48:07 +01002807 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2808 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002809 goto unlock;
2810 }
2811
Vinicius Costa Gomesc908df32011-09-02 14:51:22 -03002812 sec_level = BT_SECURITY_MEDIUM;
2813 if (cp->io_cap == 0x03)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002814 auth_type = HCI_AT_DEDICATED_BONDING;
Vinicius Costa Gomesc908df32011-09-02 14:51:22 -03002815 else
Johan Hedberge9a416b2011-02-19 12:05:56 -03002816 auth_type = HCI_AT_DEDICATED_BONDING_MITM;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002817
Andre Guedes6f77d8c2014-02-26 20:21:45 -03002818 if (cp->addr.type == BDADDR_BREDR) {
Andre Guedes04a6c582014-02-26 20:21:44 -03002819 conn = hci_connect_acl(hdev, &cp->addr.bdaddr, sec_level,
2820 auth_type);
Andre Guedes6f77d8c2014-02-26 20:21:45 -03002821 } else {
2822 u8 addr_type;
2823
2824 /* Convert from L2CAP channel address type to HCI address type
2825 */
2826 if (cp->addr.type == BDADDR_LE_PUBLIC)
2827 addr_type = ADDR_LE_DEV_PUBLIC;
2828 else
2829 addr_type = ADDR_LE_DEV_RANDOM;
2830
2831 conn = hci_connect_le(hdev, &cp->addr.bdaddr, addr_type,
Andre Guedes04a6c582014-02-26 20:21:44 -03002832 sec_level, auth_type);
Andre Guedes6f77d8c2014-02-26 20:21:45 -03002833 }
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002834
Ville Tervo30e76272011-02-22 16:10:53 -03002835 if (IS_ERR(conn)) {
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002836 int status;
2837
2838 if (PTR_ERR(conn) == -EBUSY)
2839 status = MGMT_STATUS_BUSY;
2840 else
2841 status = MGMT_STATUS_CONNECT_FAILED;
2842
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002843 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002844 status, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002845 sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002846 goto unlock;
2847 }
2848
2849 if (conn->connect_cfm_cb) {
David Herrmann76a68ba2013-04-06 20:28:37 +02002850 hci_conn_drop(conn);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002851 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002852 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002853 goto unlock;
2854 }
2855
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002856 cmd = mgmt_pending_add(sk, MGMT_OP_PAIR_DEVICE, hdev, data, len);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002857 if (!cmd) {
2858 err = -ENOMEM;
David Herrmann76a68ba2013-04-06 20:28:37 +02002859 hci_conn_drop(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002860 goto unlock;
2861 }
2862
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002863 /* For LE, just connecting isn't a proof that the pairing finished */
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002864 if (cp->addr.type == BDADDR_BREDR) {
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002865 conn->connect_cfm_cb = pairing_complete_cb;
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002866 conn->security_cfm_cb = pairing_complete_cb;
2867 conn->disconn_cfm_cb = pairing_complete_cb;
2868 } else {
2869 conn->connect_cfm_cb = le_pairing_complete_cb;
2870 conn->security_cfm_cb = le_pairing_complete_cb;
2871 conn->disconn_cfm_cb = le_pairing_complete_cb;
2872 }
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002873
Johan Hedberge9a416b2011-02-19 12:05:56 -03002874 conn->io_capability = cp->io_cap;
2875 cmd->user_data = conn;
2876
2877 if (conn->state == BT_CONNECTED &&
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002878 hci_conn_security(conn, sec_level, auth_type))
Johan Hedberge9a416b2011-02-19 12:05:56 -03002879 pairing_complete(cmd, 0);
2880
2881 err = 0;
2882
2883unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002884 hci_dev_unlock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002885 return err;
2886}
2887
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002888static int cancel_pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
2889 u16 len)
Johan Hedberg28424702012-02-02 04:02:29 +02002890{
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02002891 struct mgmt_addr_info *addr = data;
Johan Hedberg28424702012-02-02 04:02:29 +02002892 struct pending_cmd *cmd;
2893 struct hci_conn *conn;
2894 int err;
2895
2896 BT_DBG("");
2897
Johan Hedberg28424702012-02-02 04:02:29 +02002898 hci_dev_lock(hdev);
2899
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002900 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002901 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002902 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002903 goto unlock;
2904 }
2905
Johan Hedberg28424702012-02-02 04:02:29 +02002906 cmd = mgmt_pending_find(MGMT_OP_PAIR_DEVICE, hdev);
2907 if (!cmd) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002908 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002909 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002910 goto unlock;
2911 }
2912
2913 conn = cmd->user_data;
2914
2915 if (bacmp(&addr->bdaddr, &conn->dst) != 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002916 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002917 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002918 goto unlock;
2919 }
2920
2921 pairing_complete(cmd, MGMT_STATUS_CANCELLED);
2922
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002923 err = cmd_complete(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002924 addr, sizeof(*addr));
Johan Hedberg28424702012-02-02 04:02:29 +02002925unlock:
2926 hci_dev_unlock(hdev);
Johan Hedberg28424702012-02-02 04:02:29 +02002927 return err;
2928}
2929
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002930static int user_pairing_resp(struct sock *sk, struct hci_dev *hdev,
Johan Hedberg1707c602013-03-15 17:07:15 -05002931 struct mgmt_addr_info *addr, u16 mgmt_op,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002932 u16 hci_op, __le32 passkey)
Johan Hedberga5c29682011-02-19 12:05:57 -03002933{
Johan Hedberga5c29682011-02-19 12:05:57 -03002934 struct pending_cmd *cmd;
Brian Gix0df4c182011-11-16 13:53:13 -08002935 struct hci_conn *conn;
Johan Hedberga5c29682011-02-19 12:05:57 -03002936 int err;
2937
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002938 hci_dev_lock(hdev);
Johan Hedberg08ba5382011-03-16 14:29:34 +02002939
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002940 if (!hdev_is_powered(hdev)) {
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002941 err = cmd_complete(sk, hdev->id, mgmt_op,
2942 MGMT_STATUS_NOT_POWERED, addr,
2943 sizeof(*addr));
Brian Gix0df4c182011-11-16 13:53:13 -08002944 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002945 }
2946
Johan Hedberg1707c602013-03-15 17:07:15 -05002947 if (addr->type == BDADDR_BREDR)
2948 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &addr->bdaddr);
Johan Hedberg272d90d2012-02-09 15:26:12 +02002949 else
Johan Hedberg1707c602013-03-15 17:07:15 -05002950 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &addr->bdaddr);
Brian Gix47c15e22011-11-16 13:53:14 -08002951
Johan Hedberg272d90d2012-02-09 15:26:12 +02002952 if (!conn) {
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002953 err = cmd_complete(sk, hdev->id, mgmt_op,
2954 MGMT_STATUS_NOT_CONNECTED, addr,
2955 sizeof(*addr));
Johan Hedberg272d90d2012-02-09 15:26:12 +02002956 goto done;
2957 }
2958
Johan Hedberg1707c602013-03-15 17:07:15 -05002959 if (addr->type == BDADDR_LE_PUBLIC || addr->type == BDADDR_LE_RANDOM) {
Brian Gix47c15e22011-11-16 13:53:14 -08002960 /* Continue with pairing via SMP */
Brian Gix5fe57d92011-12-21 16:12:13 -08002961 err = smp_user_confirm_reply(conn, mgmt_op, passkey);
Brian Gix47c15e22011-11-16 13:53:14 -08002962
Brian Gix5fe57d92011-12-21 16:12:13 -08002963 if (!err)
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002964 err = cmd_complete(sk, hdev->id, mgmt_op,
2965 MGMT_STATUS_SUCCESS, addr,
2966 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002967 else
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002968 err = cmd_complete(sk, hdev->id, mgmt_op,
2969 MGMT_STATUS_FAILED, addr,
2970 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002971
Brian Gix47c15e22011-11-16 13:53:14 -08002972 goto done;
2973 }
2974
Johan Hedberg1707c602013-03-15 17:07:15 -05002975 cmd = mgmt_pending_add(sk, mgmt_op, hdev, addr, sizeof(*addr));
Johan Hedberga5c29682011-02-19 12:05:57 -03002976 if (!cmd) {
2977 err = -ENOMEM;
Brian Gix0df4c182011-11-16 13:53:13 -08002978 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002979 }
2980
Brian Gix0df4c182011-11-16 13:53:13 -08002981 /* Continue with pairing via HCI */
Brian Gix604086b2011-11-23 08:28:33 -08002982 if (hci_op == HCI_OP_USER_PASSKEY_REPLY) {
2983 struct hci_cp_user_passkey_reply cp;
2984
Johan Hedberg1707c602013-03-15 17:07:15 -05002985 bacpy(&cp.bdaddr, &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002986 cp.passkey = passkey;
2987 err = hci_send_cmd(hdev, hci_op, sizeof(cp), &cp);
2988 } else
Johan Hedberg1707c602013-03-15 17:07:15 -05002989 err = hci_send_cmd(hdev, hci_op, sizeof(addr->bdaddr),
2990 &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002991
Johan Hedberga664b5b2011-02-19 12:06:02 -03002992 if (err < 0)
2993 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03002994
Brian Gix0df4c182011-11-16 13:53:13 -08002995done:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002996 hci_dev_unlock(hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03002997 return err;
2998}
2999
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05303000static int pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
3001 void *data, u16 len)
3002{
3003 struct mgmt_cp_pin_code_neg_reply *cp = data;
3004
3005 BT_DBG("");
3006
Johan Hedberg1707c602013-03-15 17:07:15 -05003007 return user_pairing_resp(sk, hdev, &cp->addr,
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05303008 MGMT_OP_PIN_CODE_NEG_REPLY,
3009 HCI_OP_PIN_CODE_NEG_REPLY, 0);
3010}
3011
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003012static int user_confirm_reply(struct sock *sk, struct hci_dev *hdev, void *data,
3013 u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08003014{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003015 struct mgmt_cp_user_confirm_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08003016
3017 BT_DBG("");
3018
3019 if (len != sizeof(*cp))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003020 return cmd_status(sk, hdev->id, MGMT_OP_USER_CONFIRM_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003021 MGMT_STATUS_INVALID_PARAMS);
Brian Gix0df4c182011-11-16 13:53:13 -08003022
Johan Hedberg1707c602013-03-15 17:07:15 -05003023 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003024 MGMT_OP_USER_CONFIRM_REPLY,
3025 HCI_OP_USER_CONFIRM_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08003026}
3027
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003028static int user_confirm_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003029 void *data, u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08003030{
Johan Hedbergc9c26592011-12-15 00:47:41 +02003031 struct mgmt_cp_user_confirm_neg_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08003032
3033 BT_DBG("");
3034
Johan Hedberg1707c602013-03-15 17:07:15 -05003035 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003036 MGMT_OP_USER_CONFIRM_NEG_REPLY,
3037 HCI_OP_USER_CONFIRM_NEG_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08003038}
3039
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003040static int user_passkey_reply(struct sock *sk, struct hci_dev *hdev, void *data,
3041 u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08003042{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003043 struct mgmt_cp_user_passkey_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08003044
3045 BT_DBG("");
3046
Johan Hedberg1707c602013-03-15 17:07:15 -05003047 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003048 MGMT_OP_USER_PASSKEY_REPLY,
3049 HCI_OP_USER_PASSKEY_REPLY, cp->passkey);
Brian Gix604086b2011-11-23 08:28:33 -08003050}
3051
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003052static int user_passkey_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003053 void *data, u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08003054{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003055 struct mgmt_cp_user_passkey_neg_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08003056
3057 BT_DBG("");
3058
Johan Hedberg1707c602013-03-15 17:07:15 -05003059 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003060 MGMT_OP_USER_PASSKEY_NEG_REPLY,
3061 HCI_OP_USER_PASSKEY_NEG_REPLY, 0);
Brian Gix604086b2011-11-23 08:28:33 -08003062}
3063
Johan Hedberg13928972013-03-15 17:07:00 -05003064static void update_name(struct hci_request *req)
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003065{
Johan Hedberg13928972013-03-15 17:07:00 -05003066 struct hci_dev *hdev = req->hdev;
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003067 struct hci_cp_write_local_name cp;
3068
Johan Hedberg13928972013-03-15 17:07:00 -05003069 memcpy(cp.name, hdev->dev_name, sizeof(cp.name));
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003070
Johan Hedberg890ea892013-03-15 17:06:52 -05003071 hci_req_add(req, HCI_OP_WRITE_LOCAL_NAME, sizeof(cp), &cp);
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003072}
3073
Johan Hedberg13928972013-03-15 17:07:00 -05003074static void set_name_complete(struct hci_dev *hdev, u8 status)
3075{
3076 struct mgmt_cp_set_local_name *cp;
3077 struct pending_cmd *cmd;
3078
3079 BT_DBG("status 0x%02x", status);
3080
3081 hci_dev_lock(hdev);
3082
3083 cmd = mgmt_pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
3084 if (!cmd)
3085 goto unlock;
3086
3087 cp = cmd->param;
3088
3089 if (status)
3090 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME,
3091 mgmt_status(status));
3092 else
3093 cmd_complete(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
3094 cp, sizeof(*cp));
3095
3096 mgmt_pending_remove(cmd);
3097
3098unlock:
3099 hci_dev_unlock(hdev);
3100}
3101
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003102static int set_local_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003103 u16 len)
Johan Hedbergb312b1612011-03-16 14:29:37 +02003104{
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003105 struct mgmt_cp_set_local_name *cp = data;
Johan Hedbergb312b1612011-03-16 14:29:37 +02003106 struct pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05003107 struct hci_request req;
Johan Hedbergb312b1612011-03-16 14:29:37 +02003108 int err;
3109
3110 BT_DBG("");
3111
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003112 hci_dev_lock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003113
Johan Hedbergb3f2ca92013-03-15 17:07:03 -05003114 /* If the old values are the same as the new ones just return a
3115 * direct command complete event.
3116 */
3117 if (!memcmp(hdev->dev_name, cp->name, sizeof(hdev->dev_name)) &&
3118 !memcmp(hdev->short_name, cp->short_name,
3119 sizeof(hdev->short_name))) {
3120 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
3121 data, len);
3122 goto failed;
3123 }
3124
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003125 memcpy(hdev->short_name, cp->short_name, sizeof(hdev->short_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003126
Johan Hedbergb5235a62012-02-21 14:32:24 +02003127 if (!hdev_is_powered(hdev)) {
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003128 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003129
3130 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003131 data, len);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003132 if (err < 0)
3133 goto failed;
3134
3135 err = mgmt_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, data, len,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003136 sk);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003137
Johan Hedbergb5235a62012-02-21 14:32:24 +02003138 goto failed;
3139 }
3140
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003141 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LOCAL_NAME, hdev, data, len);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003142 if (!cmd) {
3143 err = -ENOMEM;
3144 goto failed;
3145 }
3146
Johan Hedberg13928972013-03-15 17:07:00 -05003147 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
3148
Johan Hedberg890ea892013-03-15 17:06:52 -05003149 hci_req_init(&req, hdev);
Johan Hedberg3f985052013-03-15 17:07:02 -05003150
3151 if (lmp_bredr_capable(hdev)) {
3152 update_name(&req);
3153 update_eir(&req);
3154 }
3155
Marcel Holtmann7a5f4992013-10-16 00:16:49 -07003156 /* The name is stored in the scan response data and so
3157 * no need to udpate the advertising data here.
3158 */
Johan Hedberg3f985052013-03-15 17:07:02 -05003159 if (lmp_le_capable(hdev))
Marcel Holtmann7a5f4992013-10-16 00:16:49 -07003160 update_scan_rsp_data(&req);
Johan Hedberg3f985052013-03-15 17:07:02 -05003161
Johan Hedberg13928972013-03-15 17:07:00 -05003162 err = hci_req_run(&req, set_name_complete);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003163 if (err < 0)
3164 mgmt_pending_remove(cmd);
3165
3166failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003167 hci_dev_unlock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003168 return err;
3169}
3170
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02003171static int read_local_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003172 void *data, u16 data_len)
Szymon Jancc35938b2011-03-22 13:12:21 +01003173{
Szymon Jancc35938b2011-03-22 13:12:21 +01003174 struct pending_cmd *cmd;
3175 int err;
3176
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003177 BT_DBG("%s", hdev->name);
Szymon Jancc35938b2011-03-22 13:12:21 +01003178
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003179 hci_dev_lock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003180
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003181 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003182 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003183 MGMT_STATUS_NOT_POWERED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003184 goto unlock;
3185 }
3186
Andre Guedes9a1a1992012-07-24 15:03:48 -03003187 if (!lmp_ssp_capable(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003188 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003189 MGMT_STATUS_NOT_SUPPORTED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003190 goto unlock;
3191 }
3192
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003193 if (mgmt_pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003194 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003195 MGMT_STATUS_BUSY);
Szymon Jancc35938b2011-03-22 13:12:21 +01003196 goto unlock;
3197 }
3198
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003199 cmd = mgmt_pending_add(sk, MGMT_OP_READ_LOCAL_OOB_DATA, hdev, NULL, 0);
Szymon Jancc35938b2011-03-22 13:12:21 +01003200 if (!cmd) {
3201 err = -ENOMEM;
3202 goto unlock;
3203 }
3204
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08003205 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags))
3206 err = hci_send_cmd(hdev, HCI_OP_READ_LOCAL_OOB_EXT_DATA,
3207 0, NULL);
3208 else
3209 err = hci_send_cmd(hdev, HCI_OP_READ_LOCAL_OOB_DATA, 0, NULL);
3210
Szymon Jancc35938b2011-03-22 13:12:21 +01003211 if (err < 0)
3212 mgmt_pending_remove(cmd);
3213
3214unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003215 hci_dev_unlock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003216 return err;
3217}
3218
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003219static int add_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003220 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003221{
Szymon Janc2763eda2011-03-22 13:12:22 +01003222 int err;
3223
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003224 BT_DBG("%s ", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003225
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003226 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003227
Marcel Holtmannec109112014-01-10 02:07:30 -08003228 if (len == MGMT_ADD_REMOTE_OOB_DATA_SIZE) {
3229 struct mgmt_cp_add_remote_oob_data *cp = data;
3230 u8 status;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003231
Marcel Holtmannec109112014-01-10 02:07:30 -08003232 err = hci_add_remote_oob_data(hdev, &cp->addr.bdaddr,
3233 cp->hash, cp->randomizer);
3234 if (err < 0)
3235 status = MGMT_STATUS_FAILED;
3236 else
3237 status = MGMT_STATUS_SUCCESS;
3238
3239 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3240 status, &cp->addr, sizeof(cp->addr));
3241 } else if (len == MGMT_ADD_REMOTE_OOB_EXT_DATA_SIZE) {
3242 struct mgmt_cp_add_remote_oob_ext_data *cp = data;
3243 u8 status;
3244
3245 err = hci_add_remote_oob_ext_data(hdev, &cp->addr.bdaddr,
3246 cp->hash192,
3247 cp->randomizer192,
3248 cp->hash256,
3249 cp->randomizer256);
3250 if (err < 0)
3251 status = MGMT_STATUS_FAILED;
3252 else
3253 status = MGMT_STATUS_SUCCESS;
3254
3255 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3256 status, &cp->addr, sizeof(cp->addr));
3257 } else {
3258 BT_ERR("add_remote_oob_data: invalid length of %u bytes", len);
3259 err = cmd_status(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3260 MGMT_STATUS_INVALID_PARAMS);
3261 }
Szymon Janc2763eda2011-03-22 13:12:22 +01003262
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003263 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003264 return err;
3265}
3266
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003267static int remove_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003268 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003269{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003270 struct mgmt_cp_remove_remote_oob_data *cp = data;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003271 u8 status;
Szymon Janc2763eda2011-03-22 13:12:22 +01003272 int err;
3273
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003274 BT_DBG("%s", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003275
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003276 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003277
Johan Hedberg664ce4c2012-02-09 15:44:09 +02003278 err = hci_remove_remote_oob_data(hdev, &cp->addr.bdaddr);
Szymon Janc2763eda2011-03-22 13:12:22 +01003279 if (err < 0)
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003280 status = MGMT_STATUS_INVALID_PARAMS;
Szymon Janc2763eda2011-03-22 13:12:22 +01003281 else
Szymon Janca6785be2012-12-13 15:11:21 +01003282 status = MGMT_STATUS_SUCCESS;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003283
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003284 err = cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_REMOTE_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003285 status, &cp->addr, sizeof(cp->addr));
Szymon Janc2763eda2011-03-22 13:12:22 +01003286
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003287 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003288 return err;
3289}
3290
Andre Guedes41dc2bd2013-04-30 15:29:30 -03003291static int mgmt_start_discovery_failed(struct hci_dev *hdev, u8 status)
3292{
3293 struct pending_cmd *cmd;
3294 u8 type;
3295 int err;
3296
3297 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3298
3299 cmd = mgmt_pending_find(MGMT_OP_START_DISCOVERY, hdev);
3300 if (!cmd)
3301 return -ENOENT;
3302
3303 type = hdev->discovery.type;
3304
3305 err = cmd_complete(cmd->sk, hdev->id, cmd->opcode, mgmt_status(status),
3306 &type, sizeof(type));
3307 mgmt_pending_remove(cmd);
3308
3309 return err;
3310}
3311
Andre Guedes7c307722013-04-30 15:29:28 -03003312static void start_discovery_complete(struct hci_dev *hdev, u8 status)
3313{
3314 BT_DBG("status %d", status);
3315
3316 if (status) {
3317 hci_dev_lock(hdev);
3318 mgmt_start_discovery_failed(hdev, status);
3319 hci_dev_unlock(hdev);
3320 return;
3321 }
3322
3323 hci_dev_lock(hdev);
3324 hci_discovery_set_state(hdev, DISCOVERY_FINDING);
3325 hci_dev_unlock(hdev);
3326
3327 switch (hdev->discovery.type) {
3328 case DISCOV_TYPE_LE:
3329 queue_delayed_work(hdev->workqueue, &hdev->le_scan_disable,
Andre Guedes0d8cc932013-04-30 15:29:31 -03003330 DISCOV_LE_TIMEOUT);
Andre Guedes7c307722013-04-30 15:29:28 -03003331 break;
3332
3333 case DISCOV_TYPE_INTERLEAVED:
3334 queue_delayed_work(hdev->workqueue, &hdev->le_scan_disable,
Andre Guedes0d8cc932013-04-30 15:29:31 -03003335 DISCOV_INTERLEAVED_TIMEOUT);
Andre Guedes7c307722013-04-30 15:29:28 -03003336 break;
3337
3338 case DISCOV_TYPE_BREDR:
3339 break;
3340
3341 default:
3342 BT_ERR("Invalid discovery type %d", hdev->discovery.type);
3343 }
3344}
3345
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003346static int start_discovery(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003347 void *data, u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003348{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003349 struct mgmt_cp_start_discovery *cp = data;
Johan Hedberg14a53662011-04-27 10:29:56 -04003350 struct pending_cmd *cmd;
Andre Guedes7c307722013-04-30 15:29:28 -03003351 struct hci_cp_le_set_scan_param param_cp;
3352 struct hci_cp_le_set_scan_enable enable_cp;
3353 struct hci_cp_inquiry inq_cp;
3354 struct hci_request req;
3355 /* General inquiry access code (GIAC) */
3356 u8 lap[3] = { 0x33, 0x8b, 0x9e };
Johan Hedbergd9483942014-02-23 19:42:24 +02003357 u8 status, own_addr_type;
Johan Hedberg14a53662011-04-27 10:29:56 -04003358 int err;
3359
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003360 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003361
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003362 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003363
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003364 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003365 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003366 MGMT_STATUS_NOT_POWERED);
Johan Hedbergbd2d1332011-11-07 23:13:37 +02003367 goto failed;
3368 }
3369
Andre Guedes642be6c2012-03-21 00:03:37 -03003370 if (test_bit(HCI_PERIODIC_INQ, &hdev->dev_flags)) {
3371 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3372 MGMT_STATUS_BUSY);
3373 goto failed;
3374 }
3375
Johan Hedbergff9ef572012-01-04 14:23:45 +02003376 if (hdev->discovery.state != DISCOVERY_STOPPED) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003377 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003378 MGMT_STATUS_BUSY);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003379 goto failed;
3380 }
3381
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003382 cmd = mgmt_pending_add(sk, MGMT_OP_START_DISCOVERY, hdev, NULL, 0);
Johan Hedberg14a53662011-04-27 10:29:56 -04003383 if (!cmd) {
3384 err = -ENOMEM;
3385 goto failed;
3386 }
3387
Andre Guedes4aab14e2012-02-17 20:39:36 -03003388 hdev->discovery.type = cp->type;
3389
Andre Guedes7c307722013-04-30 15:29:28 -03003390 hci_req_init(&req, hdev);
3391
Andre Guedes4aab14e2012-02-17 20:39:36 -03003392 switch (hdev->discovery.type) {
Andre Guedesf39799f2012-02-17 20:39:35 -03003393 case DISCOV_TYPE_BREDR:
Johan Hedberge6fe7982013-10-02 15:45:22 +03003394 status = mgmt_bredr_support(hdev);
3395 if (status) {
Johan Hedberg04106752013-01-10 14:54:09 +02003396 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003397 status);
Johan Hedberg04106752013-01-10 14:54:09 +02003398 mgmt_pending_remove(cmd);
3399 goto failed;
3400 }
3401
Andre Guedes7c307722013-04-30 15:29:28 -03003402 if (test_bit(HCI_INQUIRY, &hdev->flags)) {
3403 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3404 MGMT_STATUS_BUSY);
3405 mgmt_pending_remove(cmd);
3406 goto failed;
3407 }
3408
3409 hci_inquiry_cache_flush(hdev);
3410
3411 memset(&inq_cp, 0, sizeof(inq_cp));
3412 memcpy(&inq_cp.lap, lap, sizeof(inq_cp.lap));
Andre Guedes0d8cc932013-04-30 15:29:31 -03003413 inq_cp.length = DISCOV_BREDR_INQUIRY_LEN;
Andre Guedes7c307722013-04-30 15:29:28 -03003414 hci_req_add(&req, HCI_OP_INQUIRY, sizeof(inq_cp), &inq_cp);
Andre Guedesf39799f2012-02-17 20:39:35 -03003415 break;
3416
3417 case DISCOV_TYPE_LE:
Andre Guedes7c307722013-04-30 15:29:28 -03003418 case DISCOV_TYPE_INTERLEAVED:
Johan Hedberge6fe7982013-10-02 15:45:22 +03003419 status = mgmt_le_support(hdev);
3420 if (status) {
Johan Hedberg04106752013-01-10 14:54:09 +02003421 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003422 status);
Johan Hedberg04106752013-01-10 14:54:09 +02003423 mgmt_pending_remove(cmd);
3424 goto failed;
3425 }
3426
Andre Guedes7c307722013-04-30 15:29:28 -03003427 if (hdev->discovery.type == DISCOV_TYPE_INTERLEAVED &&
Johan Hedberg56f87902013-10-02 13:43:13 +03003428 !test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
Johan Hedberg04106752013-01-10 14:54:09 +02003429 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3430 MGMT_STATUS_NOT_SUPPORTED);
3431 mgmt_pending_remove(cmd);
3432 goto failed;
3433 }
3434
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003435 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
Andre Guedes7c307722013-04-30 15:29:28 -03003436 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3437 MGMT_STATUS_REJECTED);
3438 mgmt_pending_remove(cmd);
3439 goto failed;
3440 }
3441
3442 if (test_bit(HCI_LE_SCAN, &hdev->dev_flags)) {
3443 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3444 MGMT_STATUS_BUSY);
3445 mgmt_pending_remove(cmd);
3446 goto failed;
3447 }
3448
3449 memset(&param_cp, 0, sizeof(param_cp));
Johan Hedbergd9483942014-02-23 19:42:24 +02003450
Marcel Holtmann94b1fc92014-02-23 20:25:54 -08003451 /* All active scans will be done with either a resolvable
3452 * private address (when privacy feature has been enabled)
3453 * or unresolvable private address.
3454 */
3455 err = hci_update_random_address(&req, true, &own_addr_type);
Johan Hedbergd9483942014-02-23 19:42:24 +02003456 if (err < 0) {
3457 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3458 MGMT_STATUS_FAILED);
3459 mgmt_pending_remove(cmd);
3460 goto failed;
3461 }
3462
Andre Guedes7c307722013-04-30 15:29:28 -03003463 param_cp.type = LE_SCAN_ACTIVE;
Andre Guedes0d8cc932013-04-30 15:29:31 -03003464 param_cp.interval = cpu_to_le16(DISCOV_LE_SCAN_INT);
3465 param_cp.window = cpu_to_le16(DISCOV_LE_SCAN_WIN);
Johan Hedbergd9483942014-02-23 19:42:24 +02003466 param_cp.own_address_type = own_addr_type;
Andre Guedes7c307722013-04-30 15:29:28 -03003467 hci_req_add(&req, HCI_OP_LE_SET_SCAN_PARAM, sizeof(param_cp),
3468 &param_cp);
3469
3470 memset(&enable_cp, 0, sizeof(enable_cp));
3471 enable_cp.enable = LE_SCAN_ENABLE;
3472 enable_cp.filter_dup = LE_SCAN_FILTER_DUP_ENABLE;
3473 hci_req_add(&req, HCI_OP_LE_SET_SCAN_ENABLE, sizeof(enable_cp),
3474 &enable_cp);
Andre Guedes5e0452c2012-02-17 20:39:38 -03003475 break;
3476
Andre Guedesf39799f2012-02-17 20:39:35 -03003477 default:
Johan Hedberg04106752013-01-10 14:54:09 +02003478 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3479 MGMT_STATUS_INVALID_PARAMS);
3480 mgmt_pending_remove(cmd);
3481 goto failed;
Andre Guedesf39799f2012-02-17 20:39:35 -03003482 }
Andre Guedes3fd24152012-02-03 17:48:01 -03003483
Andre Guedes7c307722013-04-30 15:29:28 -03003484 err = hci_req_run(&req, start_discovery_complete);
Johan Hedberg14a53662011-04-27 10:29:56 -04003485 if (err < 0)
3486 mgmt_pending_remove(cmd);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003487 else
3488 hci_discovery_set_state(hdev, DISCOVERY_STARTING);
Johan Hedberg14a53662011-04-27 10:29:56 -04003489
3490failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003491 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003492 return err;
3493}
3494
Andre Guedes1183fdc2013-04-30 15:29:35 -03003495static int mgmt_stop_discovery_failed(struct hci_dev *hdev, u8 status)
3496{
3497 struct pending_cmd *cmd;
3498 int err;
3499
3500 cmd = mgmt_pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
3501 if (!cmd)
3502 return -ENOENT;
3503
3504 err = cmd_complete(cmd->sk, hdev->id, cmd->opcode, mgmt_status(status),
3505 &hdev->discovery.type, sizeof(hdev->discovery.type));
3506 mgmt_pending_remove(cmd);
3507
3508 return err;
3509}
3510
Andre Guedes0e05bba2013-04-30 15:29:33 -03003511static void stop_discovery_complete(struct hci_dev *hdev, u8 status)
3512{
3513 BT_DBG("status %d", status);
3514
3515 hci_dev_lock(hdev);
3516
3517 if (status) {
3518 mgmt_stop_discovery_failed(hdev, status);
3519 goto unlock;
3520 }
3521
3522 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3523
3524unlock:
3525 hci_dev_unlock(hdev);
3526}
3527
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003528static int stop_discovery(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003529 u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003530{
Johan Hedbergd9306502012-02-20 23:25:18 +02003531 struct mgmt_cp_stop_discovery *mgmt_cp = data;
Johan Hedberg14a53662011-04-27 10:29:56 -04003532 struct pending_cmd *cmd;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003533 struct hci_cp_remote_name_req_cancel cp;
3534 struct inquiry_entry *e;
Andre Guedes0e05bba2013-04-30 15:29:33 -03003535 struct hci_request req;
Johan Hedberg14a53662011-04-27 10:29:56 -04003536 int err;
3537
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003538 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003539
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003540 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003541
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003542 if (!hci_discovery_active(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003543 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003544 MGMT_STATUS_REJECTED, &mgmt_cp->type,
3545 sizeof(mgmt_cp->type));
Johan Hedbergd9306502012-02-20 23:25:18 +02003546 goto unlock;
3547 }
3548
3549 if (hdev->discovery.type != mgmt_cp->type) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003550 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003551 MGMT_STATUS_INVALID_PARAMS, &mgmt_cp->type,
3552 sizeof(mgmt_cp->type));
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003553 goto unlock;
Johan Hedbergff9ef572012-01-04 14:23:45 +02003554 }
3555
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003556 cmd = mgmt_pending_add(sk, MGMT_OP_STOP_DISCOVERY, hdev, NULL, 0);
Johan Hedberg14a53662011-04-27 10:29:56 -04003557 if (!cmd) {
3558 err = -ENOMEM;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003559 goto unlock;
Johan Hedberg14a53662011-04-27 10:29:56 -04003560 }
3561
Andre Guedes0e05bba2013-04-30 15:29:33 -03003562 hci_req_init(&req, hdev);
3563
Andre Guedese0d9727e2012-03-20 15:15:36 -03003564 switch (hdev->discovery.state) {
3565 case DISCOVERY_FINDING:
Andre Guedes0e05bba2013-04-30 15:29:33 -03003566 if (test_bit(HCI_INQUIRY, &hdev->flags)) {
3567 hci_req_add(&req, HCI_OP_INQUIRY_CANCEL, 0, NULL);
3568 } else {
3569 cancel_delayed_work(&hdev->le_scan_disable);
3570
Andre Guedesb1efcc22014-02-26 20:21:40 -03003571 hci_req_add_le_scan_disable(&req);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003572 }
Andre Guedesc9ecc482012-03-15 16:52:08 -03003573
Andre Guedese0d9727e2012-03-20 15:15:36 -03003574 break;
3575
3576 case DISCOVERY_RESOLVING:
3577 e = hci_inquiry_cache_lookup_resolve(hdev, BDADDR_ANY,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003578 NAME_PENDING);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003579 if (!e) {
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003580 mgmt_pending_remove(cmd);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003581 err = cmd_complete(sk, hdev->id,
3582 MGMT_OP_STOP_DISCOVERY, 0,
3583 &mgmt_cp->type,
3584 sizeof(mgmt_cp->type));
3585 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3586 goto unlock;
3587 }
3588
3589 bacpy(&cp.bdaddr, &e->data.bdaddr);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003590 hci_req_add(&req, HCI_OP_REMOTE_NAME_REQ_CANCEL, sizeof(cp),
3591 &cp);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003592
3593 break;
3594
3595 default:
3596 BT_DBG("unknown discovery state %u", hdev->discovery.state);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003597
3598 mgmt_pending_remove(cmd);
3599 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
3600 MGMT_STATUS_FAILED, &mgmt_cp->type,
3601 sizeof(mgmt_cp->type));
3602 goto unlock;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003603 }
3604
Andre Guedes0e05bba2013-04-30 15:29:33 -03003605 err = hci_req_run(&req, stop_discovery_complete);
Johan Hedberg14a53662011-04-27 10:29:56 -04003606 if (err < 0)
3607 mgmt_pending_remove(cmd);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003608 else
3609 hci_discovery_set_state(hdev, DISCOVERY_STOPPING);
Johan Hedberg14a53662011-04-27 10:29:56 -04003610
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003611unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003612 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003613 return err;
3614}
3615
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003616static int confirm_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003617 u16 len)
Johan Hedberg561aafb2012-01-04 13:31:59 +02003618{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003619 struct mgmt_cp_confirm_name *cp = data;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003620 struct inquiry_entry *e;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003621 int err;
3622
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003623 BT_DBG("%s", hdev->name);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003624
Johan Hedberg561aafb2012-01-04 13:31:59 +02003625 hci_dev_lock(hdev);
3626
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003627 if (!hci_discovery_active(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003628 err = cmd_status(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003629 MGMT_STATUS_FAILED);
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003630 goto failed;
3631 }
3632
Johan Hedberga198e7b2012-02-17 14:27:06 +02003633 e = hci_inquiry_cache_lookup_unknown(hdev, &cp->addr.bdaddr);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003634 if (!e) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003635 err = cmd_status(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003636 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003637 goto failed;
3638 }
3639
3640 if (cp->name_known) {
3641 e->name_state = NAME_KNOWN;
3642 list_del(&e->list);
3643 } else {
3644 e->name_state = NAME_NEEDED;
Johan Hedberga3d4e202012-01-09 00:53:02 +02003645 hci_inquiry_cache_update_resolve(hdev, e);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003646 }
3647
Johan Hedberge3846622013-01-09 15:29:33 +02003648 err = cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME, 0, &cp->addr,
3649 sizeof(cp->addr));
Johan Hedberg561aafb2012-01-04 13:31:59 +02003650
3651failed:
3652 hci_dev_unlock(hdev);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003653 return err;
3654}
3655
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003656static int block_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003657 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003658{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003659 struct mgmt_cp_block_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003660 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003661 int err;
3662
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003663 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003664
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003665 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg5d0846d2013-01-20 14:27:22 +02003666 return cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE,
3667 MGMT_STATUS_INVALID_PARAMS,
3668 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003669
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003670 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003671
Johan Hedberg88c1fe42012-02-09 15:56:11 +02003672 err = hci_blacklist_add(hdev, &cp->addr.bdaddr, cp->addr.type);
Antti Julku7fbec222011-06-15 12:01:15 +03003673 if (err < 0)
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003674 status = MGMT_STATUS_FAILED;
Antti Julku7fbec222011-06-15 12:01:15 +03003675 else
Szymon Janca6785be2012-12-13 15:11:21 +01003676 status = MGMT_STATUS_SUCCESS;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003677
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003678 err = cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003679 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003680
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003681 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003682
3683 return err;
3684}
3685
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003686static int unblock_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003687 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003688{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003689 struct mgmt_cp_unblock_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003690 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003691 int err;
3692
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003693 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003694
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003695 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg5d0846d2013-01-20 14:27:22 +02003696 return cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE,
3697 MGMT_STATUS_INVALID_PARAMS,
3698 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003699
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003700 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003701
Johan Hedberg88c1fe42012-02-09 15:56:11 +02003702 err = hci_blacklist_del(hdev, &cp->addr.bdaddr, cp->addr.type);
Antti Julku7fbec222011-06-15 12:01:15 +03003703 if (err < 0)
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003704 status = MGMT_STATUS_INVALID_PARAMS;
Antti Julku7fbec222011-06-15 12:01:15 +03003705 else
Szymon Janca6785be2012-12-13 15:11:21 +01003706 status = MGMT_STATUS_SUCCESS;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003707
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003708 err = cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003709 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003710
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003711 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003712
3713 return err;
3714}
3715
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003716static int set_device_id(struct sock *sk, struct hci_dev *hdev, void *data,
3717 u16 len)
3718{
3719 struct mgmt_cp_set_device_id *cp = data;
Johan Hedberg890ea892013-03-15 17:06:52 -05003720 struct hci_request req;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003721 int err;
Szymon Jancc72d4b82012-03-16 16:02:57 +01003722 __u16 source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003723
3724 BT_DBG("%s", hdev->name);
3725
Szymon Jancc72d4b82012-03-16 16:02:57 +01003726 source = __le16_to_cpu(cp->source);
3727
3728 if (source > 0x0002)
3729 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEVICE_ID,
3730 MGMT_STATUS_INVALID_PARAMS);
3731
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003732 hci_dev_lock(hdev);
3733
Szymon Jancc72d4b82012-03-16 16:02:57 +01003734 hdev->devid_source = source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003735 hdev->devid_vendor = __le16_to_cpu(cp->vendor);
3736 hdev->devid_product = __le16_to_cpu(cp->product);
3737 hdev->devid_version = __le16_to_cpu(cp->version);
3738
3739 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_DEVICE_ID, 0, NULL, 0);
3740
Johan Hedberg890ea892013-03-15 17:06:52 -05003741 hci_req_init(&req, hdev);
3742 update_eir(&req);
3743 hci_req_run(&req, NULL);
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003744
3745 hci_dev_unlock(hdev);
3746
3747 return err;
3748}
3749
Johan Hedberg4375f102013-09-25 13:26:10 +03003750static void set_advertising_complete(struct hci_dev *hdev, u8 status)
3751{
3752 struct cmd_lookup match = { NULL, hdev };
3753
3754 if (status) {
3755 u8 mgmt_err = mgmt_status(status);
3756
3757 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev,
3758 cmd_status_rsp, &mgmt_err);
3759 return;
3760 }
3761
3762 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev, settings_rsp,
3763 &match);
3764
3765 new_settings(hdev, match.sk);
3766
3767 if (match.sk)
3768 sock_put(match.sk);
3769}
3770
Marcel Holtmann21b51872013-10-10 09:47:53 -07003771static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
3772 u16 len)
Johan Hedberg4375f102013-09-25 13:26:10 +03003773{
3774 struct mgmt_mode *cp = data;
3775 struct pending_cmd *cmd;
3776 struct hci_request req;
Johan Hedberge6fe7982013-10-02 15:45:22 +03003777 u8 val, enabled, status;
Johan Hedberg4375f102013-09-25 13:26:10 +03003778 int err;
3779
3780 BT_DBG("request for %s", hdev->name);
3781
Johan Hedberge6fe7982013-10-02 15:45:22 +03003782 status = mgmt_le_support(hdev);
3783 if (status)
Johan Hedberg4375f102013-09-25 13:26:10 +03003784 return cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003785 status);
Johan Hedberg4375f102013-09-25 13:26:10 +03003786
3787 if (cp->val != 0x00 && cp->val != 0x01)
3788 return cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3789 MGMT_STATUS_INVALID_PARAMS);
3790
3791 hci_dev_lock(hdev);
3792
3793 val = !!cp->val;
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003794 enabled = test_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedberg4375f102013-09-25 13:26:10 +03003795
Johan Hedbergf74ca9b2013-10-08 15:52:18 +02003796 /* The following conditions are ones which mean that we should
3797 * not do any HCI communication but directly send a mgmt
3798 * response to user space (after toggling the flag if
3799 * necessary).
3800 */
3801 if (!hdev_is_powered(hdev) || val == enabled ||
Marcel Holtmannb145edc2013-10-10 09:47:54 -07003802 hci_conn_num(hdev, LE_LINK) > 0) {
Johan Hedberg4375f102013-09-25 13:26:10 +03003803 bool changed = false;
3804
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003805 if (val != test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
3806 change_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedberg4375f102013-09-25 13:26:10 +03003807 changed = true;
3808 }
3809
3810 err = send_settings_rsp(sk, MGMT_OP_SET_ADVERTISING, hdev);
3811 if (err < 0)
3812 goto unlock;
3813
3814 if (changed)
3815 err = new_settings(hdev, sk);
3816
3817 goto unlock;
3818 }
3819
3820 if (mgmt_pending_find(MGMT_OP_SET_ADVERTISING, hdev) ||
3821 mgmt_pending_find(MGMT_OP_SET_LE, hdev)) {
3822 err = cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3823 MGMT_STATUS_BUSY);
3824 goto unlock;
3825 }
3826
3827 cmd = mgmt_pending_add(sk, MGMT_OP_SET_ADVERTISING, hdev, data, len);
3828 if (!cmd) {
3829 err = -ENOMEM;
3830 goto unlock;
3831 }
3832
3833 hci_req_init(&req, hdev);
3834
Marcel Holtmannbba3aa52013-10-06 02:55:21 -07003835 if (val)
3836 enable_advertising(&req);
3837 else
3838 disable_advertising(&req);
Johan Hedberg4375f102013-09-25 13:26:10 +03003839
3840 err = hci_req_run(&req, set_advertising_complete);
3841 if (err < 0)
3842 mgmt_pending_remove(cmd);
3843
3844unlock:
3845 hci_dev_unlock(hdev);
3846 return err;
3847}
3848
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003849static int set_static_address(struct sock *sk, struct hci_dev *hdev,
3850 void *data, u16 len)
3851{
3852 struct mgmt_cp_set_static_address *cp = data;
3853 int err;
3854
3855 BT_DBG("%s", hdev->name);
3856
Marcel Holtmann62af4442013-10-02 22:10:32 -07003857 if (!lmp_le_capable(hdev))
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003858 return cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
Marcel Holtmann62af4442013-10-02 22:10:32 -07003859 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003860
3861 if (hdev_is_powered(hdev))
3862 return cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
3863 MGMT_STATUS_REJECTED);
3864
3865 if (bacmp(&cp->bdaddr, BDADDR_ANY)) {
3866 if (!bacmp(&cp->bdaddr, BDADDR_NONE))
3867 return cmd_status(sk, hdev->id,
3868 MGMT_OP_SET_STATIC_ADDRESS,
3869 MGMT_STATUS_INVALID_PARAMS);
3870
3871 /* Two most significant bits shall be set */
3872 if ((cp->bdaddr.b[5] & 0xc0) != 0xc0)
3873 return cmd_status(sk, hdev->id,
3874 MGMT_OP_SET_STATIC_ADDRESS,
3875 MGMT_STATUS_INVALID_PARAMS);
3876 }
3877
3878 hci_dev_lock(hdev);
3879
3880 bacpy(&hdev->static_addr, &cp->bdaddr);
3881
3882 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS, 0, NULL, 0);
3883
3884 hci_dev_unlock(hdev);
3885
3886 return err;
3887}
3888
Marcel Holtmann14b49b92013-10-11 08:23:20 -07003889static int set_scan_params(struct sock *sk, struct hci_dev *hdev,
3890 void *data, u16 len)
3891{
3892 struct mgmt_cp_set_scan_params *cp = data;
3893 __u16 interval, window;
3894 int err;
3895
3896 BT_DBG("%s", hdev->name);
3897
3898 if (!lmp_le_capable(hdev))
3899 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3900 MGMT_STATUS_NOT_SUPPORTED);
3901
3902 interval = __le16_to_cpu(cp->interval);
3903
3904 if (interval < 0x0004 || interval > 0x4000)
3905 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3906 MGMT_STATUS_INVALID_PARAMS);
3907
3908 window = __le16_to_cpu(cp->window);
3909
3910 if (window < 0x0004 || window > 0x4000)
3911 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3912 MGMT_STATUS_INVALID_PARAMS);
3913
Marcel Holtmann899e1072013-10-14 09:55:32 -07003914 if (window > interval)
3915 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3916 MGMT_STATUS_INVALID_PARAMS);
3917
Marcel Holtmann14b49b92013-10-11 08:23:20 -07003918 hci_dev_lock(hdev);
3919
3920 hdev->le_scan_interval = interval;
3921 hdev->le_scan_window = window;
3922
3923 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS, 0, NULL, 0);
3924
3925 hci_dev_unlock(hdev);
3926
3927 return err;
3928}
3929
Johan Hedberg33e38b32013-03-15 17:07:05 -05003930static void fast_connectable_complete(struct hci_dev *hdev, u8 status)
3931{
3932 struct pending_cmd *cmd;
3933
3934 BT_DBG("status 0x%02x", status);
3935
3936 hci_dev_lock(hdev);
3937
3938 cmd = mgmt_pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev);
3939 if (!cmd)
3940 goto unlock;
3941
3942 if (status) {
3943 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3944 mgmt_status(status));
3945 } else {
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05003946 struct mgmt_mode *cp = cmd->param;
3947
3948 if (cp->val)
3949 set_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3950 else
3951 clear_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3952
Johan Hedberg33e38b32013-03-15 17:07:05 -05003953 send_settings_rsp(cmd->sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev);
3954 new_settings(hdev, cmd->sk);
3955 }
3956
3957 mgmt_pending_remove(cmd);
3958
3959unlock:
3960 hci_dev_unlock(hdev);
3961}
3962
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003963static int set_fast_connectable(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003964 void *data, u16 len)
Antti Julkuf6422ec2011-06-22 13:11:56 +03003965{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003966 struct mgmt_mode *cp = data;
Johan Hedberg33e38b32013-03-15 17:07:05 -05003967 struct pending_cmd *cmd;
3968 struct hci_request req;
Antti Julkuf6422ec2011-06-22 13:11:56 +03003969 int err;
3970
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003971 BT_DBG("%s", hdev->name);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003972
Johan Hedberg56f87902013-10-02 13:43:13 +03003973 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags) ||
3974 hdev->hci_ver < BLUETOOTH_VER_1_2)
Johan Hedberg33c525c2012-10-24 21:11:58 +03003975 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3976 MGMT_STATUS_NOT_SUPPORTED);
3977
Johan Hedberga7e80f22013-01-09 16:05:19 +02003978 if (cp->val != 0x00 && cp->val != 0x01)
3979 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3980 MGMT_STATUS_INVALID_PARAMS);
3981
Johan Hedberg5400c042012-02-21 16:40:33 +02003982 if (!hdev_is_powered(hdev))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003983 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003984 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5400c042012-02-21 16:40:33 +02003985
3986 if (!test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003987 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003988 MGMT_STATUS_REJECTED);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003989
3990 hci_dev_lock(hdev);
3991
Johan Hedberg05cbf292013-03-15 17:07:07 -05003992 if (mgmt_pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev)) {
3993 err = cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3994 MGMT_STATUS_BUSY);
3995 goto unlock;
3996 }
3997
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05003998 if (!!cp->val == test_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags)) {
3999 err = send_settings_rsp(sk, MGMT_OP_SET_FAST_CONNECTABLE,
4000 hdev);
4001 goto unlock;
4002 }
4003
Johan Hedberg33e38b32013-03-15 17:07:05 -05004004 cmd = mgmt_pending_add(sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev,
4005 data, len);
4006 if (!cmd) {
4007 err = -ENOMEM;
4008 goto unlock;
4009 }
4010
4011 hci_req_init(&req, hdev);
4012
Johan Hedberg406d7802013-03-15 17:07:09 -05004013 write_fast_connectable(&req, cp->val);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004014
4015 err = hci_req_run(&req, fast_connectable_complete);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004016 if (err < 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004017 err = cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004018 MGMT_STATUS_FAILED);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004019 mgmt_pending_remove(cmd);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004020 }
4021
Johan Hedberg33e38b32013-03-15 17:07:05 -05004022unlock:
Antti Julkuf6422ec2011-06-22 13:11:56 +03004023 hci_dev_unlock(hdev);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004024
Antti Julkuf6422ec2011-06-22 13:11:56 +03004025 return err;
4026}
4027
Johan Hedberg67e5a7a2013-10-14 21:15:25 +03004028static void set_bredr_scan(struct hci_request *req)
4029{
4030 struct hci_dev *hdev = req->hdev;
4031 u8 scan = 0;
4032
4033 /* Ensure that fast connectable is disabled. This function will
4034 * not do anything if the page scan parameters are already what
4035 * they should be.
4036 */
4037 write_fast_connectable(req, false);
4038
4039 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
4040 scan |= SCAN_PAGE;
4041 if (test_bit(HCI_DISCOVERABLE, &hdev->dev_flags))
4042 scan |= SCAN_INQUIRY;
4043
4044 if (scan)
4045 hci_req_add(req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
4046}
4047
Johan Hedberg0663ca22013-10-02 13:43:14 +03004048static void set_bredr_complete(struct hci_dev *hdev, u8 status)
4049{
4050 struct pending_cmd *cmd;
4051
4052 BT_DBG("status 0x%02x", status);
4053
4054 hci_dev_lock(hdev);
4055
4056 cmd = mgmt_pending_find(MGMT_OP_SET_BREDR, hdev);
4057 if (!cmd)
4058 goto unlock;
4059
4060 if (status) {
4061 u8 mgmt_err = mgmt_status(status);
4062
4063 /* We need to restore the flag if related HCI commands
4064 * failed.
4065 */
4066 clear_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
4067
4068 cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
4069 } else {
4070 send_settings_rsp(cmd->sk, MGMT_OP_SET_BREDR, hdev);
4071 new_settings(hdev, cmd->sk);
4072 }
4073
4074 mgmt_pending_remove(cmd);
4075
4076unlock:
4077 hci_dev_unlock(hdev);
4078}
4079
4080static int set_bredr(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
4081{
4082 struct mgmt_mode *cp = data;
4083 struct pending_cmd *cmd;
4084 struct hci_request req;
4085 int err;
4086
4087 BT_DBG("request for %s", hdev->name);
4088
4089 if (!lmp_bredr_capable(hdev) || !lmp_le_capable(hdev))
4090 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4091 MGMT_STATUS_NOT_SUPPORTED);
4092
4093 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
4094 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4095 MGMT_STATUS_REJECTED);
4096
4097 if (cp->val != 0x00 && cp->val != 0x01)
4098 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4099 MGMT_STATUS_INVALID_PARAMS);
4100
4101 hci_dev_lock(hdev);
4102
4103 if (cp->val == test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
4104 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4105 goto unlock;
4106 }
4107
4108 if (!hdev_is_powered(hdev)) {
4109 if (!cp->val) {
Johan Hedberg0663ca22013-10-02 13:43:14 +03004110 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
4111 clear_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
4112 clear_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
4113 clear_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
4114 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
4115 }
4116
4117 change_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
4118
4119 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4120 if (err < 0)
4121 goto unlock;
4122
4123 err = new_settings(hdev, sk);
4124 goto unlock;
4125 }
4126
4127 /* Reject disabling when powered on */
4128 if (!cp->val) {
4129 err = cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4130 MGMT_STATUS_REJECTED);
4131 goto unlock;
4132 }
4133
4134 if (mgmt_pending_find(MGMT_OP_SET_BREDR, hdev)) {
4135 err = cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4136 MGMT_STATUS_BUSY);
4137 goto unlock;
4138 }
4139
4140 cmd = mgmt_pending_add(sk, MGMT_OP_SET_BREDR, hdev, data, len);
4141 if (!cmd) {
4142 err = -ENOMEM;
4143 goto unlock;
4144 }
4145
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004146 /* We need to flip the bit already here so that update_adv_data
Johan Hedberg0663ca22013-10-02 13:43:14 +03004147 * generates the correct flags.
4148 */
4149 set_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
4150
4151 hci_req_init(&req, hdev);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004152
4153 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
4154 set_bredr_scan(&req);
4155
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004156 /* Since only the advertising data flags will change, there
4157 * is no need to update the scan response data.
4158 */
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004159 update_adv_data(&req);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004160
Johan Hedberg0663ca22013-10-02 13:43:14 +03004161 err = hci_req_run(&req, set_bredr_complete);
4162 if (err < 0)
4163 mgmt_pending_remove(cmd);
4164
4165unlock:
4166 hci_dev_unlock(hdev);
4167 return err;
4168}
4169
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004170static int set_secure_conn(struct sock *sk, struct hci_dev *hdev,
4171 void *data, u16 len)
4172{
4173 struct mgmt_mode *cp = data;
4174 struct pending_cmd *cmd;
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004175 u8 val, status;
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004176 int err;
4177
4178 BT_DBG("request for %s", hdev->name);
4179
4180 status = mgmt_bredr_support(hdev);
4181 if (status)
4182 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4183 status);
4184
Marcel Holtmann5afeac142014-01-10 02:07:27 -08004185 if (!lmp_sc_capable(hdev) &&
4186 !test_bit(HCI_FORCE_SC, &hdev->dev_flags))
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004187 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4188 MGMT_STATUS_NOT_SUPPORTED);
4189
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004190 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004191 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4192 MGMT_STATUS_INVALID_PARAMS);
4193
4194 hci_dev_lock(hdev);
4195
4196 if (!hdev_is_powered(hdev)) {
4197 bool changed;
4198
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004199 if (cp->val) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004200 changed = !test_and_set_bit(HCI_SC_ENABLED,
4201 &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004202 if (cp->val == 0x02)
4203 set_bit(HCI_SC_ONLY, &hdev->dev_flags);
4204 else
4205 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4206 } else {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004207 changed = test_and_clear_bit(HCI_SC_ENABLED,
4208 &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004209 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4210 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004211
4212 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4213 if (err < 0)
4214 goto failed;
4215
4216 if (changed)
4217 err = new_settings(hdev, sk);
4218
4219 goto failed;
4220 }
4221
4222 if (mgmt_pending_find(MGMT_OP_SET_SECURE_CONN, hdev)) {
4223 err = cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4224 MGMT_STATUS_BUSY);
4225 goto failed;
4226 }
4227
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004228 val = !!cp->val;
4229
4230 if (val == test_bit(HCI_SC_ENABLED, &hdev->dev_flags) &&
4231 (cp->val == 0x02) == test_bit(HCI_SC_ONLY, &hdev->dev_flags)) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004232 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4233 goto failed;
4234 }
4235
4236 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SECURE_CONN, hdev, data, len);
4237 if (!cmd) {
4238 err = -ENOMEM;
4239 goto failed;
4240 }
4241
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004242 err = hci_send_cmd(hdev, HCI_OP_WRITE_SC_SUPPORT, 1, &val);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004243 if (err < 0) {
4244 mgmt_pending_remove(cmd);
4245 goto failed;
4246 }
4247
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004248 if (cp->val == 0x02)
4249 set_bit(HCI_SC_ONLY, &hdev->dev_flags);
4250 else
4251 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4252
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004253failed:
4254 hci_dev_unlock(hdev);
4255 return err;
4256}
4257
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004258static int set_debug_keys(struct sock *sk, struct hci_dev *hdev,
4259 void *data, u16 len)
4260{
4261 struct mgmt_mode *cp = data;
4262 bool changed;
4263 int err;
4264
4265 BT_DBG("request for %s", hdev->name);
4266
4267 if (cp->val != 0x00 && cp->val != 0x01)
4268 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEBUG_KEYS,
4269 MGMT_STATUS_INVALID_PARAMS);
4270
4271 hci_dev_lock(hdev);
4272
4273 if (cp->val)
4274 changed = !test_and_set_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
4275 else
4276 changed = test_and_clear_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
4277
4278 err = send_settings_rsp(sk, MGMT_OP_SET_DEBUG_KEYS, hdev);
4279 if (err < 0)
4280 goto unlock;
4281
4282 if (changed)
4283 err = new_settings(hdev, sk);
4284
4285unlock:
4286 hci_dev_unlock(hdev);
4287 return err;
4288}
4289
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004290static int set_privacy(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4291 u16 len)
4292{
4293 struct mgmt_cp_set_privacy *cp = cp_data;
4294 bool changed;
4295 int err;
4296
4297 BT_DBG("request for %s", hdev->name);
4298
4299 if (!lmp_le_capable(hdev))
4300 return cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4301 MGMT_STATUS_NOT_SUPPORTED);
4302
4303 if (cp->privacy != 0x00 && cp->privacy != 0x01)
4304 return cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4305 MGMT_STATUS_INVALID_PARAMS);
4306
4307 if (hdev_is_powered(hdev))
4308 return cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4309 MGMT_STATUS_REJECTED);
4310
4311 hci_dev_lock(hdev);
4312
Johan Hedbergc21c0ea2014-02-24 11:10:30 +02004313 /* If user space supports this command it is also expected to
4314 * handle IRKs. Therefore, set the HCI_RPA_RESOLVING flag.
4315 */
4316 set_bit(HCI_RPA_RESOLVING, &hdev->dev_flags);
4317
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004318 if (cp->privacy) {
4319 changed = !test_and_set_bit(HCI_PRIVACY, &hdev->dev_flags);
4320 memcpy(hdev->irk, cp->irk, sizeof(hdev->irk));
4321 set_bit(HCI_RPA_EXPIRED, &hdev->dev_flags);
4322 } else {
4323 changed = test_and_clear_bit(HCI_PRIVACY, &hdev->dev_flags);
4324 memset(hdev->irk, 0, sizeof(hdev->irk));
4325 clear_bit(HCI_RPA_EXPIRED, &hdev->dev_flags);
4326 }
4327
4328 err = send_settings_rsp(sk, MGMT_OP_SET_PRIVACY, hdev);
4329 if (err < 0)
4330 goto unlock;
4331
4332 if (changed)
4333 err = new_settings(hdev, sk);
4334
4335unlock:
4336 hci_dev_unlock(hdev);
4337 return err;
4338}
4339
Johan Hedberg41edf162014-02-18 10:19:35 +02004340static bool irk_is_valid(struct mgmt_irk_info *irk)
4341{
4342 switch (irk->addr.type) {
4343 case BDADDR_LE_PUBLIC:
4344 return true;
4345
4346 case BDADDR_LE_RANDOM:
4347 /* Two most significant bits shall be set */
4348 if ((irk->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4349 return false;
4350 return true;
4351 }
4352
4353 return false;
4354}
4355
4356static int load_irks(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4357 u16 len)
4358{
4359 struct mgmt_cp_load_irks *cp = cp_data;
4360 u16 irk_count, expected_len;
4361 int i, err;
4362
4363 BT_DBG("request for %s", hdev->name);
4364
4365 if (!lmp_le_capable(hdev))
4366 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4367 MGMT_STATUS_NOT_SUPPORTED);
4368
4369 irk_count = __le16_to_cpu(cp->irk_count);
4370
4371 expected_len = sizeof(*cp) + irk_count * sizeof(struct mgmt_irk_info);
4372 if (expected_len != len) {
4373 BT_ERR("load_irks: expected %u bytes, got %u bytes",
4374 len, expected_len);
4375 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4376 MGMT_STATUS_INVALID_PARAMS);
4377 }
4378
4379 BT_DBG("%s irk_count %u", hdev->name, irk_count);
4380
4381 for (i = 0; i < irk_count; i++) {
4382 struct mgmt_irk_info *key = &cp->irks[i];
4383
4384 if (!irk_is_valid(key))
4385 return cmd_status(sk, hdev->id,
4386 MGMT_OP_LOAD_IRKS,
4387 MGMT_STATUS_INVALID_PARAMS);
4388 }
4389
4390 hci_dev_lock(hdev);
4391
4392 hci_smp_irks_clear(hdev);
4393
4394 for (i = 0; i < irk_count; i++) {
4395 struct mgmt_irk_info *irk = &cp->irks[i];
4396 u8 addr_type;
4397
4398 if (irk->addr.type == BDADDR_LE_PUBLIC)
4399 addr_type = ADDR_LE_DEV_PUBLIC;
4400 else
4401 addr_type = ADDR_LE_DEV_RANDOM;
4402
4403 hci_add_irk(hdev, &irk->addr.bdaddr, addr_type, irk->val,
4404 BDADDR_ANY);
4405 }
4406
4407 set_bit(HCI_RPA_RESOLVING, &hdev->dev_flags);
4408
4409 err = cmd_complete(sk, hdev->id, MGMT_OP_LOAD_IRKS, 0, NULL, 0);
4410
4411 hci_dev_unlock(hdev);
4412
4413 return err;
4414}
4415
Johan Hedberg3f706b72013-01-20 14:27:16 +02004416static bool ltk_is_valid(struct mgmt_ltk_info *key)
4417{
4418 if (key->master != 0x00 && key->master != 0x01)
4419 return false;
Marcel Holtmann490cb0b2014-02-16 12:59:05 -08004420
4421 switch (key->addr.type) {
4422 case BDADDR_LE_PUBLIC:
4423 return true;
4424
4425 case BDADDR_LE_RANDOM:
4426 /* Two most significant bits shall be set */
4427 if ((key->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4428 return false;
4429 return true;
4430 }
4431
4432 return false;
Johan Hedberg3f706b72013-01-20 14:27:16 +02004433}
4434
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004435static int load_long_term_keys(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004436 void *cp_data, u16 len)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004437{
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004438 struct mgmt_cp_load_long_term_keys *cp = cp_data;
4439 u16 key_count, expected_len;
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004440 int i, err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004441
Marcel Holtmanncf99ba12013-10-02 21:16:08 -07004442 BT_DBG("request for %s", hdev->name);
4443
4444 if (!lmp_le_capable(hdev))
4445 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4446 MGMT_STATUS_NOT_SUPPORTED);
4447
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004448 key_count = __le16_to_cpu(cp->key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004449
4450 expected_len = sizeof(*cp) + key_count *
4451 sizeof(struct mgmt_ltk_info);
4452 if (expected_len != len) {
4453 BT_ERR("load_keys: expected %u bytes, got %u bytes",
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004454 len, expected_len);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004455 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
Johan Hedberge57e6192013-01-20 14:27:14 +02004456 MGMT_STATUS_INVALID_PARAMS);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004457 }
4458
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004459 BT_DBG("%s key_count %u", hdev->name, key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004460
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004461 for (i = 0; i < key_count; i++) {
4462 struct mgmt_ltk_info *key = &cp->keys[i];
4463
Johan Hedberg3f706b72013-01-20 14:27:16 +02004464 if (!ltk_is_valid(key))
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004465 return cmd_status(sk, hdev->id,
4466 MGMT_OP_LOAD_LONG_TERM_KEYS,
4467 MGMT_STATUS_INVALID_PARAMS);
4468 }
4469
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004470 hci_dev_lock(hdev);
4471
4472 hci_smp_ltks_clear(hdev);
4473
4474 for (i = 0; i < key_count; i++) {
4475 struct mgmt_ltk_info *key = &cp->keys[i];
Marcel Holtmann79d95a12013-10-13 03:57:38 -07004476 u8 type, addr_type;
4477
4478 if (key->addr.type == BDADDR_LE_PUBLIC)
4479 addr_type = ADDR_LE_DEV_PUBLIC;
4480 else
4481 addr_type = ADDR_LE_DEV_RANDOM;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004482
4483 if (key->master)
4484 type = HCI_SMP_LTK;
4485 else
4486 type = HCI_SMP_LTK_SLAVE;
4487
Johan Hedberg35d70272014-02-19 14:57:47 +02004488 hci_add_ltk(hdev, &key->addr.bdaddr, addr_type, type,
4489 key->type, key->val, key->enc_size, key->ediv,
4490 key->rand);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004491 }
4492
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004493 err = cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS, 0,
4494 NULL, 0);
4495
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004496 hci_dev_unlock(hdev);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004497
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004498 return err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004499}
4500
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +02004501static const struct mgmt_handler {
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004502 int (*func) (struct sock *sk, struct hci_dev *hdev, void *data,
4503 u16 data_len);
Johan Hedbergbe22b542012-03-01 22:24:41 +02004504 bool var_len;
4505 size_t data_len;
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004506} mgmt_handlers[] = {
4507 { NULL }, /* 0x0000 (no command) */
Johan Hedbergbe22b542012-03-01 22:24:41 +02004508 { read_version, false, MGMT_READ_VERSION_SIZE },
4509 { read_commands, false, MGMT_READ_COMMANDS_SIZE },
4510 { read_index_list, false, MGMT_READ_INDEX_LIST_SIZE },
4511 { read_controller_info, false, MGMT_READ_INFO_SIZE },
4512 { set_powered, false, MGMT_SETTING_SIZE },
4513 { set_discoverable, false, MGMT_SET_DISCOVERABLE_SIZE },
4514 { set_connectable, false, MGMT_SETTING_SIZE },
4515 { set_fast_connectable, false, MGMT_SETTING_SIZE },
4516 { set_pairable, false, MGMT_SETTING_SIZE },
4517 { set_link_security, false, MGMT_SETTING_SIZE },
4518 { set_ssp, false, MGMT_SETTING_SIZE },
4519 { set_hs, false, MGMT_SETTING_SIZE },
4520 { set_le, false, MGMT_SETTING_SIZE },
4521 { set_dev_class, false, MGMT_SET_DEV_CLASS_SIZE },
4522 { set_local_name, false, MGMT_SET_LOCAL_NAME_SIZE },
4523 { add_uuid, false, MGMT_ADD_UUID_SIZE },
4524 { remove_uuid, false, MGMT_REMOVE_UUID_SIZE },
4525 { load_link_keys, true, MGMT_LOAD_LINK_KEYS_SIZE },
4526 { load_long_term_keys, true, MGMT_LOAD_LONG_TERM_KEYS_SIZE },
4527 { disconnect, false, MGMT_DISCONNECT_SIZE },
4528 { get_connections, false, MGMT_GET_CONNECTIONS_SIZE },
4529 { pin_code_reply, false, MGMT_PIN_CODE_REPLY_SIZE },
4530 { pin_code_neg_reply, false, MGMT_PIN_CODE_NEG_REPLY_SIZE },
4531 { set_io_capability, false, MGMT_SET_IO_CAPABILITY_SIZE },
4532 { pair_device, false, MGMT_PAIR_DEVICE_SIZE },
4533 { cancel_pair_device, false, MGMT_CANCEL_PAIR_DEVICE_SIZE },
4534 { unpair_device, false, MGMT_UNPAIR_DEVICE_SIZE },
4535 { user_confirm_reply, false, MGMT_USER_CONFIRM_REPLY_SIZE },
4536 { user_confirm_neg_reply, false, MGMT_USER_CONFIRM_NEG_REPLY_SIZE },
4537 { user_passkey_reply, false, MGMT_USER_PASSKEY_REPLY_SIZE },
4538 { user_passkey_neg_reply, false, MGMT_USER_PASSKEY_NEG_REPLY_SIZE },
4539 { read_local_oob_data, false, MGMT_READ_LOCAL_OOB_DATA_SIZE },
Marcel Holtmannec109112014-01-10 02:07:30 -08004540 { add_remote_oob_data, true, MGMT_ADD_REMOTE_OOB_DATA_SIZE },
Johan Hedbergbe22b542012-03-01 22:24:41 +02004541 { remove_remote_oob_data, false, MGMT_REMOVE_REMOTE_OOB_DATA_SIZE },
4542 { start_discovery, false, MGMT_START_DISCOVERY_SIZE },
4543 { stop_discovery, false, MGMT_STOP_DISCOVERY_SIZE },
4544 { confirm_name, false, MGMT_CONFIRM_NAME_SIZE },
4545 { block_device, false, MGMT_BLOCK_DEVICE_SIZE },
4546 { unblock_device, false, MGMT_UNBLOCK_DEVICE_SIZE },
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07004547 { set_device_id, false, MGMT_SET_DEVICE_ID_SIZE },
Johan Hedberg4375f102013-09-25 13:26:10 +03004548 { set_advertising, false, MGMT_SETTING_SIZE },
Johan Hedberg0663ca22013-10-02 13:43:14 +03004549 { set_bredr, false, MGMT_SETTING_SIZE },
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004550 { set_static_address, false, MGMT_SET_STATIC_ADDRESS_SIZE },
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004551 { set_scan_params, false, MGMT_SET_SCAN_PARAMS_SIZE },
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004552 { set_secure_conn, false, MGMT_SETTING_SIZE },
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004553 { set_debug_keys, false, MGMT_SETTING_SIZE },
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004554 { set_privacy, false, MGMT_SET_PRIVACY_SIZE },
Johan Hedberg41edf162014-02-18 10:19:35 +02004555 { load_irks, true, MGMT_LOAD_IRKS_SIZE },
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004556};
4557
4558
Johan Hedberg03811012010-12-08 00:21:06 +02004559int mgmt_control(struct sock *sk, struct msghdr *msg, size_t msglen)
4560{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004561 void *buf;
4562 u8 *cp;
Johan Hedberg03811012010-12-08 00:21:06 +02004563 struct mgmt_hdr *hdr;
Szymon Janc4e51eae2011-02-25 19:05:48 +01004564 u16 opcode, index, len;
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004565 struct hci_dev *hdev = NULL;
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +02004566 const struct mgmt_handler *handler;
Johan Hedberg03811012010-12-08 00:21:06 +02004567 int err;
4568
4569 BT_DBG("got %zu bytes", msglen);
4570
4571 if (msglen < sizeof(*hdr))
4572 return -EINVAL;
4573
Gustavo F. Padovane63a15e2011-04-04 18:56:53 -03004574 buf = kmalloc(msglen, GFP_KERNEL);
Johan Hedberg03811012010-12-08 00:21:06 +02004575 if (!buf)
4576 return -ENOMEM;
4577
4578 if (memcpy_fromiovec(buf, msg->msg_iov, msglen)) {
4579 err = -EFAULT;
4580 goto done;
4581 }
4582
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004583 hdr = buf;
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004584 opcode = __le16_to_cpu(hdr->opcode);
4585 index = __le16_to_cpu(hdr->index);
4586 len = __le16_to_cpu(hdr->len);
Johan Hedberg03811012010-12-08 00:21:06 +02004587
4588 if (len != msglen - sizeof(*hdr)) {
4589 err = -EINVAL;
4590 goto done;
4591 }
4592
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004593 if (index != MGMT_INDEX_NONE) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004594 hdev = hci_dev_get(index);
4595 if (!hdev) {
4596 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004597 MGMT_STATUS_INVALID_INDEX);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004598 goto done;
4599 }
Marcel Holtmann0736cfa2013-08-26 21:40:51 -07004600
Johan Hedbergcebf4cf2013-10-10 18:06:04 +02004601 if (test_bit(HCI_SETUP, &hdev->dev_flags) ||
4602 test_bit(HCI_USER_CHANNEL, &hdev->dev_flags)) {
Marcel Holtmann0736cfa2013-08-26 21:40:51 -07004603 err = cmd_status(sk, index, opcode,
4604 MGMT_STATUS_INVALID_INDEX);
4605 goto done;
4606 }
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004607 }
4608
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004609 if (opcode >= ARRAY_SIZE(mgmt_handlers) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004610 mgmt_handlers[opcode].func == NULL) {
Johan Hedberg03811012010-12-08 00:21:06 +02004611 BT_DBG("Unknown op %u", opcode);
Johan Hedbergca69b792011-11-11 18:10:00 +02004612 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004613 MGMT_STATUS_UNKNOWN_COMMAND);
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004614 goto done;
Johan Hedberg03811012010-12-08 00:21:06 +02004615 }
4616
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004617 if ((hdev && opcode < MGMT_OP_READ_INFO) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004618 (!hdev && opcode >= MGMT_OP_READ_INFO)) {
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004619 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004620 MGMT_STATUS_INVALID_INDEX);
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004621 goto done;
4622 }
4623
Johan Hedbergbe22b542012-03-01 22:24:41 +02004624 handler = &mgmt_handlers[opcode];
4625
4626 if ((handler->var_len && len < handler->data_len) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004627 (!handler->var_len && len != handler->data_len)) {
Johan Hedbergbe22b542012-03-01 22:24:41 +02004628 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004629 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergbe22b542012-03-01 22:24:41 +02004630 goto done;
4631 }
4632
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004633 if (hdev)
4634 mgmt_init_hdev(sk, hdev);
4635
4636 cp = buf + sizeof(*hdr);
4637
Johan Hedbergbe22b542012-03-01 22:24:41 +02004638 err = handler->func(sk, hdev, cp, len);
Johan Hedberge41d8b42010-12-13 21:07:03 +02004639 if (err < 0)
4640 goto done;
4641
Johan Hedberg03811012010-12-08 00:21:06 +02004642 err = msglen;
4643
4644done:
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004645 if (hdev)
4646 hci_dev_put(hdev);
4647
Johan Hedberg03811012010-12-08 00:21:06 +02004648 kfree(buf);
4649 return err;
4650}
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004651
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004652void mgmt_index_added(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004653{
Marcel Holtmann1514b892013-10-06 08:25:01 -07004654 if (hdev->dev_type != HCI_BREDR)
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004655 return;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03004656
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004657 mgmt_event(MGMT_EV_INDEX_ADDED, hdev, NULL, 0, NULL);
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004658}
4659
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004660void mgmt_index_removed(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004661{
Johan Hedberg5f159032012-03-02 03:13:19 +02004662 u8 status = MGMT_STATUS_INVALID_INDEX;
Johan Hedbergb24752f2011-11-03 14:40:33 +02004663
Marcel Holtmann1514b892013-10-06 08:25:01 -07004664 if (hdev->dev_type != HCI_BREDR)
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004665 return;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03004666
Johan Hedberg744cf192011-11-08 20:40:14 +02004667 mgmt_pending_foreach(0, hdev, cmd_status_rsp, &status);
Johan Hedbergb24752f2011-11-03 14:40:33 +02004668
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004669 mgmt_event(MGMT_EV_INDEX_REMOVED, hdev, NULL, 0, NULL);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02004670}
4671
Johan Hedberg229ab392013-03-15 17:06:53 -05004672static void powered_complete(struct hci_dev *hdev, u8 status)
4673{
4674 struct cmd_lookup match = { NULL, hdev };
4675
4676 BT_DBG("status 0x%02x", status);
4677
4678 hci_dev_lock(hdev);
4679
4680 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
4681
4682 new_settings(hdev, match.sk);
4683
4684 hci_dev_unlock(hdev);
4685
4686 if (match.sk)
4687 sock_put(match.sk);
4688}
4689
Johan Hedberg70da6242013-03-15 17:06:51 -05004690static int powered_update_hci(struct hci_dev *hdev)
4691{
Johan Hedberg890ea892013-03-15 17:06:52 -05004692 struct hci_request req;
Johan Hedberg70da6242013-03-15 17:06:51 -05004693 u8 link_sec;
4694
Johan Hedberg890ea892013-03-15 17:06:52 -05004695 hci_req_init(&req, hdev);
4696
Johan Hedberg70da6242013-03-15 17:06:51 -05004697 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags) &&
4698 !lmp_host_ssp_capable(hdev)) {
4699 u8 ssp = 1;
4700
Johan Hedberg890ea892013-03-15 17:06:52 -05004701 hci_req_add(&req, HCI_OP_WRITE_SSP_MODE, 1, &ssp);
Johan Hedberg70da6242013-03-15 17:06:51 -05004702 }
4703
Johan Hedbergc73eee92013-04-19 18:35:21 +03004704 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags) &&
4705 lmp_bredr_capable(hdev)) {
Johan Hedberg70da6242013-03-15 17:06:51 -05004706 struct hci_cp_write_le_host_supported cp;
4707
4708 cp.le = 1;
4709 cp.simul = lmp_le_br_capable(hdev);
4710
4711 /* Check first if we already have the right
4712 * host state (host features set)
4713 */
4714 if (cp.le != lmp_host_le_capable(hdev) ||
4715 cp.simul != lmp_host_le_br_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05004716 hci_req_add(&req, HCI_OP_WRITE_LE_HOST_SUPPORTED,
4717 sizeof(cp), &cp);
Johan Hedberg70da6242013-03-15 17:06:51 -05004718 }
4719
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004720 if (lmp_le_capable(hdev)) {
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07004721 /* Make sure the controller has a good default for
4722 * advertising data. This also applies to the case
4723 * where BR/EDR was toggled during the AUTO_OFF phase.
4724 */
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004725 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004726 update_adv_data(&req);
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004727 update_scan_rsp_data(&req);
4728 }
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07004729
Marcel Holtmannbba3aa52013-10-06 02:55:21 -07004730 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
4731 enable_advertising(&req);
Johan Hedbergeeca6f82013-09-25 13:26:09 +03004732 }
4733
Johan Hedberg70da6242013-03-15 17:06:51 -05004734 link_sec = test_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
4735 if (link_sec != test_bit(HCI_AUTH, &hdev->flags))
Johan Hedberg890ea892013-03-15 17:06:52 -05004736 hci_req_add(&req, HCI_OP_WRITE_AUTH_ENABLE,
4737 sizeof(link_sec), &link_sec);
Johan Hedberg70da6242013-03-15 17:06:51 -05004738
4739 if (lmp_bredr_capable(hdev)) {
Johan Hedberg56f87902013-10-02 13:43:13 +03004740 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
4741 set_bredr_scan(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05004742 update_class(&req);
Johan Hedberg13928972013-03-15 17:07:00 -05004743 update_name(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05004744 update_eir(&req);
Johan Hedberg70da6242013-03-15 17:06:51 -05004745 }
4746
Johan Hedberg229ab392013-03-15 17:06:53 -05004747 return hci_req_run(&req, powered_complete);
Johan Hedberg70da6242013-03-15 17:06:51 -05004748}
4749
Johan Hedberg744cf192011-11-08 20:40:14 +02004750int mgmt_powered(struct hci_dev *hdev, u8 powered)
Johan Hedberg5add6af2010-12-16 10:00:37 +02004751{
Johan Hedberg76a7f3a2012-02-17 00:34:40 +02004752 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg229ab392013-03-15 17:06:53 -05004753 u8 status_not_powered = MGMT_STATUS_NOT_POWERED;
4754 u8 zero_cod[] = { 0, 0, 0 };
Johan Hedberg7bb895d2012-02-17 01:20:00 +02004755 int err;
Johan Hedberg5add6af2010-12-16 10:00:37 +02004756
Johan Hedberg5e5282b2012-02-21 16:01:30 +02004757 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
4758 return 0;
4759
Johan Hedberg5e5282b2012-02-21 16:01:30 +02004760 if (powered) {
Johan Hedberg229ab392013-03-15 17:06:53 -05004761 if (powered_update_hci(hdev) == 0)
4762 return 0;
Johan Hedbergfe038882013-01-16 16:15:34 +02004763
Johan Hedberg229ab392013-03-15 17:06:53 -05004764 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp,
4765 &match);
4766 goto new_settings;
Johan Hedbergb24752f2011-11-03 14:40:33 +02004767 }
4768
Johan Hedberg229ab392013-03-15 17:06:53 -05004769 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
4770 mgmt_pending_foreach(0, hdev, cmd_status_rsp, &status_not_powered);
4771
4772 if (memcmp(hdev->dev_class, zero_cod, sizeof(zero_cod)) != 0)
4773 mgmt_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev,
4774 zero_cod, sizeof(zero_cod), NULL);
4775
4776new_settings:
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02004777 err = new_settings(hdev, match.sk);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02004778
4779 if (match.sk)
4780 sock_put(match.sk);
4781
Johan Hedberg7bb895d2012-02-17 01:20:00 +02004782 return err;
Johan Hedberg5add6af2010-12-16 10:00:37 +02004783}
Johan Hedberg73f22f62010-12-29 16:00:25 +02004784
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004785void mgmt_set_powered_failed(struct hci_dev *hdev, int err)
Johan Hedberg96570ff2013-05-29 09:51:29 +03004786{
4787 struct pending_cmd *cmd;
4788 u8 status;
4789
4790 cmd = mgmt_pending_find(MGMT_OP_SET_POWERED, hdev);
4791 if (!cmd)
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004792 return;
Johan Hedberg96570ff2013-05-29 09:51:29 +03004793
4794 if (err == -ERFKILL)
4795 status = MGMT_STATUS_RFKILLED;
4796 else
4797 status = MGMT_STATUS_FAILED;
4798
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004799 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_POWERED, status);
Johan Hedberg96570ff2013-05-29 09:51:29 +03004800
4801 mgmt_pending_remove(cmd);
Johan Hedberg96570ff2013-05-29 09:51:29 +03004802}
4803
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004804void mgmt_discoverable_timeout(struct hci_dev *hdev)
4805{
4806 struct hci_request req;
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004807
4808 hci_dev_lock(hdev);
4809
4810 /* When discoverable timeout triggers, then just make sure
4811 * the limited discoverable flag is cleared. Even in the case
4812 * of a timeout triggered from general discoverable, it is
4813 * safe to unconditionally clear the flag.
4814 */
4815 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004816 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004817
4818 hci_req_init(&req, hdev);
Johan Hedberg4b580612013-10-19 23:38:21 +03004819 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
4820 u8 scan = SCAN_PAGE;
4821 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE,
4822 sizeof(scan), &scan);
4823 }
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004824 update_class(&req);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004825 update_adv_data(&req);
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004826 hci_req_run(&req, NULL);
4827
4828 hdev->discov_timeout = 0;
4829
Johan Hedberg9a43e252013-10-20 19:00:07 +03004830 new_settings(hdev, NULL);
4831
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004832 hci_dev_unlock(hdev);
4833}
4834
Marcel Holtmann86a75642013-10-15 06:33:54 -07004835void mgmt_discoverable(struct hci_dev *hdev, u8 discoverable)
Johan Hedberg73f22f62010-12-29 16:00:25 +02004836{
Marcel Holtmann86a75642013-10-15 06:33:54 -07004837 bool changed;
Johan Hedberg73f22f62010-12-29 16:00:25 +02004838
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03004839 /* Nothing needed here if there's a pending command since that
4840 * commands request completion callback takes care of everything
4841 * necessary.
4842 */
4843 if (mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev))
Marcel Holtmann86a75642013-10-15 06:33:54 -07004844 return;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03004845
Johan Hedbergbd107992014-02-24 14:52:19 +02004846 /* Powering off may clear the scan mode - don't let that interfere */
4847 if (!discoverable && mgmt_pending_find(MGMT_OP_SET_POWERED, hdev))
4848 return;
4849
Johan Hedberg9a43e252013-10-20 19:00:07 +03004850 if (discoverable) {
Marcel Holtmann86a75642013-10-15 06:33:54 -07004851 changed = !test_and_set_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004852 } else {
4853 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Marcel Holtmann86a75642013-10-15 06:33:54 -07004854 changed = test_and_clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004855 }
Johan Hedberg73f22f62010-12-29 16:00:25 +02004856
Johan Hedberg9a43e252013-10-20 19:00:07 +03004857 if (changed) {
4858 struct hci_request req;
4859
4860 /* In case this change in discoverable was triggered by
4861 * a disabling of connectable there could be a need to
4862 * update the advertising flags.
4863 */
4864 hci_req_init(&req, hdev);
4865 update_adv_data(&req);
4866 hci_req_run(&req, NULL);
4867
Marcel Holtmann86a75642013-10-15 06:33:54 -07004868 new_settings(hdev, NULL);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004869 }
Johan Hedberg73f22f62010-12-29 16:00:25 +02004870}
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004871
Marcel Holtmanna3309162013-10-15 06:33:55 -07004872void mgmt_connectable(struct hci_dev *hdev, u8 connectable)
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004873{
Marcel Holtmanna3309162013-10-15 06:33:55 -07004874 bool changed;
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004875
Johan Hedbergd7b856f2013-10-14 16:20:04 +03004876 /* Nothing needed here if there's a pending command since that
4877 * commands request completion callback takes care of everything
4878 * necessary.
4879 */
4880 if (mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev))
Marcel Holtmanna3309162013-10-15 06:33:55 -07004881 return;
Johan Hedbergd7b856f2013-10-14 16:20:04 +03004882
Johan Hedbergce3f24c2014-02-24 14:52:20 +02004883 /* Powering off may clear the scan mode - don't let that interfere */
4884 if (!connectable && mgmt_pending_find(MGMT_OP_SET_POWERED, hdev))
4885 return;
4886
Marcel Holtmanna3309162013-10-15 06:33:55 -07004887 if (connectable)
4888 changed = !test_and_set_bit(HCI_CONNECTABLE, &hdev->dev_flags);
4889 else
4890 changed = test_and_clear_bit(HCI_CONNECTABLE, &hdev->dev_flags);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004891
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02004892 if (changed)
Marcel Holtmanna3309162013-10-15 06:33:55 -07004893 new_settings(hdev, NULL);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004894}
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004895
Johan Hedberg778b2352014-02-24 14:52:17 +02004896void mgmt_advertising(struct hci_dev *hdev, u8 advertising)
4897{
Johan Hedberg7c4cfab2014-02-24 14:52:21 +02004898 /* Powering off may stop advertising - don't let that interfere */
4899 if (!advertising && mgmt_pending_find(MGMT_OP_SET_POWERED, hdev))
4900 return;
4901
Johan Hedberg778b2352014-02-24 14:52:17 +02004902 if (advertising)
4903 set_bit(HCI_ADVERTISING, &hdev->dev_flags);
4904 else
4905 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
4906}
4907
Marcel Holtmann4796e8a2013-10-15 06:33:56 -07004908void mgmt_write_scan_failed(struct hci_dev *hdev, u8 scan, u8 status)
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004909{
Johan Hedbergca69b792011-11-11 18:10:00 +02004910 u8 mgmt_err = mgmt_status(status);
4911
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004912 if (scan & SCAN_PAGE)
Johan Hedberg744cf192011-11-08 20:40:14 +02004913 mgmt_pending_foreach(MGMT_OP_SET_CONNECTABLE, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004914 cmd_status_rsp, &mgmt_err);
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004915
4916 if (scan & SCAN_INQUIRY)
Johan Hedberg744cf192011-11-08 20:40:14 +02004917 mgmt_pending_foreach(MGMT_OP_SET_DISCOVERABLE, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004918 cmd_status_rsp, &mgmt_err);
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004919}
4920
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07004921void mgmt_new_link_key(struct hci_dev *hdev, struct link_key *key,
4922 bool persistent)
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004923{
Johan Hedberg86742e12011-11-07 23:13:38 +02004924 struct mgmt_ev_new_link_key ev;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004925
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004926 memset(&ev, 0, sizeof(ev));
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004927
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004928 ev.store_hint = persistent;
Johan Hedbergd753fdc2012-02-17 14:06:34 +02004929 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004930 ev.key.addr.type = BDADDR_BREDR;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004931 ev.key.type = key->type;
Andrei Emeltchenko9b3b4462012-05-23 11:31:20 +03004932 memcpy(ev.key.val, key->val, HCI_LINK_KEY_SIZE);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004933 ev.key.pin_len = key->pin_len;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004934
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07004935 mgmt_event(MGMT_EV_NEW_LINK_KEY, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004936}
Johan Hedbergf7520542011-01-20 12:34:39 +02004937
Johan Hedbergba74b662014-02-19 14:57:45 +02004938void mgmt_new_ltk(struct hci_dev *hdev, struct smp_ltk *key)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004939{
4940 struct mgmt_ev_new_long_term_key ev;
4941
4942 memset(&ev, 0, sizeof(ev));
4943
Marcel Holtmann5192d302014-02-19 17:11:58 -08004944 /* Devices using resolvable or non-resolvable random addresses
4945 * without providing an indentity resolving key don't require
4946 * to store long term keys. Their addresses will change the
4947 * next time around.
4948 *
4949 * Only when a remote device provides an identity address
4950 * make sure the long term key is stored. If the remote
4951 * identity is known, the long term keys are internally
4952 * mapped to the identity address. So allow static random
4953 * and public addresses here.
4954 */
Johan Hedbergba74b662014-02-19 14:57:45 +02004955 if (key->bdaddr_type == ADDR_LE_DEV_RANDOM &&
4956 (key->bdaddr.b[5] & 0xc0) != 0xc0)
4957 ev.store_hint = 0x00;
4958 else
4959 ev.store_hint = 0x01;
4960
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004961 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004962 ev.key.addr.type = link_to_bdaddr(LE_LINK, key->bdaddr_type);
Marcel Holtmannd40f3ee2014-01-31 18:42:17 -08004963 ev.key.type = key->authenticated;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004964 ev.key.enc_size = key->enc_size;
4965 ev.key.ediv = key->ediv;
4966
4967 if (key->type == HCI_SMP_LTK)
4968 ev.key.master = 1;
4969
4970 memcpy(ev.key.rand, key->rand, sizeof(key->rand));
4971 memcpy(ev.key.val, key->val, sizeof(key->val));
4972
Marcel Holtmann083368f2013-10-15 14:26:29 -07004973 mgmt_event(MGMT_EV_NEW_LONG_TERM_KEY, hdev, &ev, sizeof(ev), NULL);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004974}
4975
Johan Hedberg95fbac82014-02-19 15:18:31 +02004976void mgmt_new_irk(struct hci_dev *hdev, struct smp_irk *irk)
4977{
4978 struct mgmt_ev_new_irk ev;
4979
4980 memset(&ev, 0, sizeof(ev));
4981
Marcel Holtmannbab6d1e2014-02-19 11:51:54 -08004982 /* For identity resolving keys from devices that are already
4983 * using a public address or static random address, do not
4984 * ask for storing this key. The identity resolving key really
4985 * is only mandatory for devices using resovlable random
4986 * addresses.
4987 *
4988 * Storing all identity resolving keys has the downside that
4989 * they will be also loaded on next boot of they system. More
4990 * identity resolving keys, means more time during scanning is
4991 * needed to actually resolve these addresses.
4992 */
4993 if (bacmp(&irk->rpa, BDADDR_ANY))
4994 ev.store_hint = 0x01;
4995 else
4996 ev.store_hint = 0x00;
4997
Johan Hedberg95fbac82014-02-19 15:18:31 +02004998 bacpy(&ev.rpa, &irk->rpa);
4999 bacpy(&ev.irk.addr.bdaddr, &irk->bdaddr);
5000 ev.irk.addr.type = link_to_bdaddr(LE_LINK, irk->addr_type);
5001 memcpy(ev.irk.val, irk->val, sizeof(irk->val));
5002
5003 mgmt_event(MGMT_EV_NEW_IRK, hdev, &ev, sizeof(ev), NULL);
5004}
5005
Marcel Holtmann94933992013-10-15 10:26:39 -07005006static inline u16 eir_append_data(u8 *eir, u16 eir_len, u8 type, u8 *data,
5007 u8 data_len)
5008{
5009 eir[eir_len++] = sizeof(type) + data_len;
5010 eir[eir_len++] = type;
5011 memcpy(&eir[eir_len], data, data_len);
5012 eir_len += data_len;
5013
5014 return eir_len;
5015}
5016
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07005017void mgmt_device_connected(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5018 u8 addr_type, u32 flags, u8 *name, u8 name_len,
5019 u8 *dev_class)
Johan Hedbergf7520542011-01-20 12:34:39 +02005020{
Johan Hedbergb644ba32012-01-17 21:48:47 +02005021 char buf[512];
5022 struct mgmt_ev_device_connected *ev = (void *) buf;
5023 u16 eir_len = 0;
Johan Hedbergf7520542011-01-20 12:34:39 +02005024
Johan Hedbergb644ba32012-01-17 21:48:47 +02005025 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005026 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergf7520542011-01-20 12:34:39 +02005027
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02005028 ev->flags = __cpu_to_le32(flags);
Johan Hedberg08c79b62012-02-23 22:31:51 +02005029
Johan Hedbergb644ba32012-01-17 21:48:47 +02005030 if (name_len > 0)
5031 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005032 name, name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005033
5034 if (dev_class && memcmp(dev_class, "\0\0\0", 3) != 0)
Brian Gix53156382012-03-09 14:07:03 -08005035 eir_len = eir_append_data(ev->eir, eir_len,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005036 EIR_CLASS_OF_DEV, dev_class, 3);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005037
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02005038 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005039
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07005040 mgmt_event(MGMT_EV_DEVICE_CONNECTED, hdev, buf,
5041 sizeof(*ev) + eir_len, NULL);
Johan Hedbergf7520542011-01-20 12:34:39 +02005042}
5043
Johan Hedberg8962ee72011-01-20 12:40:27 +02005044static void disconnect_rsp(struct pending_cmd *cmd, void *data)
5045{
Szymon Jancc68fb7f2011-03-22 13:12:19 +01005046 struct mgmt_cp_disconnect *cp = cmd->param;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005047 struct sock **sk = data;
Johan Hedberga38528f2011-01-22 06:46:43 +02005048 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005049
Johan Hedberg88c3df12012-02-09 14:27:38 +02005050 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
5051 rp.addr.type = cp->addr.type;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005052
Johan Hedbergaee9b2182012-02-18 15:07:59 +02005053 cmd_complete(cmd->sk, cmd->index, MGMT_OP_DISCONNECT, 0, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005054 sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02005055
5056 *sk = cmd->sk;
5057 sock_hold(*sk);
5058
Johan Hedberga664b5b2011-02-19 12:06:02 -03005059 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02005060}
5061
Johan Hedberg124f6e32012-02-09 13:50:12 +02005062static void unpair_device_rsp(struct pending_cmd *cmd, void *data)
Johan Hedberga8a1d192011-11-10 15:54:38 +02005063{
Johan Hedbergb1078ad2012-02-09 17:21:16 +02005064 struct hci_dev *hdev = data;
Johan Hedberg124f6e32012-02-09 13:50:12 +02005065 struct mgmt_cp_unpair_device *cp = cmd->param;
5066 struct mgmt_rp_unpair_device rp;
Johan Hedberga8a1d192011-11-10 15:54:38 +02005067
5068 memset(&rp, 0, sizeof(rp));
Johan Hedberg124f6e32012-02-09 13:50:12 +02005069 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
5070 rp.addr.type = cp->addr.type;
Johan Hedberga8a1d192011-11-10 15:54:38 +02005071
Johan Hedbergb1078ad2012-02-09 17:21:16 +02005072 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, cmd->sk);
5073
Johan Hedbergaee9b2182012-02-18 15:07:59 +02005074 cmd_complete(cmd->sk, cmd->index, cmd->opcode, 0, &rp, sizeof(rp));
Johan Hedberga8a1d192011-11-10 15:54:38 +02005075
5076 mgmt_pending_remove(cmd);
5077}
5078
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07005079void mgmt_device_disconnected(struct hci_dev *hdev, bdaddr_t *bdaddr,
Johan Hedberg12d4a3b2014-02-24 14:52:18 +02005080 u8 link_type, u8 addr_type, u8 reason,
5081 bool mgmt_connected)
Johan Hedbergf7520542011-01-20 12:34:39 +02005082{
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02005083 struct mgmt_ev_device_disconnected ev;
Johan Hedberg8b064a32014-02-24 14:52:22 +02005084 struct pending_cmd *power_off;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005085 struct sock *sk = NULL;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005086
Johan Hedberg8b064a32014-02-24 14:52:22 +02005087 power_off = mgmt_pending_find(MGMT_OP_SET_POWERED, hdev);
5088 if (power_off) {
5089 struct mgmt_mode *cp = power_off->param;
5090
5091 /* The connection is still in hci_conn_hash so test for 1
5092 * instead of 0 to know if this is the last one.
5093 */
5094 if (!cp->val && hci_conn_count(hdev) == 1)
5095 queue_work(hdev->req_workqueue, &hdev->power_off.work);
5096 }
5097
Johan Hedberg12d4a3b2014-02-24 14:52:18 +02005098 if (!mgmt_connected)
5099 return;
5100
Andre Guedes57eb7762013-10-30 19:01:41 -03005101 if (link_type != ACL_LINK && link_type != LE_LINK)
5102 return;
5103
Johan Hedberg744cf192011-11-08 20:40:14 +02005104 mgmt_pending_foreach(MGMT_OP_DISCONNECT, hdev, disconnect_rsp, &sk);
Johan Hedbergf7520542011-01-20 12:34:39 +02005105
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02005106 bacpy(&ev.addr.bdaddr, bdaddr);
5107 ev.addr.type = link_to_bdaddr(link_type, addr_type);
5108 ev.reason = reason;
Johan Hedbergf7520542011-01-20 12:34:39 +02005109
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07005110 mgmt_event(MGMT_EV_DEVICE_DISCONNECTED, hdev, &ev, sizeof(ev), sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02005111
5112 if (sk)
Szymon Jancd97dcb62012-03-16 16:02:56 +01005113 sock_put(sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02005114
Johan Hedberg124f6e32012-02-09 13:50:12 +02005115 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005116 hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02005117}
5118
Marcel Holtmann78929242013-10-06 23:55:47 -07005119void mgmt_disconnect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr,
5120 u8 link_type, u8 addr_type, u8 status)
Johan Hedberg8962ee72011-01-20 12:40:27 +02005121{
Andre Guedes3655bba2013-10-30 19:01:40 -03005122 u8 bdaddr_type = link_to_bdaddr(link_type, addr_type);
5123 struct mgmt_cp_disconnect *cp;
Johan Hedberg88c3df12012-02-09 14:27:38 +02005124 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005125 struct pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005126
Jefferson Delfes36a75f12012-09-18 13:36:54 -04005127 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
5128 hdev);
5129
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005130 cmd = mgmt_pending_find(MGMT_OP_DISCONNECT, hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02005131 if (!cmd)
Marcel Holtmann78929242013-10-06 23:55:47 -07005132 return;
Johan Hedberg8962ee72011-01-20 12:40:27 +02005133
Andre Guedes3655bba2013-10-30 19:01:40 -03005134 cp = cmd->param;
5135
5136 if (bacmp(bdaddr, &cp->addr.bdaddr))
5137 return;
5138
5139 if (cp->addr.type != bdaddr_type)
5140 return;
5141
Johan Hedberg88c3df12012-02-09 14:27:38 +02005142 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes3655bba2013-10-30 19:01:40 -03005143 rp.addr.type = bdaddr_type;
Johan Hedberg37d9ef72011-11-10 15:54:39 +02005144
Marcel Holtmann78929242013-10-06 23:55:47 -07005145 cmd_complete(cmd->sk, cmd->index, MGMT_OP_DISCONNECT,
5146 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02005147
Johan Hedberga664b5b2011-02-19 12:06:02 -03005148 mgmt_pending_remove(cmd);
Johan Hedbergf7520542011-01-20 12:34:39 +02005149}
Johan Hedberg17d5c042011-01-22 06:09:08 +02005150
Marcel Holtmann445608d2013-10-06 23:55:48 -07005151void mgmt_connect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5152 u8 addr_type, u8 status)
Johan Hedberg17d5c042011-01-22 06:09:08 +02005153{
5154 struct mgmt_ev_connect_failed ev;
5155
Johan Hedberg4c659c32011-11-07 23:13:39 +02005156 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005157 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02005158 ev.status = mgmt_status(status);
Johan Hedberg17d5c042011-01-22 06:09:08 +02005159
Marcel Holtmann445608d2013-10-06 23:55:48 -07005160 mgmt_event(MGMT_EV_CONNECT_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg17d5c042011-01-22 06:09:08 +02005161}
Johan Hedberg980e1a52011-01-22 06:10:07 +02005162
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07005163void mgmt_pin_code_request(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 secure)
Johan Hedberg980e1a52011-01-22 06:10:07 +02005164{
5165 struct mgmt_ev_pin_code_request ev;
5166
Johan Hedbergd8457692012-02-17 14:24:57 +02005167 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03005168 ev.addr.type = BDADDR_BREDR;
Waldemar Rymarkiewicza770bb52011-04-28 12:07:59 +02005169 ev.secure = secure;
Johan Hedberg980e1a52011-01-22 06:10:07 +02005170
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07005171 mgmt_event(MGMT_EV_PIN_CODE_REQUEST, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg980e1a52011-01-22 06:10:07 +02005172}
5173
Marcel Holtmanne669cf82013-10-15 14:26:21 -07005174void mgmt_pin_code_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
5175 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02005176{
5177 struct pending_cmd *cmd;
Johan Hedbergac56fb12011-02-19 12:05:59 -03005178 struct mgmt_rp_pin_code_reply rp;
Johan Hedberg980e1a52011-01-22 06:10:07 +02005179
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005180 cmd = mgmt_pending_find(MGMT_OP_PIN_CODE_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02005181 if (!cmd)
Marcel Holtmanne669cf82013-10-15 14:26:21 -07005182 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02005183
Johan Hedbergd8457692012-02-17 14:24:57 +02005184 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03005185 rp.addr.type = BDADDR_BREDR;
Johan Hedbergac56fb12011-02-19 12:05:59 -03005186
Marcel Holtmanne669cf82013-10-15 14:26:21 -07005187 cmd_complete(cmd->sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
5188 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg980e1a52011-01-22 06:10:07 +02005189
Johan Hedberga664b5b2011-02-19 12:06:02 -03005190 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02005191}
5192
Marcel Holtmann3eb38522013-10-15 14:26:22 -07005193void mgmt_pin_code_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
5194 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02005195{
5196 struct pending_cmd *cmd;
Johan Hedbergac56fb12011-02-19 12:05:59 -03005197 struct mgmt_rp_pin_code_reply rp;
Johan Hedberg980e1a52011-01-22 06:10:07 +02005198
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005199 cmd = mgmt_pending_find(MGMT_OP_PIN_CODE_NEG_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02005200 if (!cmd)
Marcel Holtmann3eb38522013-10-15 14:26:22 -07005201 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02005202
Johan Hedbergd8457692012-02-17 14:24:57 +02005203 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03005204 rp.addr.type = BDADDR_BREDR;
Johan Hedbergac56fb12011-02-19 12:05:59 -03005205
Marcel Holtmann3eb38522013-10-15 14:26:22 -07005206 cmd_complete(cmd->sk, hdev->id, MGMT_OP_PIN_CODE_NEG_REPLY,
5207 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg980e1a52011-01-22 06:10:07 +02005208
Johan Hedberga664b5b2011-02-19 12:06:02 -03005209 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02005210}
Johan Hedberga5c29682011-02-19 12:05:57 -03005211
Johan Hedberg744cf192011-11-08 20:40:14 +02005212int mgmt_user_confirm_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005213 u8 link_type, u8 addr_type, __le32 value,
5214 u8 confirm_hint)
Johan Hedberga5c29682011-02-19 12:05:57 -03005215{
5216 struct mgmt_ev_user_confirm_request ev;
5217
Johan Hedberg744cf192011-11-08 20:40:14 +02005218 BT_DBG("%s", hdev->name);
Johan Hedberga5c29682011-02-19 12:05:57 -03005219
Johan Hedberg272d90d2012-02-09 15:26:12 +02005220 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005221 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberg55bc1a32011-04-28 11:28:56 -07005222 ev.confirm_hint = confirm_hint;
Andrei Emeltchenko78e8098e2012-03-09 13:00:50 +02005223 ev.value = value;
Johan Hedberga5c29682011-02-19 12:05:57 -03005224
Johan Hedberg744cf192011-11-08 20:40:14 +02005225 return mgmt_event(MGMT_EV_USER_CONFIRM_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005226 NULL);
Johan Hedberga5c29682011-02-19 12:05:57 -03005227}
5228
Johan Hedberg272d90d2012-02-09 15:26:12 +02005229int mgmt_user_passkey_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03005230 u8 link_type, u8 addr_type)
Brian Gix604086b2011-11-23 08:28:33 -08005231{
5232 struct mgmt_ev_user_passkey_request ev;
5233
5234 BT_DBG("%s", hdev->name);
5235
Johan Hedberg272d90d2012-02-09 15:26:12 +02005236 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005237 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Brian Gix604086b2011-11-23 08:28:33 -08005238
5239 return mgmt_event(MGMT_EV_USER_PASSKEY_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005240 NULL);
Brian Gix604086b2011-11-23 08:28:33 -08005241}
5242
Brian Gix0df4c182011-11-16 13:53:13 -08005243static int user_pairing_resp_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03005244 u8 link_type, u8 addr_type, u8 status,
5245 u8 opcode)
Johan Hedberga5c29682011-02-19 12:05:57 -03005246{
5247 struct pending_cmd *cmd;
5248 struct mgmt_rp_user_confirm_reply rp;
5249 int err;
5250
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005251 cmd = mgmt_pending_find(opcode, hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03005252 if (!cmd)
5253 return -ENOENT;
5254
Johan Hedberg272d90d2012-02-09 15:26:12 +02005255 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005256 rp.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergaee9b2182012-02-18 15:07:59 +02005257 err = cmd_complete(cmd->sk, hdev->id, opcode, mgmt_status(status),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005258 &rp, sizeof(rp));
Johan Hedberga5c29682011-02-19 12:05:57 -03005259
Johan Hedberga664b5b2011-02-19 12:06:02 -03005260 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03005261
5262 return err;
5263}
5264
Johan Hedberg744cf192011-11-08 20:40:14 +02005265int mgmt_user_confirm_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005266 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03005267{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005268 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005269 status, MGMT_OP_USER_CONFIRM_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03005270}
5271
Johan Hedberg272d90d2012-02-09 15:26:12 +02005272int mgmt_user_confirm_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005273 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03005274{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005275 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03005276 status,
5277 MGMT_OP_USER_CONFIRM_NEG_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03005278}
Johan Hedberg2a611692011-02-19 12:06:00 -03005279
Brian Gix604086b2011-11-23 08:28:33 -08005280int mgmt_user_passkey_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005281 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08005282{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005283 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005284 status, MGMT_OP_USER_PASSKEY_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08005285}
5286
Johan Hedberg272d90d2012-02-09 15:26:12 +02005287int mgmt_user_passkey_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005288 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08005289{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005290 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03005291 status,
5292 MGMT_OP_USER_PASSKEY_NEG_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08005293}
5294
Johan Hedberg92a25252012-09-06 18:39:26 +03005295int mgmt_user_passkey_notify(struct hci_dev *hdev, bdaddr_t *bdaddr,
5296 u8 link_type, u8 addr_type, u32 passkey,
5297 u8 entered)
5298{
5299 struct mgmt_ev_passkey_notify ev;
5300
5301 BT_DBG("%s", hdev->name);
5302
5303 bacpy(&ev.addr.bdaddr, bdaddr);
5304 ev.addr.type = link_to_bdaddr(link_type, addr_type);
5305 ev.passkey = __cpu_to_le32(passkey);
5306 ev.entered = entered;
5307
5308 return mgmt_event(MGMT_EV_PASSKEY_NOTIFY, hdev, &ev, sizeof(ev), NULL);
5309}
5310
Marcel Holtmanne5460992013-10-15 14:26:23 -07005311void mgmt_auth_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5312 u8 addr_type, u8 status)
Johan Hedberg2a611692011-02-19 12:06:00 -03005313{
5314 struct mgmt_ev_auth_failed ev;
5315
Johan Hedbergbab73cb2012-02-09 16:07:29 +02005316 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005317 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02005318 ev.status = mgmt_status(status);
Johan Hedberg2a611692011-02-19 12:06:00 -03005319
Marcel Holtmanne5460992013-10-15 14:26:23 -07005320 mgmt_event(MGMT_EV_AUTH_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg2a611692011-02-19 12:06:00 -03005321}
Johan Hedbergb312b1612011-03-16 14:29:37 +02005322
Marcel Holtmann464996a2013-10-15 14:26:24 -07005323void mgmt_auth_enable_complete(struct hci_dev *hdev, u8 status)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005324{
5325 struct cmd_lookup match = { NULL, hdev };
Marcel Holtmann464996a2013-10-15 14:26:24 -07005326 bool changed;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005327
5328 if (status) {
5329 u8 mgmt_err = mgmt_status(status);
5330 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005331 cmd_status_rsp, &mgmt_err);
Marcel Holtmann464996a2013-10-15 14:26:24 -07005332 return;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005333 }
5334
Marcel Holtmann464996a2013-10-15 14:26:24 -07005335 if (test_bit(HCI_AUTH, &hdev->flags))
5336 changed = !test_and_set_bit(HCI_LINK_SECURITY,
5337 &hdev->dev_flags);
5338 else
5339 changed = test_and_clear_bit(HCI_LINK_SECURITY,
5340 &hdev->dev_flags);
Johan Hedberg47990ea2012-02-22 11:58:37 +02005341
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005342 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev, settings_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005343 &match);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005344
Johan Hedberg47990ea2012-02-22 11:58:37 +02005345 if (changed)
Marcel Holtmann464996a2013-10-15 14:26:24 -07005346 new_settings(hdev, match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005347
5348 if (match.sk)
5349 sock_put(match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005350}
5351
Johan Hedberg890ea892013-03-15 17:06:52 -05005352static void clear_eir(struct hci_request *req)
Johan Hedbergcacaf522012-02-21 00:52:42 +02005353{
Johan Hedberg890ea892013-03-15 17:06:52 -05005354 struct hci_dev *hdev = req->hdev;
Johan Hedbergcacaf522012-02-21 00:52:42 +02005355 struct hci_cp_write_eir cp;
5356
Johan Hedberg976eb202012-10-24 21:12:01 +03005357 if (!lmp_ext_inq_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05005358 return;
Johan Hedbergcacaf522012-02-21 00:52:42 +02005359
Johan Hedbergc80da272012-02-22 15:38:48 +02005360 memset(hdev->eir, 0, sizeof(hdev->eir));
5361
Johan Hedbergcacaf522012-02-21 00:52:42 +02005362 memset(&cp, 0, sizeof(cp));
5363
Johan Hedberg890ea892013-03-15 17:06:52 -05005364 hci_req_add(req, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
Johan Hedbergcacaf522012-02-21 00:52:42 +02005365}
5366
Marcel Holtmann3e248562013-10-15 14:26:25 -07005367void mgmt_ssp_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005368{
5369 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg890ea892013-03-15 17:06:52 -05005370 struct hci_request req;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005371 bool changed = false;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005372
5373 if (status) {
5374 u8 mgmt_err = mgmt_status(status);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005375
5376 if (enable && test_and_clear_bit(HCI_SSP_ENABLED,
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005377 &hdev->dev_flags)) {
5378 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Marcel Holtmann3e248562013-10-15 14:26:25 -07005379 new_settings(hdev, NULL);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005380 }
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005381
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005382 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, cmd_status_rsp,
5383 &mgmt_err);
Marcel Holtmann3e248562013-10-15 14:26:25 -07005384 return;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005385 }
5386
5387 if (enable) {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005388 changed = !test_and_set_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005389 } else {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005390 changed = test_and_clear_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
5391 if (!changed)
5392 changed = test_and_clear_bit(HCI_HS_ENABLED,
5393 &hdev->dev_flags);
5394 else
5395 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005396 }
5397
5398 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, settings_rsp, &match);
5399
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005400 if (changed)
Marcel Holtmann3e248562013-10-15 14:26:25 -07005401 new_settings(hdev, match.sk);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005402
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005403 if (match.sk)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005404 sock_put(match.sk);
5405
Johan Hedberg890ea892013-03-15 17:06:52 -05005406 hci_req_init(&req, hdev);
5407
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005408 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -05005409 update_eir(&req);
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005410 else
Johan Hedberg890ea892013-03-15 17:06:52 -05005411 clear_eir(&req);
5412
5413 hci_req_run(&req, NULL);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005414}
5415
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005416void mgmt_sc_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
5417{
5418 struct cmd_lookup match = { NULL, hdev };
5419 bool changed = false;
5420
5421 if (status) {
5422 u8 mgmt_err = mgmt_status(status);
5423
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005424 if (enable) {
5425 if (test_and_clear_bit(HCI_SC_ENABLED,
5426 &hdev->dev_flags))
5427 new_settings(hdev, NULL);
5428 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
5429 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005430
5431 mgmt_pending_foreach(MGMT_OP_SET_SECURE_CONN, hdev,
5432 cmd_status_rsp, &mgmt_err);
5433 return;
5434 }
5435
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005436 if (enable) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005437 changed = !test_and_set_bit(HCI_SC_ENABLED, &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005438 } else {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005439 changed = test_and_clear_bit(HCI_SC_ENABLED, &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005440 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
5441 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005442
5443 mgmt_pending_foreach(MGMT_OP_SET_SECURE_CONN, hdev,
5444 settings_rsp, &match);
5445
5446 if (changed)
5447 new_settings(hdev, match.sk);
5448
5449 if (match.sk)
5450 sock_put(match.sk);
5451}
5452
Johan Hedberg92da6092013-03-15 17:06:55 -05005453static void sk_lookup(struct pending_cmd *cmd, void *data)
Johan Hedberg90e70452012-02-23 23:09:40 +02005454{
5455 struct cmd_lookup *match = data;
5456
Johan Hedberg90e70452012-02-23 23:09:40 +02005457 if (match->sk == NULL) {
5458 match->sk = cmd->sk;
5459 sock_hold(match->sk);
5460 }
Johan Hedberg90e70452012-02-23 23:09:40 +02005461}
5462
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07005463void mgmt_set_class_of_dev_complete(struct hci_dev *hdev, u8 *dev_class,
5464 u8 status)
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005465{
Johan Hedberg90e70452012-02-23 23:09:40 +02005466 struct cmd_lookup match = { NULL, hdev, mgmt_status(status) };
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005467
Johan Hedberg92da6092013-03-15 17:06:55 -05005468 mgmt_pending_foreach(MGMT_OP_SET_DEV_CLASS, hdev, sk_lookup, &match);
5469 mgmt_pending_foreach(MGMT_OP_ADD_UUID, hdev, sk_lookup, &match);
5470 mgmt_pending_foreach(MGMT_OP_REMOVE_UUID, hdev, sk_lookup, &match);
Johan Hedberg90e70452012-02-23 23:09:40 +02005471
5472 if (!status)
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07005473 mgmt_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev, dev_class, 3,
5474 NULL);
Johan Hedberg90e70452012-02-23 23:09:40 +02005475
5476 if (match.sk)
5477 sock_put(match.sk);
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005478}
5479
Marcel Holtmann7667da32013-10-15 14:26:27 -07005480void mgmt_set_local_name_complete(struct hci_dev *hdev, u8 *name, u8 status)
Johan Hedbergb312b1612011-03-16 14:29:37 +02005481{
Johan Hedbergb312b1612011-03-16 14:29:37 +02005482 struct mgmt_cp_set_local_name ev;
Johan Hedberg13928972013-03-15 17:07:00 -05005483 struct pending_cmd *cmd;
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02005484
Johan Hedberg13928972013-03-15 17:07:00 -05005485 if (status)
Marcel Holtmann7667da32013-10-15 14:26:27 -07005486 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02005487
5488 memset(&ev, 0, sizeof(ev));
5489 memcpy(ev.name, name, HCI_MAX_NAME_LENGTH);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02005490 memcpy(ev.short_name, hdev->short_name, HCI_MAX_SHORT_NAME_LENGTH);
Johan Hedbergb312b1612011-03-16 14:29:37 +02005491
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005492 cmd = mgmt_pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
Johan Hedberg13928972013-03-15 17:07:00 -05005493 if (!cmd) {
5494 memcpy(hdev->dev_name, name, sizeof(hdev->dev_name));
Johan Hedbergb312b1612011-03-16 14:29:37 +02005495
Johan Hedberg13928972013-03-15 17:07:00 -05005496 /* If this is a HCI command related to powering on the
5497 * HCI dev don't send any mgmt signals.
5498 */
5499 if (mgmt_pending_find(MGMT_OP_SET_POWERED, hdev))
Marcel Holtmann7667da32013-10-15 14:26:27 -07005500 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02005501 }
5502
Marcel Holtmann7667da32013-10-15 14:26:27 -07005503 mgmt_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, &ev, sizeof(ev),
5504 cmd ? cmd->sk : NULL);
Johan Hedbergb312b1612011-03-16 14:29:37 +02005505}
Szymon Jancc35938b2011-03-22 13:12:21 +01005506
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005507void mgmt_read_local_oob_data_complete(struct hci_dev *hdev, u8 *hash192,
5508 u8 *randomizer192, u8 *hash256,
5509 u8 *randomizer256, u8 status)
Szymon Jancc35938b2011-03-22 13:12:21 +01005510{
5511 struct pending_cmd *cmd;
Szymon Jancc35938b2011-03-22 13:12:21 +01005512
Johan Hedberg744cf192011-11-08 20:40:14 +02005513 BT_DBG("%s status %u", hdev->name, status);
Szymon Jancc35938b2011-03-22 13:12:21 +01005514
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005515 cmd = mgmt_pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01005516 if (!cmd)
Marcel Holtmann3edaf092013-10-15 14:26:28 -07005517 return;
Szymon Jancc35938b2011-03-22 13:12:21 +01005518
5519 if (status) {
Marcel Holtmann3edaf092013-10-15 14:26:28 -07005520 cmd_status(cmd->sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
5521 mgmt_status(status));
Szymon Jancc35938b2011-03-22 13:12:21 +01005522 } else {
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005523 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags) &&
5524 hash256 && randomizer256) {
5525 struct mgmt_rp_read_local_oob_ext_data rp;
Szymon Jancc35938b2011-03-22 13:12:21 +01005526
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005527 memcpy(rp.hash192, hash192, sizeof(rp.hash192));
5528 memcpy(rp.randomizer192, randomizer192,
5529 sizeof(rp.randomizer192));
Szymon Jancc35938b2011-03-22 13:12:21 +01005530
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005531 memcpy(rp.hash256, hash256, sizeof(rp.hash256));
5532 memcpy(rp.randomizer256, randomizer256,
5533 sizeof(rp.randomizer256));
5534
5535 cmd_complete(cmd->sk, hdev->id,
5536 MGMT_OP_READ_LOCAL_OOB_DATA, 0,
5537 &rp, sizeof(rp));
5538 } else {
5539 struct mgmt_rp_read_local_oob_data rp;
5540
5541 memcpy(rp.hash, hash192, sizeof(rp.hash));
5542 memcpy(rp.randomizer, randomizer192,
5543 sizeof(rp.randomizer));
5544
5545 cmd_complete(cmd->sk, hdev->id,
5546 MGMT_OP_READ_LOCAL_OOB_DATA, 0,
5547 &rp, sizeof(rp));
5548 }
Szymon Jancc35938b2011-03-22 13:12:21 +01005549 }
5550
5551 mgmt_pending_remove(cmd);
Szymon Jancc35938b2011-03-22 13:12:21 +01005552}
Johan Hedberge17acd42011-03-30 23:57:16 +03005553
Marcel Holtmann901801b2013-10-06 23:55:51 -07005554void mgmt_device_found(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5555 u8 addr_type, u8 *dev_class, s8 rssi, u8 cfm_name, u8
5556 ssp, u8 *eir, u16 eir_len)
Johan Hedberge17acd42011-03-30 23:57:16 +03005557{
Johan Hedberge319d2e2012-01-15 19:51:59 +02005558 char buf[512];
5559 struct mgmt_ev_device_found *ev = (void *) buf;
Johan Hedberg5cedbb82014-02-18 21:41:37 +02005560 struct smp_irk *irk;
Johan Hedberg1dc06092012-01-15 21:01:23 +02005561 size_t ev_size;
Johan Hedberge17acd42011-03-30 23:57:16 +03005562
Andre Guedes12602d02013-04-30 15:29:40 -03005563 if (!hci_discovery_active(hdev))
Marcel Holtmann901801b2013-10-06 23:55:51 -07005564 return;
Andre Guedes12602d02013-04-30 15:29:40 -03005565
Johan Hedberg1dc06092012-01-15 21:01:23 +02005566 /* Leave 5 bytes for a potential CoD field */
5567 if (sizeof(*ev) + eir_len + 5 > sizeof(buf))
Marcel Holtmann901801b2013-10-06 23:55:51 -07005568 return;
Andre Guedes7d262f82012-01-10 18:20:49 -03005569
Johan Hedberg1dc06092012-01-15 21:01:23 +02005570 memset(buf, 0, sizeof(buf));
5571
Johan Hedberg5cedbb82014-02-18 21:41:37 +02005572 irk = hci_get_irk(hdev, bdaddr, addr_type);
5573 if (irk) {
5574 bacpy(&ev->addr.bdaddr, &irk->bdaddr);
5575 ev->addr.type = link_to_bdaddr(link_type, irk->addr_type);
5576 } else {
5577 bacpy(&ev->addr.bdaddr, bdaddr);
5578 ev->addr.type = link_to_bdaddr(link_type, addr_type);
5579 }
5580
Johan Hedberge319d2e2012-01-15 19:51:59 +02005581 ev->rssi = rssi;
Johan Hedberg9a395a82012-02-23 00:00:32 +02005582 if (cfm_name)
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05305583 ev->flags |= __constant_cpu_to_le32(MGMT_DEV_FOUND_CONFIRM_NAME);
Johan Hedberg388fc8f2012-02-23 00:38:59 +02005584 if (!ssp)
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05305585 ev->flags |= __constant_cpu_to_le32(MGMT_DEV_FOUND_LEGACY_PAIRING);
Johan Hedberge17acd42011-03-30 23:57:16 +03005586
Johan Hedberg1dc06092012-01-15 21:01:23 +02005587 if (eir_len > 0)
Johan Hedberge319d2e2012-01-15 19:51:59 +02005588 memcpy(ev->eir, eir, eir_len);
Johan Hedberge17acd42011-03-30 23:57:16 +03005589
Johan Hedberg1dc06092012-01-15 21:01:23 +02005590 if (dev_class && !eir_has_data_type(ev->eir, eir_len, EIR_CLASS_OF_DEV))
5591 eir_len = eir_append_data(ev->eir, eir_len, EIR_CLASS_OF_DEV,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005592 dev_class, 3);
Johan Hedberg1dc06092012-01-15 21:01:23 +02005593
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02005594 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedberg1dc06092012-01-15 21:01:23 +02005595 ev_size = sizeof(*ev) + eir_len;
Andre Guedesf8523592011-09-09 18:56:26 -03005596
Marcel Holtmann901801b2013-10-06 23:55:51 -07005597 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, ev_size, NULL);
Johan Hedberge17acd42011-03-30 23:57:16 +03005598}
Johan Hedberga88a9652011-03-30 13:18:12 +03005599
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07005600void mgmt_remote_name(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5601 u8 addr_type, s8 rssi, u8 *name, u8 name_len)
Johan Hedberga88a9652011-03-30 13:18:12 +03005602{
Johan Hedbergb644ba32012-01-17 21:48:47 +02005603 struct mgmt_ev_device_found *ev;
5604 char buf[sizeof(*ev) + HCI_MAX_NAME_LENGTH + 2];
5605 u16 eir_len;
Johan Hedberga88a9652011-03-30 13:18:12 +03005606
Johan Hedbergb644ba32012-01-17 21:48:47 +02005607 ev = (struct mgmt_ev_device_found *) buf;
Johan Hedberga88a9652011-03-30 13:18:12 +03005608
Johan Hedbergb644ba32012-01-17 21:48:47 +02005609 memset(buf, 0, sizeof(buf));
Johan Hedberga88a9652011-03-30 13:18:12 +03005610
Johan Hedbergb644ba32012-01-17 21:48:47 +02005611 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005612 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005613 ev->rssi = rssi;
5614
5615 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE, name,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005616 name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005617
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02005618 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005619
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07005620 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, sizeof(*ev) + eir_len, NULL);
Johan Hedberga88a9652011-03-30 13:18:12 +03005621}
Johan Hedberg314b2382011-04-27 10:29:57 -04005622
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07005623void mgmt_discovering(struct hci_dev *hdev, u8 discovering)
Johan Hedberg314b2382011-04-27 10:29:57 -04005624{
Johan Hedbergf963e8e2012-02-20 23:30:44 +02005625 struct mgmt_ev_discovering ev;
Johan Hedberg164a6e72011-11-01 17:06:44 +02005626 struct pending_cmd *cmd;
5627
Andre Guedes343fb142011-11-22 17:14:19 -03005628 BT_DBG("%s discovering %u", hdev->name, discovering);
5629
Johan Hedberg164a6e72011-11-01 17:06:44 +02005630 if (discovering)
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005631 cmd = mgmt_pending_find(MGMT_OP_START_DISCOVERY, hdev);
Johan Hedberg164a6e72011-11-01 17:06:44 +02005632 else
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005633 cmd = mgmt_pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
Johan Hedberg164a6e72011-11-01 17:06:44 +02005634
5635 if (cmd != NULL) {
Johan Hedbergf808e162012-02-19 12:52:07 +02005636 u8 type = hdev->discovery.type;
5637
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005638 cmd_complete(cmd->sk, hdev->id, cmd->opcode, 0, &type,
5639 sizeof(type));
Johan Hedberg164a6e72011-11-01 17:06:44 +02005640 mgmt_pending_remove(cmd);
5641 }
5642
Johan Hedbergf963e8e2012-02-20 23:30:44 +02005643 memset(&ev, 0, sizeof(ev));
5644 ev.type = hdev->discovery.type;
5645 ev.discovering = discovering;
5646
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07005647 mgmt_event(MGMT_EV_DISCOVERING, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg314b2382011-04-27 10:29:57 -04005648}
Antti Julku5e762442011-08-25 16:48:02 +03005649
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005650int mgmt_device_blocked(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 type)
Antti Julku5e762442011-08-25 16:48:02 +03005651{
5652 struct pending_cmd *cmd;
5653 struct mgmt_ev_device_blocked ev;
5654
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005655 cmd = mgmt_pending_find(MGMT_OP_BLOCK_DEVICE, hdev);
Antti Julku5e762442011-08-25 16:48:02 +03005656
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005657 bacpy(&ev.addr.bdaddr, bdaddr);
5658 ev.addr.type = type;
Antti Julku5e762442011-08-25 16:48:02 +03005659
Johan Hedberg744cf192011-11-08 20:40:14 +02005660 return mgmt_event(MGMT_EV_DEVICE_BLOCKED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005661 cmd ? cmd->sk : NULL);
Antti Julku5e762442011-08-25 16:48:02 +03005662}
5663
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005664int mgmt_device_unblocked(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 type)
Antti Julku5e762442011-08-25 16:48:02 +03005665{
5666 struct pending_cmd *cmd;
5667 struct mgmt_ev_device_unblocked ev;
5668
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005669 cmd = mgmt_pending_find(MGMT_OP_UNBLOCK_DEVICE, hdev);
Antti Julku5e762442011-08-25 16:48:02 +03005670
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005671 bacpy(&ev.addr.bdaddr, bdaddr);
5672 ev.addr.type = type;
Antti Julku5e762442011-08-25 16:48:02 +03005673
Johan Hedberg744cf192011-11-08 20:40:14 +02005674 return mgmt_event(MGMT_EV_DEVICE_UNBLOCKED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005675 cmd ? cmd->sk : NULL);
Antti Julku5e762442011-08-25 16:48:02 +03005676}
Marcel Holtmann5976e602013-10-06 04:08:14 -07005677
5678static void adv_enable_complete(struct hci_dev *hdev, u8 status)
5679{
5680 BT_DBG("%s status %u", hdev->name, status);
5681
5682 /* Clear the advertising mgmt setting if we failed to re-enable it */
5683 if (status) {
5684 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Marcel Holtmanna6d811e2013-10-06 04:11:12 -07005685 new_settings(hdev, NULL);
Marcel Holtmann5976e602013-10-06 04:08:14 -07005686 }
5687}
5688
5689void mgmt_reenable_advertising(struct hci_dev *hdev)
5690{
5691 struct hci_request req;
5692
Marcel Holtmannb145edc2013-10-10 09:47:54 -07005693 if (hci_conn_num(hdev, LE_LINK) > 0)
Marcel Holtmann5976e602013-10-06 04:08:14 -07005694 return;
5695
5696 if (!test_bit(HCI_ADVERTISING, &hdev->dev_flags))
5697 return;
5698
5699 hci_req_init(&req, hdev);
5700 enable_advertising(&req);
5701
5702 /* If this fails we have no option but to let user space know
5703 * that we've disabled advertising.
5704 */
5705 if (hci_req_run(&req, adv_enable_complete) < 0) {
5706 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Marcel Holtmanna6d811e2013-10-06 04:11:12 -07005707 new_settings(hdev, NULL);
Marcel Holtmann5976e602013-10-06 04:08:14 -07005708 }
5709}