blob: 90aac905a98b173234793e7b50c7aa8c003ed8a2 [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 Hedberg41edf162014-02-18 10:19:35 +020084 MGMT_OP_LOAD_IRKS,
Johan Hedberge70bb2e2012-02-13 16:59:33 +020085};
86
87static const u16 mgmt_events[] = {
88 MGMT_EV_CONTROLLER_ERROR,
89 MGMT_EV_INDEX_ADDED,
90 MGMT_EV_INDEX_REMOVED,
91 MGMT_EV_NEW_SETTINGS,
92 MGMT_EV_CLASS_OF_DEV_CHANGED,
93 MGMT_EV_LOCAL_NAME_CHANGED,
94 MGMT_EV_NEW_LINK_KEY,
95 MGMT_EV_NEW_LONG_TERM_KEY,
96 MGMT_EV_DEVICE_CONNECTED,
97 MGMT_EV_DEVICE_DISCONNECTED,
98 MGMT_EV_CONNECT_FAILED,
99 MGMT_EV_PIN_CODE_REQUEST,
100 MGMT_EV_USER_CONFIRM_REQUEST,
101 MGMT_EV_USER_PASSKEY_REQUEST,
102 MGMT_EV_AUTH_FAILED,
103 MGMT_EV_DEVICE_FOUND,
104 MGMT_EV_DISCOVERING,
105 MGMT_EV_DEVICE_BLOCKED,
106 MGMT_EV_DEVICE_UNBLOCKED,
107 MGMT_EV_DEVICE_UNPAIRED,
Johan Hedberg92a25252012-09-06 18:39:26 +0300108 MGMT_EV_PASSKEY_NOTIFY,
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200109};
110
Marcel Holtmann17b02e62012-03-01 14:32:37 -0800111#define CACHE_TIMEOUT msecs_to_jiffies(2 * 1000)
Johan Hedberg7d785252011-12-15 00:47:39 +0200112
Johan Hedberg4b34ee782012-02-21 14:13:02 +0200113#define hdev_is_powered(hdev) (test_bit(HCI_UP, &hdev->flags) && \
114 !test_bit(HCI_AUTO_OFF, &hdev->dev_flags))
115
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200116struct pending_cmd {
117 struct list_head list;
Johan Hedbergfc2f4b12011-11-09 13:58:56 +0200118 u16 opcode;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200119 int index;
Szymon Jancc68fb7f2011-03-22 13:12:19 +0100120 void *param;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200121 struct sock *sk;
Johan Hedberge9a416b2011-02-19 12:05:56 -0300122 void *user_data;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +0200123};
124
Johan Hedbergca69b792011-11-11 18:10:00 +0200125/* HCI to MGMT error code conversion table */
126static u8 mgmt_status_table[] = {
127 MGMT_STATUS_SUCCESS,
128 MGMT_STATUS_UNKNOWN_COMMAND, /* Unknown Command */
129 MGMT_STATUS_NOT_CONNECTED, /* No Connection */
130 MGMT_STATUS_FAILED, /* Hardware Failure */
131 MGMT_STATUS_CONNECT_FAILED, /* Page Timeout */
132 MGMT_STATUS_AUTH_FAILED, /* Authentication Failed */
Johan Hedbergeadd6632014-01-13 17:15:53 +0200133 MGMT_STATUS_AUTH_FAILED, /* PIN or Key Missing */
Johan Hedbergca69b792011-11-11 18:10:00 +0200134 MGMT_STATUS_NO_RESOURCES, /* Memory Full */
135 MGMT_STATUS_TIMEOUT, /* Connection Timeout */
136 MGMT_STATUS_NO_RESOURCES, /* Max Number of Connections */
137 MGMT_STATUS_NO_RESOURCES, /* Max Number of SCO Connections */
138 MGMT_STATUS_ALREADY_CONNECTED, /* ACL Connection Exists */
139 MGMT_STATUS_BUSY, /* Command Disallowed */
140 MGMT_STATUS_NO_RESOURCES, /* Rejected Limited Resources */
141 MGMT_STATUS_REJECTED, /* Rejected Security */
142 MGMT_STATUS_REJECTED, /* Rejected Personal */
143 MGMT_STATUS_TIMEOUT, /* Host Timeout */
144 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported Feature */
145 MGMT_STATUS_INVALID_PARAMS, /* Invalid Parameters */
146 MGMT_STATUS_DISCONNECTED, /* OE User Ended Connection */
147 MGMT_STATUS_NO_RESOURCES, /* OE Low Resources */
148 MGMT_STATUS_DISCONNECTED, /* OE Power Off */
149 MGMT_STATUS_DISCONNECTED, /* Connection Terminated */
150 MGMT_STATUS_BUSY, /* Repeated Attempts */
151 MGMT_STATUS_REJECTED, /* Pairing Not Allowed */
152 MGMT_STATUS_FAILED, /* Unknown LMP PDU */
153 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported Remote Feature */
154 MGMT_STATUS_REJECTED, /* SCO Offset Rejected */
155 MGMT_STATUS_REJECTED, /* SCO Interval Rejected */
156 MGMT_STATUS_REJECTED, /* Air Mode Rejected */
157 MGMT_STATUS_INVALID_PARAMS, /* Invalid LMP Parameters */
158 MGMT_STATUS_FAILED, /* Unspecified Error */
159 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported LMP Parameter Value */
160 MGMT_STATUS_FAILED, /* Role Change Not Allowed */
161 MGMT_STATUS_TIMEOUT, /* LMP Response Timeout */
162 MGMT_STATUS_FAILED, /* LMP Error Transaction Collision */
163 MGMT_STATUS_FAILED, /* LMP PDU Not Allowed */
164 MGMT_STATUS_REJECTED, /* Encryption Mode Not Accepted */
165 MGMT_STATUS_FAILED, /* Unit Link Key Used */
166 MGMT_STATUS_NOT_SUPPORTED, /* QoS Not Supported */
167 MGMT_STATUS_TIMEOUT, /* Instant Passed */
168 MGMT_STATUS_NOT_SUPPORTED, /* Pairing Not Supported */
169 MGMT_STATUS_FAILED, /* Transaction Collision */
170 MGMT_STATUS_INVALID_PARAMS, /* Unacceptable Parameter */
171 MGMT_STATUS_REJECTED, /* QoS Rejected */
172 MGMT_STATUS_NOT_SUPPORTED, /* Classification Not Supported */
173 MGMT_STATUS_REJECTED, /* Insufficient Security */
174 MGMT_STATUS_INVALID_PARAMS, /* Parameter Out Of Range */
175 MGMT_STATUS_BUSY, /* Role Switch Pending */
176 MGMT_STATUS_FAILED, /* Slot Violation */
177 MGMT_STATUS_FAILED, /* Role Switch Failed */
178 MGMT_STATUS_INVALID_PARAMS, /* EIR Too Large */
179 MGMT_STATUS_NOT_SUPPORTED, /* Simple Pairing Not Supported */
180 MGMT_STATUS_BUSY, /* Host Busy Pairing */
181 MGMT_STATUS_REJECTED, /* Rejected, No Suitable Channel */
182 MGMT_STATUS_BUSY, /* Controller Busy */
183 MGMT_STATUS_INVALID_PARAMS, /* Unsuitable Connection Interval */
184 MGMT_STATUS_TIMEOUT, /* Directed Advertising Timeout */
185 MGMT_STATUS_AUTH_FAILED, /* Terminated Due to MIC Failure */
186 MGMT_STATUS_CONNECT_FAILED, /* Connection Establishment Failed */
187 MGMT_STATUS_CONNECT_FAILED, /* MAC Connection Failed */
188};
189
190static u8 mgmt_status(u8 hci_status)
191{
192 if (hci_status < ARRAY_SIZE(mgmt_status_table))
193 return mgmt_status_table[hci_status];
194
195 return MGMT_STATUS_FAILED;
196}
197
Szymon Janc4e51eae2011-02-25 19:05:48 +0100198static int cmd_status(struct sock *sk, u16 index, u16 cmd, u8 status)
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200199{
200 struct sk_buff *skb;
201 struct mgmt_hdr *hdr;
202 struct mgmt_ev_cmd_status *ev;
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300203 int err;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200204
Szymon Janc34eb5252011-02-28 14:10:08 +0100205 BT_DBG("sock %p, index %u, cmd %u, status %u", sk, index, cmd, status);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200206
Andre Guedes790eff42012-06-07 19:05:46 -0300207 skb = alloc_skb(sizeof(*hdr) + sizeof(*ev), GFP_KERNEL);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200208 if (!skb)
209 return -ENOMEM;
210
211 hdr = (void *) skb_put(skb, sizeof(*hdr));
212
Syam Sidhardhan612dfce2012-10-29 22:37:36 +0530213 hdr->opcode = __constant_cpu_to_le16(MGMT_EV_CMD_STATUS);
Szymon Janc4e51eae2011-02-25 19:05:48 +0100214 hdr->index = cpu_to_le16(index);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200215 hdr->len = cpu_to_le16(sizeof(*ev));
216
217 ev = (void *) skb_put(skb, sizeof(*ev));
218 ev->status = status;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200219 ev->opcode = cpu_to_le16(cmd);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200220
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300221 err = sock_queue_rcv_skb(sk, skb);
222 if (err < 0)
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200223 kfree_skb(skb);
224
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300225 return err;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200226}
227
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200228static int cmd_complete(struct sock *sk, u16 index, u16 cmd, u8 status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300229 void *rp, size_t rp_len)
Johan Hedberg02d98122010-12-13 21:07:04 +0200230{
231 struct sk_buff *skb;
232 struct mgmt_hdr *hdr;
233 struct mgmt_ev_cmd_complete *ev;
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300234 int err;
Johan Hedberg02d98122010-12-13 21:07:04 +0200235
236 BT_DBG("sock %p", sk);
237
Andre Guedes790eff42012-06-07 19:05:46 -0300238 skb = alloc_skb(sizeof(*hdr) + sizeof(*ev) + rp_len, GFP_KERNEL);
Johan Hedberg02d98122010-12-13 21:07:04 +0200239 if (!skb)
240 return -ENOMEM;
241
242 hdr = (void *) skb_put(skb, sizeof(*hdr));
Johan Hedberga38528f2011-01-22 06:46:43 +0200243
Syam Sidhardhan612dfce2012-10-29 22:37:36 +0530244 hdr->opcode = __constant_cpu_to_le16(MGMT_EV_CMD_COMPLETE);
Szymon Janc4e51eae2011-02-25 19:05:48 +0100245 hdr->index = cpu_to_le16(index);
Johan Hedberga38528f2011-01-22 06:46:43 +0200246 hdr->len = cpu_to_le16(sizeof(*ev) + rp_len);
Johan Hedberg02d98122010-12-13 21:07:04 +0200247
Johan Hedberga38528f2011-01-22 06:46:43 +0200248 ev = (void *) skb_put(skb, sizeof(*ev) + rp_len);
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200249 ev->opcode = cpu_to_le16(cmd);
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200250 ev->status = status;
Szymon Janc8020c162011-02-28 14:09:50 +0100251
252 if (rp)
253 memcpy(ev->data, rp, rp_len);
Johan Hedberg02d98122010-12-13 21:07:04 +0200254
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300255 err = sock_queue_rcv_skb(sk, skb);
256 if (err < 0)
Johan Hedberg02d98122010-12-13 21:07:04 +0200257 kfree_skb(skb);
258
Marcel Holtmanne5f0e152012-02-22 11:59:01 +0100259 return err;
Johan Hedberg02d98122010-12-13 21:07:04 +0200260}
261
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300262static int read_version(struct sock *sk, struct hci_dev *hdev, void *data,
263 u16 data_len)
Johan Hedberga38528f2011-01-22 06:46:43 +0200264{
265 struct mgmt_rp_read_version rp;
266
267 BT_DBG("sock %p", sk);
268
269 rp.version = MGMT_VERSION;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200270 rp.revision = __constant_cpu_to_le16(MGMT_REVISION);
Johan Hedberga38528f2011-01-22 06:46:43 +0200271
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200272 return cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_VERSION, 0, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300273 sizeof(rp));
Johan Hedberga38528f2011-01-22 06:46:43 +0200274}
275
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300276static int read_commands(struct sock *sk, struct hci_dev *hdev, void *data,
277 u16 data_len)
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200278{
279 struct mgmt_rp_read_commands *rp;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200280 const u16 num_commands = ARRAY_SIZE(mgmt_commands);
281 const u16 num_events = ARRAY_SIZE(mgmt_events);
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +0200282 __le16 *opcode;
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200283 size_t rp_size;
284 int i, err;
285
286 BT_DBG("sock %p", sk);
287
288 rp_size = sizeof(*rp) + ((num_commands + num_events) * sizeof(u16));
289
290 rp = kmalloc(rp_size, GFP_KERNEL);
291 if (!rp)
292 return -ENOMEM;
293
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200294 rp->num_commands = __constant_cpu_to_le16(num_commands);
295 rp->num_events = __constant_cpu_to_le16(num_events);
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200296
297 for (i = 0, opcode = rp->opcodes; i < num_commands; i++, opcode++)
298 put_unaligned_le16(mgmt_commands[i], opcode);
299
300 for (i = 0; i < num_events; i++, opcode++)
301 put_unaligned_le16(mgmt_events[i], opcode);
302
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200303 err = cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_COMMANDS, 0, rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300304 rp_size);
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200305 kfree(rp);
306
307 return err;
308}
309
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300310static int read_index_list(struct sock *sk, struct hci_dev *hdev, void *data,
311 u16 data_len)
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200312{
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200313 struct mgmt_rp_read_index_list *rp;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +0200314 struct hci_dev *d;
Johan Hedberga38528f2011-01-22 06:46:43 +0200315 size_t rp_len;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200316 u16 count;
Johan Hedberg476e44c2012-10-19 20:10:46 +0300317 int err;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200318
319 BT_DBG("sock %p", sk);
320
321 read_lock(&hci_dev_list_lock);
322
323 count = 0;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +0300324 list_for_each_entry(d, &hci_dev_list, list) {
Marcel Holtmann1514b892013-10-06 08:25:01 -0700325 if (d->dev_type == HCI_BREDR)
326 count++;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200327 }
328
Johan Hedberga38528f2011-01-22 06:46:43 +0200329 rp_len = sizeof(*rp) + (2 * count);
330 rp = kmalloc(rp_len, GFP_ATOMIC);
331 if (!rp) {
Jesper Juhlb2c60d42011-01-14 00:18:49 +0100332 read_unlock(&hci_dev_list_lock);
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200333 return -ENOMEM;
Jesper Juhlb2c60d42011-01-14 00:18:49 +0100334 }
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200335
Johan Hedberg476e44c2012-10-19 20:10:46 +0300336 count = 0;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +0200337 list_for_each_entry(d, &hci_dev_list, list) {
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200338 if (test_bit(HCI_SETUP, &d->dev_flags))
Johan Hedbergab81cbf2010-12-15 13:53:18 +0200339 continue;
340
Marcel Holtmann0736cfa2013-08-26 21:40:51 -0700341 if (test_bit(HCI_USER_CHANNEL, &d->dev_flags))
342 continue;
343
Marcel Holtmann1514b892013-10-06 08:25:01 -0700344 if (d->dev_type == HCI_BREDR) {
345 rp->index[count++] = cpu_to_le16(d->id);
346 BT_DBG("Added hci%u", d->id);
347 }
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200348 }
349
Johan Hedberg476e44c2012-10-19 20:10:46 +0300350 rp->num_controllers = cpu_to_le16(count);
351 rp_len = sizeof(*rp) + (2 * count);
352
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200353 read_unlock(&hci_dev_list_lock);
354
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200355 err = cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_INDEX_LIST, 0, rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300356 rp_len);
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200357
Johan Hedberga38528f2011-01-22 06:46:43 +0200358 kfree(rp);
359
360 return err;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200361}
362
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200363static u32 get_supported_settings(struct hci_dev *hdev)
Johan Hedberg03811012010-12-08 00:21:06 +0200364{
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200365 u32 settings = 0;
Johan Hedberg03811012010-12-08 00:21:06 +0200366
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200367 settings |= MGMT_SETTING_POWERED;
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200368 settings |= MGMT_SETTING_PAIRABLE;
Marcel Holtmannb1de97d2014-01-31 11:55:21 -0800369 settings |= MGMT_SETTING_DEBUG_KEYS;
Johan Hedberg03811012010-12-08 00:21:06 +0200370
Andre Guedesed3fa312012-07-24 15:03:46 -0300371 if (lmp_bredr_capable(hdev)) {
Johan Hedberg33c525c2012-10-24 21:11:58 +0300372 settings |= MGMT_SETTING_CONNECTABLE;
Johan Hedberg1a47aee2013-03-15 17:07:06 -0500373 if (hdev->hci_ver >= BLUETOOTH_VER_1_2)
374 settings |= MGMT_SETTING_FAST_CONNECTABLE;
Johan Hedberg33c525c2012-10-24 21:11:58 +0300375 settings |= MGMT_SETTING_DISCOVERABLE;
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200376 settings |= MGMT_SETTING_BREDR;
377 settings |= MGMT_SETTING_LINK_SECURITY;
Marcel Holtmanna82974c2013-10-11 09:48:47 -0700378
379 if (lmp_ssp_capable(hdev)) {
380 settings |= MGMT_SETTING_SSP;
381 settings |= MGMT_SETTING_HS;
382 }
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800383
Marcel Holtmann5afeac142014-01-10 02:07:27 -0800384 if (lmp_sc_capable(hdev) ||
385 test_bit(HCI_FORCE_SC, &hdev->dev_flags))
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800386 settings |= MGMT_SETTING_SECURE_CONN;
Marcel Holtmann848566b2013-10-01 22:59:22 -0700387 }
Marcel Holtmannd7b7e792012-02-20 21:47:49 +0100388
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300389 if (lmp_le_capable(hdev)) {
Marcel Holtmann9d428202012-05-03 07:12:31 +0200390 settings |= MGMT_SETTING_LE;
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300391 settings |= MGMT_SETTING_ADVERTISING;
392 }
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200393
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200394 return settings;
395}
Johan Hedbergebc99fe2011-01-04 11:54:26 +0200396
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200397static u32 get_current_settings(struct hci_dev *hdev)
398{
399 u32 settings = 0;
Johan Hedbergdc4fe302011-03-16 14:29:36 +0200400
Johan Hedbergf1f0eb02012-02-21 17:15:41 +0200401 if (hdev_is_powered(hdev))
Marcel Holtmannf0d4b782012-02-21 12:14:25 +0100402 settings |= MGMT_SETTING_POWERED;
403
Johan Hedberg5e5282b2012-02-21 16:01:30 +0200404 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200405 settings |= MGMT_SETTING_CONNECTABLE;
406
Johan Hedberg1a4d3c42013-03-15 17:07:08 -0500407 if (test_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags))
408 settings |= MGMT_SETTING_FAST_CONNECTABLE;
409
Johan Hedberg5e5282b2012-02-21 16:01:30 +0200410 if (test_bit(HCI_DISCOVERABLE, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200411 settings |= MGMT_SETTING_DISCOVERABLE;
412
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200413 if (test_bit(HCI_PAIRABLE, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200414 settings |= MGMT_SETTING_PAIRABLE;
415
Johan Hedberg56f87902013-10-02 13:43:13 +0300416 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200417 settings |= MGMT_SETTING_BREDR;
418
Johan Hedberg06199cf2012-02-22 16:37:11 +0200419 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200420 settings |= MGMT_SETTING_LE;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200421
Johan Hedberg47990ea2012-02-22 11:58:37 +0200422 if (test_bit(HCI_LINK_SECURITY, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200423 settings |= MGMT_SETTING_LINK_SECURITY;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200424
Johan Hedberg84bde9d2012-01-25 14:21:06 +0200425 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200426 settings |= MGMT_SETTING_SSP;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200427
Johan Hedberg6d80dfd2012-02-20 23:50:38 +0200428 if (test_bit(HCI_HS_ENABLED, &hdev->dev_flags))
429 settings |= MGMT_SETTING_HS;
430
Johan Hedbergf3d3444a2013-10-05 12:01:04 +0200431 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300432 settings |= MGMT_SETTING_ADVERTISING;
433
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800434 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags))
435 settings |= MGMT_SETTING_SECURE_CONN;
436
Marcel Holtmannb1de97d2014-01-31 11:55:21 -0800437 if (test_bit(HCI_DEBUG_KEYS, &hdev->dev_flags))
438 settings |= MGMT_SETTING_DEBUG_KEYS;
439
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200440 return settings;
Johan Hedbergc542a062011-01-26 13:11:03 +0200441}
442
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300443#define PNP_INFO_SVCLASS_ID 0x1200
444
Johan Hedberg213202e2013-01-27 00:31:33 +0200445static u8 *create_uuid16_list(struct hci_dev *hdev, u8 *data, ptrdiff_t len)
446{
447 u8 *ptr = data, *uuids_start = NULL;
448 struct bt_uuid *uuid;
449
450 if (len < 4)
451 return ptr;
452
453 list_for_each_entry(uuid, &hdev->uuids, list) {
454 u16 uuid16;
455
456 if (uuid->size != 16)
457 continue;
458
459 uuid16 = get_unaligned_le16(&uuid->uuid[12]);
460 if (uuid16 < 0x1100)
461 continue;
462
463 if (uuid16 == PNP_INFO_SVCLASS_ID)
464 continue;
465
466 if (!uuids_start) {
467 uuids_start = ptr;
468 uuids_start[0] = 1;
469 uuids_start[1] = EIR_UUID16_ALL;
470 ptr += 2;
471 }
472
473 /* Stop if not enough space to put next UUID */
474 if ((ptr - data) + sizeof(u16) > len) {
475 uuids_start[1] = EIR_UUID16_SOME;
476 break;
477 }
478
479 *ptr++ = (uuid16 & 0x00ff);
480 *ptr++ = (uuid16 & 0xff00) >> 8;
481 uuids_start[0] += sizeof(uuid16);
482 }
483
484 return ptr;
485}
486
Johan Hedbergcdf19632013-01-27 00:31:34 +0200487static u8 *create_uuid32_list(struct hci_dev *hdev, u8 *data, ptrdiff_t len)
488{
489 u8 *ptr = data, *uuids_start = NULL;
490 struct bt_uuid *uuid;
491
492 if (len < 6)
493 return ptr;
494
495 list_for_each_entry(uuid, &hdev->uuids, list) {
496 if (uuid->size != 32)
497 continue;
498
499 if (!uuids_start) {
500 uuids_start = ptr;
501 uuids_start[0] = 1;
502 uuids_start[1] = EIR_UUID32_ALL;
503 ptr += 2;
504 }
505
506 /* Stop if not enough space to put next UUID */
507 if ((ptr - data) + sizeof(u32) > len) {
508 uuids_start[1] = EIR_UUID32_SOME;
509 break;
510 }
511
512 memcpy(ptr, &uuid->uuid[12], sizeof(u32));
513 ptr += sizeof(u32);
514 uuids_start[0] += sizeof(u32);
515 }
516
517 return ptr;
518}
519
Johan Hedbergc00d5752013-01-27 00:31:35 +0200520static u8 *create_uuid128_list(struct hci_dev *hdev, u8 *data, ptrdiff_t len)
521{
522 u8 *ptr = data, *uuids_start = NULL;
523 struct bt_uuid *uuid;
524
525 if (len < 18)
526 return ptr;
527
528 list_for_each_entry(uuid, &hdev->uuids, list) {
529 if (uuid->size != 128)
530 continue;
531
532 if (!uuids_start) {
533 uuids_start = ptr;
534 uuids_start[0] = 1;
535 uuids_start[1] = EIR_UUID128_ALL;
536 ptr += 2;
537 }
538
539 /* Stop if not enough space to put next UUID */
540 if ((ptr - data) + 16 > len) {
541 uuids_start[1] = EIR_UUID128_SOME;
542 break;
543 }
544
545 memcpy(ptr, uuid->uuid, 16);
546 ptr += 16;
547 uuids_start[0] += 16;
548 }
549
550 return ptr;
551}
552
Johan Hedbergeb2a8d22013-10-19 23:38:20 +0300553static struct pending_cmd *mgmt_pending_find(u16 opcode, struct hci_dev *hdev)
554{
555 struct pending_cmd *cmd;
556
557 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
558 if (cmd->opcode == opcode)
559 return cmd;
560 }
561
562 return NULL;
563}
564
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700565static u8 create_scan_rsp_data(struct hci_dev *hdev, u8 *ptr)
566{
Marcel Holtmann7a5f4992013-10-16 00:16:49 -0700567 u8 ad_len = 0;
568 size_t name_len;
569
570 name_len = strlen(hdev->dev_name);
571 if (name_len > 0) {
572 size_t max_len = HCI_MAX_AD_LENGTH - ad_len - 2;
573
574 if (name_len > max_len) {
575 name_len = max_len;
576 ptr[1] = EIR_NAME_SHORT;
577 } else
578 ptr[1] = EIR_NAME_COMPLETE;
579
580 ptr[0] = name_len + 1;
581
582 memcpy(ptr + 2, hdev->dev_name, name_len);
583
584 ad_len += (name_len + 2);
585 ptr += (name_len + 2);
586 }
587
588 return ad_len;
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700589}
590
591static void update_scan_rsp_data(struct hci_request *req)
592{
593 struct hci_dev *hdev = req->hdev;
594 struct hci_cp_le_set_scan_rsp_data cp;
595 u8 len;
596
Johan Hedberg7751ef12013-10-19 23:38:15 +0300597 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700598 return;
599
600 memset(&cp, 0, sizeof(cp));
601
602 len = create_scan_rsp_data(hdev, cp.data);
603
Johan Hedbergeb438b52013-10-16 15:31:07 +0300604 if (hdev->scan_rsp_data_len == len &&
605 memcmp(cp.data, hdev->scan_rsp_data, len) == 0)
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700606 return;
607
Johan Hedbergeb438b52013-10-16 15:31:07 +0300608 memcpy(hdev->scan_rsp_data, cp.data, sizeof(cp.data));
609 hdev->scan_rsp_data_len = len;
Marcel Holtmannf14d8f62013-10-16 00:16:48 -0700610
611 cp.length = len;
612
613 hci_req_add(req, HCI_OP_LE_SET_SCAN_RSP_DATA, sizeof(cp), &cp);
614}
615
Johan Hedberg9a43e252013-10-20 19:00:07 +0300616static u8 get_adv_discov_flags(struct hci_dev *hdev)
617{
618 struct pending_cmd *cmd;
619
620 /* If there's a pending mgmt command the flags will not yet have
621 * their final values, so check for this first.
622 */
623 cmd = mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev);
624 if (cmd) {
625 struct mgmt_mode *cp = cmd->param;
626 if (cp->val == 0x01)
627 return LE_AD_GENERAL;
628 else if (cp->val == 0x02)
629 return LE_AD_LIMITED;
630 } else {
631 if (test_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags))
632 return LE_AD_LIMITED;
633 else if (test_bit(HCI_DISCOVERABLE, &hdev->dev_flags))
634 return LE_AD_GENERAL;
635 }
636
637 return 0;
638}
639
Marcel Holtmann46cad2e2013-10-16 00:16:46 -0700640static u8 create_adv_data(struct hci_dev *hdev, u8 *ptr)
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700641{
642 u8 ad_len = 0, flags = 0;
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700643
Johan Hedberg9a43e252013-10-20 19:00:07 +0300644 flags |= get_adv_discov_flags(hdev);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700645
Johan Hedberge8340042014-01-30 11:16:50 -0800646 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700647 flags |= LE_AD_NO_BREDR;
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700648
649 if (flags) {
650 BT_DBG("adv flags 0x%02x", flags);
651
652 ptr[0] = 2;
653 ptr[1] = EIR_FLAGS;
654 ptr[2] = flags;
655
656 ad_len += 3;
657 ptr += 3;
658 }
659
660 if (hdev->adv_tx_power != HCI_TX_POWER_INVALID) {
661 ptr[0] = 2;
662 ptr[1] = EIR_TX_POWER;
663 ptr[2] = (u8) hdev->adv_tx_power;
664
665 ad_len += 3;
666 ptr += 3;
667 }
668
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700669 return ad_len;
670}
671
Marcel Holtmann5947f4b2013-10-16 00:16:50 -0700672static void update_adv_data(struct hci_request *req)
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700673{
674 struct hci_dev *hdev = req->hdev;
675 struct hci_cp_le_set_adv_data cp;
676 u8 len;
677
Johan Hedberg10994ce2013-10-19 23:38:16 +0300678 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700679 return;
680
681 memset(&cp, 0, sizeof(cp));
682
Marcel Holtmann46cad2e2013-10-16 00:16:46 -0700683 len = create_adv_data(hdev, cp.data);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -0700684
685 if (hdev->adv_data_len == len &&
686 memcmp(cp.data, hdev->adv_data, len) == 0)
687 return;
688
689 memcpy(hdev->adv_data, cp.data, sizeof(cp.data));
690 hdev->adv_data_len = len;
691
692 cp.length = len;
693
694 hci_req_add(req, HCI_OP_LE_SET_ADV_DATA, sizeof(cp), &cp);
695}
696
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300697static void create_eir(struct hci_dev *hdev, u8 *data)
698{
699 u8 *ptr = data;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300700 size_t name_len;
701
702 name_len = strlen(hdev->dev_name);
703
704 if (name_len > 0) {
705 /* EIR Data type */
706 if (name_len > 48) {
707 name_len = 48;
708 ptr[1] = EIR_NAME_SHORT;
709 } else
710 ptr[1] = EIR_NAME_COMPLETE;
711
712 /* EIR Data length */
713 ptr[0] = name_len + 1;
714
715 memcpy(ptr + 2, hdev->dev_name, name_len);
716
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300717 ptr += (name_len + 2);
718 }
719
Johan Hedbergbbaf4442012-11-08 01:22:59 +0100720 if (hdev->inq_tx_power != HCI_TX_POWER_INVALID) {
Marcel Holtmann91c4e9b2012-03-11 19:27:21 -0700721 ptr[0] = 2;
722 ptr[1] = EIR_TX_POWER;
723 ptr[2] = (u8) hdev->inq_tx_power;
724
Marcel Holtmann91c4e9b2012-03-11 19:27:21 -0700725 ptr += 3;
726 }
727
Marcel Holtmann2b9be132012-03-11 19:32:12 -0700728 if (hdev->devid_source > 0) {
729 ptr[0] = 9;
730 ptr[1] = EIR_DEVICE_ID;
731
732 put_unaligned_le16(hdev->devid_source, ptr + 2);
733 put_unaligned_le16(hdev->devid_vendor, ptr + 4);
734 put_unaligned_le16(hdev->devid_product, ptr + 6);
735 put_unaligned_le16(hdev->devid_version, ptr + 8);
736
Marcel Holtmann2b9be132012-03-11 19:32:12 -0700737 ptr += 10;
738 }
739
Johan Hedberg213202e2013-01-27 00:31:33 +0200740 ptr = create_uuid16_list(hdev, ptr, HCI_MAX_EIR_LENGTH - (ptr - data));
Johan Hedbergcdf19632013-01-27 00:31:34 +0200741 ptr = create_uuid32_list(hdev, ptr, HCI_MAX_EIR_LENGTH - (ptr - data));
Johan Hedbergc00d5752013-01-27 00:31:35 +0200742 ptr = create_uuid128_list(hdev, ptr, HCI_MAX_EIR_LENGTH - (ptr - data));
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300743}
744
Johan Hedberg890ea892013-03-15 17:06:52 -0500745static void update_eir(struct hci_request *req)
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300746{
Johan Hedberg890ea892013-03-15 17:06:52 -0500747 struct hci_dev *hdev = req->hdev;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300748 struct hci_cp_write_eir cp;
749
Johan Hedberg504c8dc2012-02-23 13:30:41 +0200750 if (!hdev_is_powered(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -0500751 return;
Johan Hedberg7770c4a2012-02-22 22:06:38 +0200752
Johan Hedberg976eb202012-10-24 21:12:01 +0300753 if (!lmp_ext_inq_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -0500754 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300755
Johan Hedberg84bde9d2012-01-25 14:21:06 +0200756 if (!test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -0500757 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300758
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200759 if (test_bit(HCI_SERVICE_CACHE, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -0500760 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300761
762 memset(&cp, 0, sizeof(cp));
763
764 create_eir(hdev, cp.data);
765
766 if (memcmp(cp.data, hdev->eir, sizeof(cp.data)) == 0)
Johan Hedberg890ea892013-03-15 17:06:52 -0500767 return;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300768
769 memcpy(hdev->eir, cp.data, sizeof(cp.data));
770
Johan Hedberg890ea892013-03-15 17:06:52 -0500771 hci_req_add(req, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
Johan Hedberg80a1e1d2011-03-28 14:07:23 +0300772}
773
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200774static u8 get_service_classes(struct hci_dev *hdev)
775{
Gustavo F. Padovan12dc0742011-10-14 19:32:56 -0300776 struct bt_uuid *uuid;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200777 u8 val = 0;
778
Gustavo F. Padovan12dc0742011-10-14 19:32:56 -0300779 list_for_each_entry(uuid, &hdev->uuids, list)
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200780 val |= uuid->svc_hint;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200781
782 return val;
783}
784
Johan Hedberg890ea892013-03-15 17:06:52 -0500785static void update_class(struct hci_request *req)
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200786{
Johan Hedberg890ea892013-03-15 17:06:52 -0500787 struct hci_dev *hdev = req->hdev;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200788 u8 cod[3];
789
790 BT_DBG("%s", hdev->name);
791
Johan Hedberg504c8dc2012-02-23 13:30:41 +0200792 if (!hdev_is_powered(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -0500793 return;
Johan Hedberg7770c4a2012-02-22 22:06:38 +0200794
Johan Hedbergf87ea1d2013-10-19 23:38:17 +0300795 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
796 return;
797
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200798 if (test_bit(HCI_SERVICE_CACHE, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -0500799 return;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200800
801 cod[0] = hdev->minor_class;
802 cod[1] = hdev->major_class;
803 cod[2] = get_service_classes(hdev);
804
Marcel Holtmann6acd7db2013-10-15 06:33:53 -0700805 if (test_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags))
806 cod[1] |= 0x20;
807
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200808 if (memcmp(cod, hdev->dev_class, 3) == 0)
Johan Hedberg890ea892013-03-15 17:06:52 -0500809 return;
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200810
Johan Hedberg890ea892013-03-15 17:06:52 -0500811 hci_req_add(req, HCI_OP_WRITE_CLASS_OF_DEV, sizeof(cod), cod);
Johan Hedberg1aff6f02011-01-13 21:56:52 +0200812}
813
Johan Hedberg7d785252011-12-15 00:47:39 +0200814static void service_cache_off(struct work_struct *work)
815{
816 struct hci_dev *hdev = container_of(work, struct hci_dev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300817 service_cache.work);
Johan Hedberg890ea892013-03-15 17:06:52 -0500818 struct hci_request req;
Johan Hedberg7d785252011-12-15 00:47:39 +0200819
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200820 if (!test_and_clear_bit(HCI_SERVICE_CACHE, &hdev->dev_flags))
Johan Hedberg7d785252011-12-15 00:47:39 +0200821 return;
822
Johan Hedberg890ea892013-03-15 17:06:52 -0500823 hci_req_init(&req, hdev);
824
Johan Hedberg7d785252011-12-15 00:47:39 +0200825 hci_dev_lock(hdev);
826
Johan Hedberg890ea892013-03-15 17:06:52 -0500827 update_eir(&req);
828 update_class(&req);
Johan Hedberg7d785252011-12-15 00:47:39 +0200829
830 hci_dev_unlock(hdev);
Johan Hedberg890ea892013-03-15 17:06:52 -0500831
832 hci_req_run(&req, NULL);
Johan Hedberg7d785252011-12-15 00:47:39 +0200833}
834
Johan Hedberg6a919082012-02-28 06:17:26 +0200835static void mgmt_init_hdev(struct sock *sk, struct hci_dev *hdev)
Johan Hedberg7d785252011-12-15 00:47:39 +0200836{
Johan Hedberg4f87da82012-03-02 19:55:56 +0200837 if (test_and_set_bit(HCI_MGMT, &hdev->dev_flags))
Johan Hedberg6a919082012-02-28 06:17:26 +0200838 return;
839
Johan Hedberg4f87da82012-03-02 19:55:56 +0200840 INIT_DELAYED_WORK(&hdev->service_cache, service_cache_off);
Johan Hedberg7d785252011-12-15 00:47:39 +0200841
Johan Hedberg4f87da82012-03-02 19:55:56 +0200842 /* Non-mgmt controlled devices get this bit set
843 * implicitly so that pairing works for them, however
844 * for mgmt we require user-space to explicitly enable
845 * it
846 */
847 clear_bit(HCI_PAIRABLE, &hdev->dev_flags);
Johan Hedberg7d785252011-12-15 00:47:39 +0200848}
849
Johan Hedberg0f4e68c2012-02-28 17:18:30 +0200850static int read_controller_info(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300851 void *data, u16 data_len)
Johan Hedberg03811012010-12-08 00:21:06 +0200852{
853 struct mgmt_rp_read_info rp;
Johan Hedberg03811012010-12-08 00:21:06 +0200854
Johan Hedbergbdb6d972012-02-28 06:13:32 +0200855 BT_DBG("sock %p %s", sk, hdev->name);
Johan Hedberg03811012010-12-08 00:21:06 +0200856
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300857 hci_dev_lock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200858
Johan Hedberg03811012010-12-08 00:21:06 +0200859 memset(&rp, 0, sizeof(rp));
860
Johan Hedberg03811012010-12-08 00:21:06 +0200861 bacpy(&rp.bdaddr, &hdev->bdaddr);
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200862
863 rp.version = hdev->hci_ver;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200864 rp.manufacturer = cpu_to_le16(hdev->manufacturer);
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200865
866 rp.supported_settings = cpu_to_le32(get_supported_settings(hdev));
867 rp.current_settings = cpu_to_le32(get_current_settings(hdev));
868
869 memcpy(rp.dev_class, hdev->dev_class, 3);
Johan Hedberg03811012010-12-08 00:21:06 +0200870
871 memcpy(rp.name, hdev->dev_name, sizeof(hdev->dev_name));
Johan Hedberg27fcc362012-02-22 21:46:22 +0200872 memcpy(rp.short_name, hdev->short_name, sizeof(hdev->short_name));
Johan Hedberg03811012010-12-08 00:21:06 +0200873
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300874 hci_dev_unlock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200875
Johan Hedbergbdb6d972012-02-28 06:13:32 +0200876 return cmd_complete(sk, hdev->id, MGMT_OP_READ_INFO, 0, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300877 sizeof(rp));
Johan Hedberg03811012010-12-08 00:21:06 +0200878}
879
880static void mgmt_pending_free(struct pending_cmd *cmd)
881{
882 sock_put(cmd->sk);
883 kfree(cmd->param);
884 kfree(cmd);
885}
886
887static struct pending_cmd *mgmt_pending_add(struct sock *sk, u16 opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300888 struct hci_dev *hdev, void *data,
889 u16 len)
Johan Hedberg03811012010-12-08 00:21:06 +0200890{
891 struct pending_cmd *cmd;
892
Andre Guedes12b94562012-06-07 19:05:45 -0300893 cmd = kmalloc(sizeof(*cmd), GFP_KERNEL);
Johan Hedberg03811012010-12-08 00:21:06 +0200894 if (!cmd)
895 return NULL;
896
897 cmd->opcode = opcode;
898 cmd->index = hdev->id;
899
Andre Guedes12b94562012-06-07 19:05:45 -0300900 cmd->param = kmalloc(len, GFP_KERNEL);
Johan Hedberg03811012010-12-08 00:21:06 +0200901 if (!cmd->param) {
902 kfree(cmd);
903 return NULL;
904 }
905
906 if (data)
907 memcpy(cmd->param, data, len);
908
909 cmd->sk = sk;
910 sock_hold(sk);
911
912 list_add(&cmd->list, &hdev->mgmt_pending);
913
914 return cmd;
915}
916
917static void mgmt_pending_foreach(u16 opcode, struct hci_dev *hdev,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -0300918 void (*cb)(struct pending_cmd *cmd,
919 void *data),
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300920 void *data)
Johan Hedberg03811012010-12-08 00:21:06 +0200921{
Andre Guedesa3d09352013-02-01 11:21:30 -0300922 struct pending_cmd *cmd, *tmp;
Johan Hedberg03811012010-12-08 00:21:06 +0200923
Andre Guedesa3d09352013-02-01 11:21:30 -0300924 list_for_each_entry_safe(cmd, tmp, &hdev->mgmt_pending, list) {
Johan Hedberg03811012010-12-08 00:21:06 +0200925 if (opcode > 0 && cmd->opcode != opcode)
926 continue;
927
928 cb(cmd, data);
929 }
930}
931
Johan Hedberg03811012010-12-08 00:21:06 +0200932static void mgmt_pending_remove(struct pending_cmd *cmd)
933{
934 list_del(&cmd->list);
935 mgmt_pending_free(cmd);
936}
937
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200938static int send_settings_rsp(struct sock *sk, u16 opcode, struct hci_dev *hdev)
Johan Hedberg86805702011-11-11 16:18:52 +0200939{
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200940 __le32 settings = cpu_to_le32(get_current_settings(hdev));
Johan Hedberg86805702011-11-11 16:18:52 +0200941
Johan Hedbergaee9b2182012-02-18 15:07:59 +0200942 return cmd_complete(sk, hdev->id, opcode, 0, &settings,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300943 sizeof(settings));
Johan Hedberg86805702011-11-11 16:18:52 +0200944}
945
Johan Hedbergbdb6d972012-02-28 06:13:32 +0200946static int set_powered(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300947 u16 len)
Johan Hedberg03811012010-12-08 00:21:06 +0200948{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -0300949 struct mgmt_mode *cp = data;
Johan Hedberg03811012010-12-08 00:21:06 +0200950 struct pending_cmd *cmd;
Johan Hedberg4b34ee782012-02-21 14:13:02 +0200951 int err;
Johan Hedberg03811012010-12-08 00:21:06 +0200952
Johan Hedbergbdb6d972012-02-28 06:13:32 +0200953 BT_DBG("request for %s", hdev->name);
Johan Hedberg03811012010-12-08 00:21:06 +0200954
Johan Hedberga7e80f22013-01-09 16:05:19 +0200955 if (cp->val != 0x00 && cp->val != 0x01)
956 return cmd_status(sk, hdev->id, MGMT_OP_SET_POWERED,
957 MGMT_STATUS_INVALID_PARAMS);
958
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300959 hci_dev_lock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200960
Johan Hedberg87b95ba2013-09-25 13:26:06 +0300961 if (mgmt_pending_find(MGMT_OP_SET_POWERED, hdev)) {
962 err = cmd_status(sk, hdev->id, MGMT_OP_SET_POWERED,
963 MGMT_STATUS_BUSY);
964 goto failed;
965 }
966
Marcel Holtmannf0d4b782012-02-21 12:14:25 +0100967 if (test_and_clear_bit(HCI_AUTO_OFF, &hdev->dev_flags)) {
968 cancel_delayed_work(&hdev->power_off);
969
970 if (cp->val) {
Johan Hedberga1d70452013-01-09 15:29:40 +0200971 mgmt_pending_add(sk, MGMT_OP_SET_POWERED, hdev,
972 data, len);
973 err = mgmt_powered(hdev, 1);
Marcel Holtmannf0d4b782012-02-21 12:14:25 +0100974 goto failed;
975 }
976 }
977
Johan Hedberg4b34ee782012-02-21 14:13:02 +0200978 if (!!cp->val == hdev_is_powered(hdev)) {
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200979 err = send_settings_rsp(sk, MGMT_OP_SET_POWERED, hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200980 goto failed;
981 }
982
Johan Hedberg03811012010-12-08 00:21:06 +0200983 cmd = mgmt_pending_add(sk, MGMT_OP_SET_POWERED, hdev, data, len);
984 if (!cmd) {
985 err = -ENOMEM;
986 goto failed;
987 }
988
989 if (cp->val)
Johan Hedberg19202572013-01-14 22:33:51 +0200990 queue_work(hdev->req_workqueue, &hdev->power_on);
Johan Hedberg03811012010-12-08 00:21:06 +0200991 else
Johan Hedberg19202572013-01-14 22:33:51 +0200992 queue_work(hdev->req_workqueue, &hdev->power_off.work);
Johan Hedberg03811012010-12-08 00:21:06 +0200993
994 err = 0;
995
996failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300997 hci_dev_unlock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200998 return err;
999}
1000
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001001static int mgmt_event(u16 event, struct hci_dev *hdev, void *data, u16 data_len,
1002 struct sock *skip_sk)
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001003{
1004 struct sk_buff *skb;
1005 struct mgmt_hdr *hdr;
1006
Andre Guedes790eff42012-06-07 19:05:46 -03001007 skb = alloc_skb(sizeof(*hdr) + data_len, GFP_KERNEL);
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001008 if (!skb)
1009 return -ENOMEM;
1010
1011 hdr = (void *) skb_put(skb, sizeof(*hdr));
1012 hdr->opcode = cpu_to_le16(event);
1013 if (hdev)
1014 hdr->index = cpu_to_le16(hdev->id);
1015 else
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05301016 hdr->index = __constant_cpu_to_le16(MGMT_INDEX_NONE);
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001017 hdr->len = cpu_to_le16(data_len);
1018
1019 if (data)
1020 memcpy(skb_put(skb, data_len), data, data_len);
1021
Marcel Holtmann97e0bde2012-02-22 13:49:28 +01001022 /* Time stamp */
1023 __net_timestamp(skb);
1024
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001025 hci_send_to_control(skb, skip_sk);
1026 kfree_skb(skb);
1027
1028 return 0;
1029}
1030
1031static int new_settings(struct hci_dev *hdev, struct sock *skip)
1032{
1033 __le32 ev;
1034
1035 ev = cpu_to_le32(get_current_settings(hdev));
1036
1037 return mgmt_event(MGMT_EV_NEW_SETTINGS, hdev, &ev, sizeof(ev), skip);
1038}
1039
Johan Hedbergbd99abd2013-09-25 13:26:07 +03001040struct cmd_lookup {
1041 struct sock *sk;
1042 struct hci_dev *hdev;
1043 u8 mgmt_status;
1044};
1045
1046static void settings_rsp(struct pending_cmd *cmd, void *data)
1047{
1048 struct cmd_lookup *match = data;
1049
1050 send_settings_rsp(cmd->sk, cmd->opcode, match->hdev);
1051
1052 list_del(&cmd->list);
1053
1054 if (match->sk == NULL) {
1055 match->sk = cmd->sk;
1056 sock_hold(match->sk);
1057 }
1058
1059 mgmt_pending_free(cmd);
1060}
1061
1062static void cmd_status_rsp(struct pending_cmd *cmd, void *data)
1063{
1064 u8 *status = data;
1065
1066 cmd_status(cmd->sk, cmd->index, cmd->opcode, *status);
1067 mgmt_pending_remove(cmd);
1068}
1069
Johan Hedberge6fe7982013-10-02 15:45:22 +03001070static u8 mgmt_bredr_support(struct hci_dev *hdev)
1071{
1072 if (!lmp_bredr_capable(hdev))
1073 return MGMT_STATUS_NOT_SUPPORTED;
1074 else if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
1075 return MGMT_STATUS_REJECTED;
1076 else
1077 return MGMT_STATUS_SUCCESS;
1078}
1079
1080static u8 mgmt_le_support(struct hci_dev *hdev)
1081{
1082 if (!lmp_le_capable(hdev))
1083 return MGMT_STATUS_NOT_SUPPORTED;
1084 else if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
1085 return MGMT_STATUS_REJECTED;
1086 else
1087 return MGMT_STATUS_SUCCESS;
1088}
1089
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001090static void set_discoverable_complete(struct hci_dev *hdev, u8 status)
1091{
1092 struct pending_cmd *cmd;
1093 struct mgmt_mode *cp;
Marcel Holtmann970ba522013-10-15 06:33:57 -07001094 struct hci_request req;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001095 bool changed;
1096
1097 BT_DBG("status 0x%02x", status);
1098
1099 hci_dev_lock(hdev);
1100
1101 cmd = mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev);
1102 if (!cmd)
1103 goto unlock;
1104
1105 if (status) {
1106 u8 mgmt_err = mgmt_status(status);
1107 cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001108 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001109 goto remove_cmd;
1110 }
1111
1112 cp = cmd->param;
Marcel Holtmannd4462a02013-10-15 08:11:02 -07001113 if (cp->val) {
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001114 changed = !test_and_set_bit(HCI_DISCOVERABLE,
1115 &hdev->dev_flags);
Marcel Holtmannd4462a02013-10-15 08:11:02 -07001116
1117 if (hdev->discov_timeout > 0) {
1118 int to = msecs_to_jiffies(hdev->discov_timeout * 1000);
1119 queue_delayed_work(hdev->workqueue, &hdev->discov_off,
1120 to);
1121 }
1122 } else {
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001123 changed = test_and_clear_bit(HCI_DISCOVERABLE,
1124 &hdev->dev_flags);
Marcel Holtmannd4462a02013-10-15 08:11:02 -07001125 }
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001126
1127 send_settings_rsp(cmd->sk, MGMT_OP_SET_DISCOVERABLE, hdev);
1128
1129 if (changed)
1130 new_settings(hdev, cmd->sk);
1131
Marcel Holtmann970ba522013-10-15 06:33:57 -07001132 /* When the discoverable mode gets changed, make sure
1133 * that class of device has the limited discoverable
1134 * bit correctly set.
1135 */
1136 hci_req_init(&req, hdev);
1137 update_class(&req);
1138 hci_req_run(&req, NULL);
1139
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001140remove_cmd:
1141 mgmt_pending_remove(cmd);
1142
1143unlock:
1144 hci_dev_unlock(hdev);
1145}
1146
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001147static int set_discoverable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001148 u16 len)
Johan Hedberg03811012010-12-08 00:21:06 +02001149{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001150 struct mgmt_cp_set_discoverable *cp = data;
Johan Hedberg03811012010-12-08 00:21:06 +02001151 struct pending_cmd *cmd;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001152 struct hci_request req;
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001153 u16 timeout;
Johan Hedberg9a43e252013-10-20 19:00:07 +03001154 u8 scan;
Johan Hedberg03811012010-12-08 00:21:06 +02001155 int err;
1156
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001157 BT_DBG("request for %s", hdev->name);
Johan Hedberg03811012010-12-08 00:21:06 +02001158
Johan Hedberg9a43e252013-10-20 19:00:07 +03001159 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags) &&
1160 !test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedberg33c525c2012-10-24 21:11:58 +03001161 return cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Johan Hedberg9a43e252013-10-20 19:00:07 +03001162 MGMT_STATUS_REJECTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001163
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001164 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga7e80f22013-01-09 16:05:19 +02001165 return cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1166 MGMT_STATUS_INVALID_PARAMS);
1167
Marcel Holtmann1f350c82012-03-12 20:31:08 -07001168 timeout = __le16_to_cpu(cp->timeout);
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001169
1170 /* Disabling discoverable requires that no timeout is set,
1171 * and enabling limited discoverable requires a timeout.
1172 */
1173 if ((cp->val == 0x00 && timeout > 0) ||
1174 (cp->val == 0x02 && timeout == 0))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001175 return cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001176 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001177
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001178 hci_dev_lock(hdev);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001179
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001180 if (!hdev_is_powered(hdev) && timeout > 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001181 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001182 MGMT_STATUS_NOT_POWERED);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001183 goto failed;
1184 }
1185
1186 if (mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03001187 mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001188 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001189 MGMT_STATUS_BUSY);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001190 goto failed;
1191 }
1192
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001193 if (!test_bit(HCI_CONNECTABLE, &hdev->dev_flags)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001194 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001195 MGMT_STATUS_REJECTED);
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001196 goto failed;
1197 }
1198
1199 if (!hdev_is_powered(hdev)) {
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001200 bool changed = false;
1201
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001202 /* Setting limited discoverable when powered off is
1203 * not a valid operation since it requires a timeout
1204 * and so no need to check HCI_LIMITED_DISCOVERABLE.
1205 */
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001206 if (!!cp->val != test_bit(HCI_DISCOVERABLE, &hdev->dev_flags)) {
1207 change_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
1208 changed = true;
1209 }
1210
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001211 err = send_settings_rsp(sk, MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001212 if (err < 0)
1213 goto failed;
1214
1215 if (changed)
1216 err = new_settings(hdev, sk);
1217
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001218 goto failed;
1219 }
1220
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001221 /* If the current mode is the same, then just update the timeout
1222 * value with the new value. And if only the timeout gets updated,
1223 * then no need for any HCI transactions.
1224 */
1225 if (!!cp->val == test_bit(HCI_DISCOVERABLE, &hdev->dev_flags) &&
1226 (cp->val == 0x02) == test_bit(HCI_LIMITED_DISCOVERABLE,
1227 &hdev->dev_flags)) {
Marcel Holtmann36261542013-10-15 08:28:51 -07001228 cancel_delayed_work(&hdev->discov_off);
1229 hdev->discov_timeout = timeout;
Marcel Holtmann955638e2012-02-22 18:21:00 +01001230
Marcel Holtmann36261542013-10-15 08:28:51 -07001231 if (cp->val && hdev->discov_timeout > 0) {
1232 int to = msecs_to_jiffies(hdev->discov_timeout * 1000);
Marcel Holtmann955638e2012-02-22 18:21:00 +01001233 queue_delayed_work(hdev->workqueue, &hdev->discov_off,
Marcel Holtmann36261542013-10-15 08:28:51 -07001234 to);
Marcel Holtmann955638e2012-02-22 18:21:00 +01001235 }
1236
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001237 err = send_settings_rsp(sk, MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001238 goto failed;
1239 }
1240
1241 cmd = mgmt_pending_add(sk, MGMT_OP_SET_DISCOVERABLE, hdev, data, len);
1242 if (!cmd) {
1243 err = -ENOMEM;
1244 goto failed;
1245 }
1246
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001247 /* Cancel any potential discoverable timeout that might be
1248 * still active and store new timeout value. The arming of
1249 * the timeout happens in the complete handler.
1250 */
1251 cancel_delayed_work(&hdev->discov_off);
1252 hdev->discov_timeout = timeout;
1253
Johan Hedbergb456f872013-10-19 23:38:22 +03001254 /* Limited discoverable mode */
1255 if (cp->val == 0x02)
1256 set_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1257 else
1258 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1259
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001260 hci_req_init(&req, hdev);
1261
Johan Hedberg9a43e252013-10-20 19:00:07 +03001262 /* The procedure for LE-only controllers is much simpler - just
1263 * update the advertising data.
1264 */
1265 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
1266 goto update_ad;
1267
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001268 scan = SCAN_PAGE;
1269
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001270 if (cp->val) {
1271 struct hci_cp_write_current_iac_lap hci_cp;
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001272
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001273 if (cp->val == 0x02) {
1274 /* Limited discoverable mode */
Marcel Holtmann33337dc2013-10-23 08:28:01 -07001275 hci_cp.num_iac = min_t(u8, hdev->num_iac, 2);
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001276 hci_cp.iac_lap[0] = 0x00; /* LIAC */
1277 hci_cp.iac_lap[1] = 0x8b;
1278 hci_cp.iac_lap[2] = 0x9e;
1279 hci_cp.iac_lap[3] = 0x33; /* GIAC */
1280 hci_cp.iac_lap[4] = 0x8b;
1281 hci_cp.iac_lap[5] = 0x9e;
1282 } else {
1283 /* General discoverable mode */
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001284 hci_cp.num_iac = 1;
1285 hci_cp.iac_lap[0] = 0x33; /* GIAC */
1286 hci_cp.iac_lap[1] = 0x8b;
1287 hci_cp.iac_lap[2] = 0x9e;
1288 }
1289
1290 hci_req_add(&req, HCI_OP_WRITE_CURRENT_IAC_LAP,
1291 (hci_cp.num_iac * 3) + 1, &hci_cp);
1292
1293 scan |= SCAN_INQUIRY;
1294 } else {
1295 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1296 }
1297
1298 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE, sizeof(scan), &scan);
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001299
Johan Hedberg9a43e252013-10-20 19:00:07 +03001300update_ad:
1301 update_adv_data(&req);
1302
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001303 err = hci_req_run(&req, set_discoverable_complete);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001304 if (err < 0)
1305 mgmt_pending_remove(cmd);
1306
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001307failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001308 hci_dev_unlock(hdev);
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001309 return err;
1310}
1311
Johan Hedberg406d7802013-03-15 17:07:09 -05001312static void write_fast_connectable(struct hci_request *req, bool enable)
1313{
Johan Hedbergbd98b992013-03-15 17:07:13 -05001314 struct hci_dev *hdev = req->hdev;
Johan Hedberg406d7802013-03-15 17:07:09 -05001315 struct hci_cp_write_page_scan_activity acp;
1316 u8 type;
1317
Johan Hedberg547003b2013-10-21 16:51:53 +03001318 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
1319 return;
1320
Johan Hedberg4c01f8b2013-03-15 17:07:14 -05001321 if (hdev->hci_ver < BLUETOOTH_VER_1_2)
1322 return;
1323
Johan Hedberg406d7802013-03-15 17:07:09 -05001324 if (enable) {
1325 type = PAGE_SCAN_TYPE_INTERLACED;
1326
1327 /* 160 msec page scan interval */
1328 acp.interval = __constant_cpu_to_le16(0x0100);
1329 } else {
1330 type = PAGE_SCAN_TYPE_STANDARD; /* default */
1331
1332 /* default 1.28 sec page scan */
1333 acp.interval = __constant_cpu_to_le16(0x0800);
1334 }
1335
1336 acp.window = __constant_cpu_to_le16(0x0012);
1337
Johan Hedbergbd98b992013-03-15 17:07:13 -05001338 if (__cpu_to_le16(hdev->page_scan_interval) != acp.interval ||
1339 __cpu_to_le16(hdev->page_scan_window) != acp.window)
1340 hci_req_add(req, HCI_OP_WRITE_PAGE_SCAN_ACTIVITY,
1341 sizeof(acp), &acp);
1342
1343 if (hdev->page_scan_type != type)
1344 hci_req_add(req, HCI_OP_WRITE_PAGE_SCAN_TYPE, 1, &type);
Johan Hedberg406d7802013-03-15 17:07:09 -05001345}
1346
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001347static u8 get_adv_type(struct hci_dev *hdev)
1348{
1349 struct pending_cmd *cmd;
1350 bool connectable;
1351
1352 /* If there's a pending mgmt command the flag will not yet have
1353 * it's final value, so check for this first.
1354 */
1355 cmd = mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev);
1356 if (cmd) {
1357 struct mgmt_mode *cp = cmd->param;
1358 connectable = !!cp->val;
1359 } else {
1360 connectable = test_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1361 }
1362
1363 return connectable ? LE_ADV_IND : LE_ADV_NONCONN_IND;
1364}
1365
Johan Hedberg95c66e72013-10-14 16:20:06 +03001366static void enable_advertising(struct hci_request *req)
1367{
1368 struct hci_dev *hdev = req->hdev;
1369 struct hci_cp_le_set_adv_param cp;
1370 u8 enable = 0x01;
1371
1372 memset(&cp, 0, sizeof(cp));
1373 cp.min_interval = __constant_cpu_to_le16(0x0800);
1374 cp.max_interval = __constant_cpu_to_le16(0x0800);
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001375 cp.type = get_adv_type(hdev);
Marcel Holtmann79830f62013-10-18 16:38:09 -07001376 cp.own_address_type = hdev->own_addr_type;
Johan Hedberg95c66e72013-10-14 16:20:06 +03001377 cp.channel_map = 0x07;
1378
1379 hci_req_add(req, HCI_OP_LE_SET_ADV_PARAM, sizeof(cp), &cp);
1380
1381 hci_req_add(req, HCI_OP_LE_SET_ADV_ENABLE, sizeof(enable), &enable);
1382}
1383
1384static void disable_advertising(struct hci_request *req)
1385{
1386 u8 enable = 0x00;
1387
1388 hci_req_add(req, HCI_OP_LE_SET_ADV_ENABLE, sizeof(enable), &enable);
1389}
1390
Johan Hedberg2b76f452013-03-15 17:07:04 -05001391static void set_connectable_complete(struct hci_dev *hdev, u8 status)
1392{
1393 struct pending_cmd *cmd;
Johan Hedbergd7b856f2013-10-14 16:20:04 +03001394 struct mgmt_mode *cp;
1395 bool changed;
Johan Hedberg2b76f452013-03-15 17:07:04 -05001396
1397 BT_DBG("status 0x%02x", status);
1398
1399 hci_dev_lock(hdev);
1400
1401 cmd = mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev);
1402 if (!cmd)
1403 goto unlock;
1404
Johan Hedberg37438c12013-10-14 16:20:05 +03001405 if (status) {
1406 u8 mgmt_err = mgmt_status(status);
1407 cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
1408 goto remove_cmd;
1409 }
1410
Johan Hedbergd7b856f2013-10-14 16:20:04 +03001411 cp = cmd->param;
1412 if (cp->val)
1413 changed = !test_and_set_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1414 else
1415 changed = test_and_clear_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1416
Johan Hedberg2b76f452013-03-15 17:07:04 -05001417 send_settings_rsp(cmd->sk, MGMT_OP_SET_CONNECTABLE, hdev);
1418
Johan Hedbergd7b856f2013-10-14 16:20:04 +03001419 if (changed)
1420 new_settings(hdev, cmd->sk);
1421
Johan Hedberg37438c12013-10-14 16:20:05 +03001422remove_cmd:
Johan Hedberg2b76f452013-03-15 17:07:04 -05001423 mgmt_pending_remove(cmd);
1424
1425unlock:
1426 hci_dev_unlock(hdev);
1427}
1428
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001429static int set_connectable_update_settings(struct hci_dev *hdev,
1430 struct sock *sk, u8 val)
1431{
1432 bool changed = false;
1433 int err;
1434
1435 if (!!val != test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
1436 changed = true;
1437
1438 if (val) {
1439 set_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1440 } else {
1441 clear_bit(HCI_CONNECTABLE, &hdev->dev_flags);
1442 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
1443 }
1444
1445 err = send_settings_rsp(sk, MGMT_OP_SET_CONNECTABLE, hdev);
1446 if (err < 0)
1447 return err;
1448
1449 if (changed)
1450 return new_settings(hdev, sk);
1451
1452 return 0;
1453}
1454
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001455static int set_connectable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001456 u16 len)
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001457{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001458 struct mgmt_mode *cp = data;
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001459 struct pending_cmd *cmd;
Johan Hedberg2b76f452013-03-15 17:07:04 -05001460 struct hci_request req;
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001461 u8 scan;
Johan Hedbergf7b64e62010-12-13 21:07:06 +02001462 int err;
Johan Hedberg03811012010-12-08 00:21:06 +02001463
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001464 BT_DBG("request for %s", hdev->name);
Johan Hedberge41d8b42010-12-13 21:07:03 +02001465
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001466 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags) &&
1467 !test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedberg33c525c2012-10-24 21:11:58 +03001468 return cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001469 MGMT_STATUS_REJECTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001470
Johan Hedberga7e80f22013-01-09 16:05:19 +02001471 if (cp->val != 0x00 && cp->val != 0x01)
1472 return cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
1473 MGMT_STATUS_INVALID_PARAMS);
1474
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001475 hci_dev_lock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001476
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001477 if (!hdev_is_powered(hdev)) {
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001478 err = set_connectable_update_settings(hdev, sk, cp->val);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001479 goto failed;
1480 }
1481
1482 if (mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03001483 mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001484 err = cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001485 MGMT_STATUS_BUSY);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001486 goto failed;
1487 }
1488
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001489 cmd = mgmt_pending_add(sk, MGMT_OP_SET_CONNECTABLE, hdev, data, len);
1490 if (!cmd) {
1491 err = -ENOMEM;
1492 goto failed;
1493 }
1494
Johan Hedberg2b76f452013-03-15 17:07:04 -05001495 hci_req_init(&req, hdev);
1496
Johan Hedberg9a43e252013-10-20 19:00:07 +03001497 /* If BR/EDR is not enabled and we disable advertising as a
1498 * by-product of disabling connectable, we need to update the
1499 * advertising flags.
1500 */
1501 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
1502 if (!cp->val) {
1503 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
1504 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
1505 }
1506 update_adv_data(&req);
1507 } else if (cp->val != test_bit(HCI_PSCAN, &hdev->flags)) {
Johan Hedberg9b742462013-10-14 16:20:03 +03001508 if (cp->val) {
1509 scan = SCAN_PAGE;
1510 } else {
1511 scan = 0;
1512
1513 if (test_bit(HCI_ISCAN, &hdev->flags) &&
Marcel Holtmann8d6083f2013-10-14 16:38:45 -07001514 hdev->discov_timeout > 0)
Johan Hedberg9b742462013-10-14 16:20:03 +03001515 cancel_delayed_work(&hdev->discov_off);
1516 }
1517
1518 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
1519 }
Johan Hedberg2b76f452013-03-15 17:07:04 -05001520
Johan Hedberg4c01f8b2013-03-15 17:07:14 -05001521 /* If we're going from non-connectable to connectable or
1522 * vice-versa when fast connectable is enabled ensure that fast
1523 * connectable gets disabled. write_fast_connectable won't do
1524 * anything if the page scan parameters are already what they
1525 * should be.
1526 */
1527 if (cp->val || test_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags))
Johan Hedberge36a3762013-03-15 17:07:10 -05001528 write_fast_connectable(&req, false);
1529
Johan Hedberg1987fdc2013-10-14 21:15:24 +03001530 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags) &&
1531 hci_conn_num(hdev, LE_LINK) == 0) {
1532 disable_advertising(&req);
1533 enable_advertising(&req);
1534 }
1535
Johan Hedberg2b76f452013-03-15 17:07:04 -05001536 err = hci_req_run(&req, set_connectable_complete);
Johan Hedberg9b742462013-10-14 16:20:03 +03001537 if (err < 0) {
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001538 mgmt_pending_remove(cmd);
Johan Hedberg9b742462013-10-14 16:20:03 +03001539 if (err == -ENODATA)
Johan Hedberga81070b2013-10-19 23:38:19 +03001540 err = set_connectable_update_settings(hdev, sk,
1541 cp->val);
Johan Hedberg9b742462013-10-14 16:20:03 +03001542 goto failed;
1543 }
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001544
1545failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001546 hci_dev_unlock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001547 return err;
1548}
1549
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001550static int set_pairable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001551 u16 len)
Johan Hedberg73f22f62010-12-29 16:00:25 +02001552{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001553 struct mgmt_mode *cp = data;
Marcel Holtmann55594352013-10-06 16:11:57 -07001554 bool changed;
Johan Hedberg73f22f62010-12-29 16:00:25 +02001555 int err;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001556
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001557 BT_DBG("request for %s", hdev->name);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001558
Johan Hedberga7e80f22013-01-09 16:05:19 +02001559 if (cp->val != 0x00 && cp->val != 0x01)
1560 return cmd_status(sk, hdev->id, MGMT_OP_SET_PAIRABLE,
1561 MGMT_STATUS_INVALID_PARAMS);
1562
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001563 hci_dev_lock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001564
1565 if (cp->val)
Marcel Holtmann55594352013-10-06 16:11:57 -07001566 changed = !test_and_set_bit(HCI_PAIRABLE, &hdev->dev_flags);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001567 else
Marcel Holtmann55594352013-10-06 16:11:57 -07001568 changed = test_and_clear_bit(HCI_PAIRABLE, &hdev->dev_flags);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001569
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001570 err = send_settings_rsp(sk, MGMT_OP_SET_PAIRABLE, hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001571 if (err < 0)
Marcel Holtmann55594352013-10-06 16:11:57 -07001572 goto unlock;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001573
Marcel Holtmann55594352013-10-06 16:11:57 -07001574 if (changed)
1575 err = new_settings(hdev, sk);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001576
Marcel Holtmann55594352013-10-06 16:11:57 -07001577unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001578 hci_dev_unlock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001579 return err;
1580}
Johan Hedberg72a734e2010-12-30 00:38:22 +02001581
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001582static int set_link_security(struct sock *sk, struct hci_dev *hdev, void *data,
1583 u16 len)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001584{
1585 struct mgmt_mode *cp = data;
1586 struct pending_cmd *cmd;
Johan Hedberge6fe7982013-10-02 15:45:22 +03001587 u8 val, status;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001588 int err;
1589
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001590 BT_DBG("request for %s", hdev->name);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001591
Johan Hedberge6fe7982013-10-02 15:45:22 +03001592 status = mgmt_bredr_support(hdev);
1593 if (status)
Johan Hedberg33c525c2012-10-24 21:11:58 +03001594 return cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03001595 status);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001596
Johan Hedberga7e80f22013-01-09 16:05:19 +02001597 if (cp->val != 0x00 && cp->val != 0x01)
1598 return cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
1599 MGMT_STATUS_INVALID_PARAMS);
1600
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001601 hci_dev_lock(hdev);
1602
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001603 if (!hdev_is_powered(hdev)) {
Johan Hedberg47990ea2012-02-22 11:58:37 +02001604 bool changed = false;
1605
1606 if (!!cp->val != test_bit(HCI_LINK_SECURITY,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03001607 &hdev->dev_flags)) {
Johan Hedberg47990ea2012-02-22 11:58:37 +02001608 change_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
1609 changed = true;
1610 }
1611
1612 err = send_settings_rsp(sk, MGMT_OP_SET_LINK_SECURITY, hdev);
1613 if (err < 0)
1614 goto failed;
1615
1616 if (changed)
1617 err = new_settings(hdev, sk);
1618
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001619 goto failed;
1620 }
1621
1622 if (mgmt_pending_find(MGMT_OP_SET_LINK_SECURITY, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001623 err = cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001624 MGMT_STATUS_BUSY);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001625 goto failed;
1626 }
1627
1628 val = !!cp->val;
1629
1630 if (test_bit(HCI_AUTH, &hdev->flags) == val) {
1631 err = send_settings_rsp(sk, MGMT_OP_SET_LINK_SECURITY, hdev);
1632 goto failed;
1633 }
1634
1635 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LINK_SECURITY, hdev, data, len);
1636 if (!cmd) {
1637 err = -ENOMEM;
1638 goto failed;
1639 }
1640
1641 err = hci_send_cmd(hdev, HCI_OP_WRITE_AUTH_ENABLE, sizeof(val), &val);
1642 if (err < 0) {
1643 mgmt_pending_remove(cmd);
1644 goto failed;
1645 }
1646
1647failed:
1648 hci_dev_unlock(hdev);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001649 return err;
1650}
1651
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001652static int set_ssp(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001653{
1654 struct mgmt_mode *cp = data;
1655 struct pending_cmd *cmd;
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001656 u8 status;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001657 int err;
1658
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001659 BT_DBG("request for %s", hdev->name);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001660
Marcel Holtmanncdba5282013-10-02 21:31:52 -07001661 status = mgmt_bredr_support(hdev);
1662 if (status)
1663 return cmd_status(sk, hdev->id, MGMT_OP_SET_SSP, status);
1664
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001665 if (!lmp_ssp_capable(hdev))
1666 return cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1667 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001668
Johan Hedberga7e80f22013-01-09 16:05:19 +02001669 if (cp->val != 0x00 && cp->val != 0x01)
1670 return cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1671 MGMT_STATUS_INVALID_PARAMS);
1672
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001673 hci_dev_lock(hdev);
Johan Hedberg6c8f12c2012-02-22 16:35:26 +02001674
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001675 if (!hdev_is_powered(hdev)) {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001676 bool changed;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02001677
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001678 if (cp->val) {
1679 changed = !test_and_set_bit(HCI_SSP_ENABLED,
1680 &hdev->dev_flags);
1681 } else {
1682 changed = test_and_clear_bit(HCI_SSP_ENABLED,
1683 &hdev->dev_flags);
1684 if (!changed)
1685 changed = test_and_clear_bit(HCI_HS_ENABLED,
1686 &hdev->dev_flags);
1687 else
1688 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02001689 }
1690
1691 err = send_settings_rsp(sk, MGMT_OP_SET_SSP, hdev);
1692 if (err < 0)
1693 goto failed;
1694
1695 if (changed)
1696 err = new_settings(hdev, sk);
1697
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001698 goto failed;
1699 }
1700
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001701 if (mgmt_pending_find(MGMT_OP_SET_SSP, hdev) ||
1702 mgmt_pending_find(MGMT_OP_SET_HS, hdev)) {
Szymon Jancd97dcb62012-03-16 16:02:56 +01001703 err = cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1704 MGMT_STATUS_BUSY);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001705 goto failed;
1706 }
1707
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001708 if (!!cp->val == test_bit(HCI_SSP_ENABLED, &hdev->dev_flags)) {
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001709 err = send_settings_rsp(sk, MGMT_OP_SET_SSP, hdev);
1710 goto failed;
1711 }
1712
1713 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SSP, hdev, data, len);
1714 if (!cmd) {
1715 err = -ENOMEM;
1716 goto failed;
1717 }
1718
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001719 err = hci_send_cmd(hdev, HCI_OP_WRITE_SSP_MODE, 1, &cp->val);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001720 if (err < 0) {
1721 mgmt_pending_remove(cmd);
1722 goto failed;
1723 }
1724
1725failed:
1726 hci_dev_unlock(hdev);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001727 return err;
1728}
1729
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001730static int set_hs(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001731{
1732 struct mgmt_mode *cp = data;
Marcel Holtmannee392692013-10-01 22:59:23 -07001733 bool changed;
Johan Hedberge6fe7982013-10-02 15:45:22 +03001734 u8 status;
Marcel Holtmannee392692013-10-01 22:59:23 -07001735 int err;
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001736
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001737 BT_DBG("request for %s", hdev->name);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001738
Johan Hedberge6fe7982013-10-02 15:45:22 +03001739 status = mgmt_bredr_support(hdev);
1740 if (status)
1741 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS, status);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001742
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001743 if (!lmp_ssp_capable(hdev))
1744 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1745 MGMT_STATUS_NOT_SUPPORTED);
1746
1747 if (!test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
1748 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1749 MGMT_STATUS_REJECTED);
1750
Johan Hedberga7e80f22013-01-09 16:05:19 +02001751 if (cp->val != 0x00 && cp->val != 0x01)
1752 return cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1753 MGMT_STATUS_INVALID_PARAMS);
1754
Marcel Holtmannee392692013-10-01 22:59:23 -07001755 hci_dev_lock(hdev);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001756
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001757 if (cp->val) {
Marcel Holtmannee392692013-10-01 22:59:23 -07001758 changed = !test_and_set_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001759 } else {
1760 if (hdev_is_powered(hdev)) {
1761 err = cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1762 MGMT_STATUS_REJECTED);
1763 goto unlock;
1764 }
1765
Marcel Holtmannee392692013-10-01 22:59:23 -07001766 changed = test_and_clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001767 }
Marcel Holtmannee392692013-10-01 22:59:23 -07001768
1769 err = send_settings_rsp(sk, MGMT_OP_SET_HS, hdev);
1770 if (err < 0)
1771 goto unlock;
1772
1773 if (changed)
1774 err = new_settings(hdev, sk);
1775
1776unlock:
1777 hci_dev_unlock(hdev);
1778 return err;
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001779}
1780
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001781static void le_enable_complete(struct hci_dev *hdev, u8 status)
1782{
1783 struct cmd_lookup match = { NULL, hdev };
1784
1785 if (status) {
1786 u8 mgmt_err = mgmt_status(status);
1787
1788 mgmt_pending_foreach(MGMT_OP_SET_LE, hdev, cmd_status_rsp,
1789 &mgmt_err);
1790 return;
1791 }
1792
1793 mgmt_pending_foreach(MGMT_OP_SET_LE, hdev, settings_rsp, &match);
1794
1795 new_settings(hdev, match.sk);
1796
1797 if (match.sk)
1798 sock_put(match.sk);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001799
1800 /* Make sure the controller has a good default for
1801 * advertising data. Restrict the update to when LE
1802 * has actually been enabled. During power on, the
1803 * update in powered_update_hci will take care of it.
1804 */
1805 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
1806 struct hci_request req;
1807
1808 hci_dev_lock(hdev);
1809
1810 hci_req_init(&req, hdev);
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07001811 update_adv_data(&req);
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07001812 update_scan_rsp_data(&req);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001813 hci_req_run(&req, NULL);
1814
1815 hci_dev_unlock(hdev);
1816 }
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001817}
1818
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001819static int set_le(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg06199cf2012-02-22 16:37:11 +02001820{
1821 struct mgmt_mode *cp = data;
1822 struct hci_cp_write_le_host_supported hci_cp;
1823 struct pending_cmd *cmd;
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001824 struct hci_request req;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001825 int err;
Johan Hedberg0b60eba2012-02-28 00:57:24 +02001826 u8 val, enabled;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001827
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001828 BT_DBG("request for %s", hdev->name);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001829
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001830 if (!lmp_le_capable(hdev))
1831 return cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1832 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg1de028c2012-02-29 19:55:35 -08001833
Johan Hedberga7e80f22013-01-09 16:05:19 +02001834 if (cp->val != 0x00 && cp->val != 0x01)
1835 return cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1836 MGMT_STATUS_INVALID_PARAMS);
1837
Johan Hedbergc73eee92013-04-19 18:35:21 +03001838 /* LE-only devices do not allow toggling LE on/off */
Johan Hedberg56f87902013-10-02 13:43:13 +03001839 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
Johan Hedbergc73eee92013-04-19 18:35:21 +03001840 return cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1841 MGMT_STATUS_REJECTED);
1842
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001843 hci_dev_lock(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001844
1845 val = !!cp->val;
Gustavo Padovanffa88e02012-11-23 16:50:51 -02001846 enabled = lmp_host_le_capable(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001847
Johan Hedberg0b60eba2012-02-28 00:57:24 +02001848 if (!hdev_is_powered(hdev) || val == enabled) {
Johan Hedberg06199cf2012-02-22 16:37:11 +02001849 bool changed = false;
1850
1851 if (val != test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
1852 change_bit(HCI_LE_ENABLED, &hdev->dev_flags);
1853 changed = true;
1854 }
1855
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02001856 if (!val && test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
1857 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedbergeeca6f82013-09-25 13:26:09 +03001858 changed = true;
1859 }
1860
Johan Hedberg06199cf2012-02-22 16:37:11 +02001861 err = send_settings_rsp(sk, MGMT_OP_SET_LE, hdev);
1862 if (err < 0)
Johan Hedberg1de028c2012-02-29 19:55:35 -08001863 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001864
1865 if (changed)
1866 err = new_settings(hdev, sk);
1867
Johan Hedberg1de028c2012-02-29 19:55:35 -08001868 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001869 }
1870
Johan Hedberg4375f102013-09-25 13:26:10 +03001871 if (mgmt_pending_find(MGMT_OP_SET_LE, hdev) ||
1872 mgmt_pending_find(MGMT_OP_SET_ADVERTISING, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001873 err = cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001874 MGMT_STATUS_BUSY);
Johan Hedberg1de028c2012-02-29 19:55:35 -08001875 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001876 }
1877
1878 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LE, hdev, data, len);
1879 if (!cmd) {
1880 err = -ENOMEM;
Johan Hedberg1de028c2012-02-29 19:55:35 -08001881 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001882 }
1883
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001884 hci_req_init(&req, hdev);
1885
Johan Hedberg06199cf2012-02-22 16:37:11 +02001886 memset(&hci_cp, 0, sizeof(hci_cp));
1887
1888 if (val) {
1889 hci_cp.le = val;
Gustavo Padovanffa88e02012-11-23 16:50:51 -02001890 hci_cp.simul = lmp_le_br_capable(hdev);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001891 } else {
1892 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
1893 disable_advertising(&req);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001894 }
1895
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001896 hci_req_add(&req, HCI_OP_WRITE_LE_HOST_SUPPORTED, sizeof(hci_cp),
1897 &hci_cp);
1898
1899 err = hci_req_run(&req, le_enable_complete);
Syam Sidhardhan0c01bc42012-04-12 20:33:21 +05301900 if (err < 0)
Johan Hedberg06199cf2012-02-22 16:37:11 +02001901 mgmt_pending_remove(cmd);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001902
Johan Hedberg1de028c2012-02-29 19:55:35 -08001903unlock:
1904 hci_dev_unlock(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001905 return err;
1906}
1907
Johan Hedberg0cab9c82013-03-15 17:06:54 -05001908/* This is a helper function to test for pending mgmt commands that can
1909 * cause CoD or EIR HCI commands. We can only allow one such pending
1910 * mgmt command at a time since otherwise we cannot easily track what
1911 * the current values are, will be, and based on that calculate if a new
1912 * HCI command needs to be sent and if yes with what value.
1913 */
1914static bool pending_eir_or_class(struct hci_dev *hdev)
1915{
1916 struct pending_cmd *cmd;
1917
1918 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
1919 switch (cmd->opcode) {
1920 case MGMT_OP_ADD_UUID:
1921 case MGMT_OP_REMOVE_UUID:
1922 case MGMT_OP_SET_DEV_CLASS:
1923 case MGMT_OP_SET_POWERED:
1924 return true;
1925 }
1926 }
1927
1928 return false;
1929}
1930
Johan Hedberg83be8ec2013-01-27 00:31:29 +02001931static const u8 bluetooth_base_uuid[] = {
1932 0xfb, 0x34, 0x9b, 0x5f, 0x80, 0x00, 0x00, 0x80,
1933 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1934};
1935
1936static u8 get_uuid_size(const u8 *uuid)
1937{
1938 u32 val;
1939
1940 if (memcmp(uuid, bluetooth_base_uuid, 12))
1941 return 128;
1942
1943 val = get_unaligned_le32(&uuid[12]);
1944 if (val > 0xffff)
1945 return 32;
1946
1947 return 16;
1948}
1949
Johan Hedberg92da6092013-03-15 17:06:55 -05001950static void mgmt_class_complete(struct hci_dev *hdev, u16 mgmt_op, u8 status)
1951{
1952 struct pending_cmd *cmd;
1953
1954 hci_dev_lock(hdev);
1955
1956 cmd = mgmt_pending_find(mgmt_op, hdev);
1957 if (!cmd)
1958 goto unlock;
1959
1960 cmd_complete(cmd->sk, cmd->index, cmd->opcode, mgmt_status(status),
1961 hdev->dev_class, 3);
1962
1963 mgmt_pending_remove(cmd);
1964
1965unlock:
1966 hci_dev_unlock(hdev);
1967}
1968
1969static void add_uuid_complete(struct hci_dev *hdev, u8 status)
1970{
1971 BT_DBG("status 0x%02x", status);
1972
1973 mgmt_class_complete(hdev, MGMT_OP_ADD_UUID, status);
1974}
1975
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001976static int add_uuid(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001977{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001978 struct mgmt_cp_add_uuid *cp = data;
Johan Hedberg90e70452012-02-23 23:09:40 +02001979 struct pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05001980 struct hci_request req;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001981 struct bt_uuid *uuid;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001982 int err;
1983
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001984 BT_DBG("request for %s", hdev->name);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001985
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001986 hci_dev_lock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001987
Johan Hedberg0cab9c82013-03-15 17:06:54 -05001988 if (pending_eir_or_class(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001989 err = cmd_status(sk, hdev->id, MGMT_OP_ADD_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001990 MGMT_STATUS_BUSY);
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02001991 goto failed;
1992 }
1993
Andre Guedes92c4c202012-06-07 19:05:44 -03001994 uuid = kmalloc(sizeof(*uuid), GFP_KERNEL);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001995 if (!uuid) {
1996 err = -ENOMEM;
1997 goto failed;
1998 }
1999
2000 memcpy(uuid->uuid, cp->uuid, 16);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002001 uuid->svc_hint = cp->svc_hint;
Johan Hedberg83be8ec2013-01-27 00:31:29 +02002002 uuid->size = get_uuid_size(cp->uuid);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002003
Johan Hedbergde66aa62013-01-27 00:31:27 +02002004 list_add_tail(&uuid->list, &hdev->uuids);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002005
Johan Hedberg890ea892013-03-15 17:06:52 -05002006 hci_req_init(&req, hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002007
Johan Hedberg890ea892013-03-15 17:06:52 -05002008 update_class(&req);
2009 update_eir(&req);
2010
Johan Hedberg92da6092013-03-15 17:06:55 -05002011 err = hci_req_run(&req, add_uuid_complete);
2012 if (err < 0) {
2013 if (err != -ENODATA)
2014 goto failed;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +03002015
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002016 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_UUID, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002017 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002018 goto failed;
2019 }
2020
2021 cmd = mgmt_pending_add(sk, MGMT_OP_ADD_UUID, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002022 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002023 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002024 goto failed;
2025 }
2026
2027 err = 0;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002028
2029failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002030 hci_dev_unlock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002031 return err;
2032}
2033
Johan Hedberg24b78d02012-02-23 23:24:30 +02002034static bool enable_service_cache(struct hci_dev *hdev)
2035{
2036 if (!hdev_is_powered(hdev))
2037 return false;
2038
2039 if (!test_and_set_bit(HCI_SERVICE_CACHE, &hdev->dev_flags)) {
Johan Hedberg46818ed2013-01-14 22:33:52 +02002040 queue_delayed_work(hdev->workqueue, &hdev->service_cache,
2041 CACHE_TIMEOUT);
Johan Hedberg24b78d02012-02-23 23:24:30 +02002042 return true;
2043 }
2044
2045 return false;
2046}
2047
Johan Hedberg92da6092013-03-15 17:06:55 -05002048static void remove_uuid_complete(struct hci_dev *hdev, u8 status)
2049{
2050 BT_DBG("status 0x%02x", status);
2051
2052 mgmt_class_complete(hdev, MGMT_OP_REMOVE_UUID, status);
2053}
2054
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002055static int remove_uuid(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002056 u16 len)
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002057{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002058 struct mgmt_cp_remove_uuid *cp = data;
Johan Hedberg90e70452012-02-23 23:09:40 +02002059 struct pending_cmd *cmd;
Johan Hedberg056341c2013-01-27 00:31:30 +02002060 struct bt_uuid *match, *tmp;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002061 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 -05002062 struct hci_request req;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002063 int err, found;
2064
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002065 BT_DBG("request for %s", hdev->name);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002066
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002067 hci_dev_lock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002068
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002069 if (pending_eir_or_class(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002070 err = cmd_status(sk, hdev->id, MGMT_OP_REMOVE_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002071 MGMT_STATUS_BUSY);
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02002072 goto unlock;
2073 }
2074
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002075 if (memcmp(cp->uuid, bt_uuid_any, 16) == 0) {
Johan Hedberg35f74982014-02-18 17:14:32 +02002076 hci_uuids_clear(hdev);
Johan Hedberg4004b6d2012-02-23 21:30:12 +02002077
Johan Hedberg24b78d02012-02-23 23:24:30 +02002078 if (enable_service_cache(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002079 err = cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002080 0, hdev->dev_class, 3);
Johan Hedberg24b78d02012-02-23 23:24:30 +02002081 goto unlock;
2082 }
Johan Hedberg4004b6d2012-02-23 21:30:12 +02002083
Johan Hedberg9246a862012-02-23 21:33:16 +02002084 goto update_class;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002085 }
2086
2087 found = 0;
2088
Johan Hedberg056341c2013-01-27 00:31:30 +02002089 list_for_each_entry_safe(match, tmp, &hdev->uuids, list) {
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002090 if (memcmp(match->uuid, cp->uuid, 16) != 0)
2091 continue;
2092
2093 list_del(&match->list);
Johan Hedberg482049f2012-11-08 10:25:26 +01002094 kfree(match);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002095 found++;
2096 }
2097
2098 if (found == 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002099 err = cmd_status(sk, hdev->id, MGMT_OP_REMOVE_UUID,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002100 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002101 goto unlock;
2102 }
2103
Johan Hedberg9246a862012-02-23 21:33:16 +02002104update_class:
Johan Hedberg890ea892013-03-15 17:06:52 -05002105 hci_req_init(&req, hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002106
Johan Hedberg890ea892013-03-15 17:06:52 -05002107 update_class(&req);
2108 update_eir(&req);
2109
Johan Hedberg92da6092013-03-15 17:06:55 -05002110 err = hci_req_run(&req, remove_uuid_complete);
2111 if (err < 0) {
2112 if (err != -ENODATA)
2113 goto unlock;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +03002114
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002115 err = cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_UUID, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002116 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002117 goto unlock;
2118 }
2119
2120 cmd = mgmt_pending_add(sk, MGMT_OP_REMOVE_UUID, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002121 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002122 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002123 goto unlock;
2124 }
2125
2126 err = 0;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002127
2128unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002129 hci_dev_unlock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002130 return err;
2131}
2132
Johan Hedberg92da6092013-03-15 17:06:55 -05002133static void set_class_complete(struct hci_dev *hdev, u8 status)
2134{
2135 BT_DBG("status 0x%02x", status);
2136
2137 mgmt_class_complete(hdev, MGMT_OP_SET_DEV_CLASS, status);
2138}
2139
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002140static int set_dev_class(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002141 u16 len)
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002142{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002143 struct mgmt_cp_set_dev_class *cp = data;
Johan Hedberg90e70452012-02-23 23:09:40 +02002144 struct pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05002145 struct hci_request req;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002146 int err;
2147
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002148 BT_DBG("request for %s", hdev->name);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002149
Marcel Holtmann6203fc92013-10-02 23:37:29 -07002150 if (!lmp_bredr_capable(hdev))
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02002151 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2152 MGMT_STATUS_NOT_SUPPORTED);
2153
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002154 hci_dev_lock(hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002155
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002156 if (pending_eir_or_class(hdev)) {
2157 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2158 MGMT_STATUS_BUSY);
2159 goto unlock;
2160 }
2161
2162 if ((cp->minor & 0x03) != 0 || (cp->major & 0xe0) != 0) {
2163 err = cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2164 MGMT_STATUS_INVALID_PARAMS);
2165 goto unlock;
2166 }
2167
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002168 hdev->major_class = cp->major;
2169 hdev->minor_class = cp->minor;
2170
Johan Hedberg932f5ff2012-02-22 22:11:32 +02002171 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002172 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_DEV_CLASS, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002173 hdev->dev_class, 3);
Johan Hedberg932f5ff2012-02-22 22:11:32 +02002174 goto unlock;
2175 }
2176
Johan Hedberg890ea892013-03-15 17:06:52 -05002177 hci_req_init(&req, hdev);
2178
Johan Hedberga8b2d5c2012-01-08 23:11:15 +02002179 if (test_and_clear_bit(HCI_SERVICE_CACHE, &hdev->dev_flags)) {
Johan Hedberg7d785252011-12-15 00:47:39 +02002180 hci_dev_unlock(hdev);
2181 cancel_delayed_work_sync(&hdev->service_cache);
2182 hci_dev_lock(hdev);
Johan Hedberg890ea892013-03-15 17:06:52 -05002183 update_eir(&req);
Johan Hedberg7d785252011-12-15 00:47:39 +02002184 }
Johan Hedberg14c0b602011-12-15 00:47:37 +02002185
Johan Hedberg890ea892013-03-15 17:06:52 -05002186 update_class(&req);
2187
Johan Hedberg92da6092013-03-15 17:06:55 -05002188 err = hci_req_run(&req, set_class_complete);
2189 if (err < 0) {
2190 if (err != -ENODATA)
2191 goto unlock;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002192
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002193 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_DEV_CLASS, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002194 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002195 goto unlock;
2196 }
2197
2198 cmd = mgmt_pending_add(sk, MGMT_OP_SET_DEV_CLASS, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002199 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002200 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002201 goto unlock;
2202 }
2203
2204 err = 0;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002205
Johan Hedbergb5235a62012-02-21 14:32:24 +02002206unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002207 hci_dev_unlock(hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002208 return err;
2209}
2210
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002211static int load_link_keys(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002212 u16 len)
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002213{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002214 struct mgmt_cp_load_link_keys *cp = data;
Szymon Janc4e51eae2011-02-25 19:05:48 +01002215 u16 key_count, expected_len;
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002216 bool changed;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002217 int i;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002218
Marcel Holtmann9060d5c2013-10-02 21:16:07 -07002219 BT_DBG("request for %s", hdev->name);
2220
2221 if (!lmp_bredr_capable(hdev))
2222 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2223 MGMT_STATUS_NOT_SUPPORTED);
2224
Marcel Holtmann1f350c82012-03-12 20:31:08 -07002225 key_count = __le16_to_cpu(cp->key_count);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002226
Johan Hedberg86742e12011-11-07 23:13:38 +02002227 expected_len = sizeof(*cp) + key_count *
2228 sizeof(struct mgmt_link_key_info);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002229 if (expected_len != len) {
Johan Hedberg86742e12011-11-07 23:13:38 +02002230 BT_ERR("load_link_keys: expected %u bytes, got %u bytes",
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002231 len, expected_len);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002232 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002233 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002234 }
2235
Johan Hedberg4ae14302013-01-20 14:27:13 +02002236 if (cp->debug_keys != 0x00 && cp->debug_keys != 0x01)
2237 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2238 MGMT_STATUS_INVALID_PARAMS);
2239
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002240 BT_DBG("%s debug_keys %u key_count %u", hdev->name, cp->debug_keys,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002241 key_count);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002242
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002243 for (i = 0; i < key_count; i++) {
2244 struct mgmt_link_key_info *key = &cp->keys[i];
2245
Marcel Holtmann8e991132014-01-10 02:07:25 -08002246 if (key->addr.type != BDADDR_BREDR || key->type > 0x08)
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002247 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2248 MGMT_STATUS_INVALID_PARAMS);
2249 }
2250
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002251 hci_dev_lock(hdev);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002252
2253 hci_link_keys_clear(hdev);
2254
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002255 if (cp->debug_keys)
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002256 changed = !test_and_set_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002257 else
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002258 changed = test_and_clear_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
2259
2260 if (changed)
2261 new_settings(hdev, NULL);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002262
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002263 for (i = 0; i < key_count; i++) {
Johan Hedberg86742e12011-11-07 23:13:38 +02002264 struct mgmt_link_key_info *key = &cp->keys[i];
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002265
Johan Hedbergd753fdc2012-02-17 14:06:34 +02002266 hci_add_link_key(hdev, NULL, 0, &key->addr.bdaddr, key->val,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002267 key->type, key->pin_len);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002268 }
2269
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002270 cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS, 0, NULL, 0);
Johan Hedberg0e5f8752011-11-11 16:18:54 +02002271
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002272 hci_dev_unlock(hdev);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002273
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002274 return 0;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002275}
2276
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002277static int device_unpaired(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002278 u8 addr_type, struct sock *skip_sk)
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002279{
2280 struct mgmt_ev_device_unpaired ev;
2281
2282 bacpy(&ev.addr.bdaddr, bdaddr);
2283 ev.addr.type = addr_type;
2284
2285 return mgmt_event(MGMT_EV_DEVICE_UNPAIRED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002286 skip_sk);
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002287}
2288
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002289static int unpair_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002290 u16 len)
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002291{
Johan Hedberg124f6e32012-02-09 13:50:12 +02002292 struct mgmt_cp_unpair_device *cp = data;
2293 struct mgmt_rp_unpair_device rp;
Johan Hedberga8a1d192011-11-10 15:54:38 +02002294 struct hci_cp_disconnect dc;
2295 struct pending_cmd *cmd;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002296 struct hci_conn *conn;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002297 int err;
2298
Johan Hedberga8a1d192011-11-10 15:54:38 +02002299 memset(&rp, 0, sizeof(rp));
Johan Hedberg124f6e32012-02-09 13:50:12 +02002300 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2301 rp.addr.type = cp->addr.type;
Johan Hedberga8a1d192011-11-10 15:54:38 +02002302
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002303 if (!bdaddr_type_is_valid(cp->addr.type))
2304 return cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2305 MGMT_STATUS_INVALID_PARAMS,
2306 &rp, sizeof(rp));
2307
Johan Hedberg118da702013-01-20 14:27:20 +02002308 if (cp->disconnect != 0x00 && cp->disconnect != 0x01)
2309 return cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2310 MGMT_STATUS_INVALID_PARAMS,
2311 &rp, sizeof(rp));
2312
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002313 hci_dev_lock(hdev);
2314
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002315 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002316 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002317 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002318 goto unlock;
2319 }
2320
Johan Hedberge0b2b272014-02-18 17:14:31 +02002321 if (cp->addr.type == BDADDR_BREDR) {
Johan Hedberg124f6e32012-02-09 13:50:12 +02002322 err = hci_remove_link_key(hdev, &cp->addr.bdaddr);
Johan Hedberge0b2b272014-02-18 17:14:31 +02002323 } else {
2324 u8 addr_type;
2325
2326 if (cp->addr.type == BDADDR_LE_PUBLIC)
2327 addr_type = ADDR_LE_DEV_PUBLIC;
2328 else
2329 addr_type = ADDR_LE_DEV_RANDOM;
2330
Johan Hedberga7ec7332014-02-18 17:14:35 +02002331 hci_remove_irk(hdev, &cp->addr.bdaddr, addr_type);
2332
Johan Hedberge0b2b272014-02-18 17:14:31 +02002333 err = hci_remove_ltk(hdev, &cp->addr.bdaddr, addr_type);
2334 }
Vinicius Costa Gomesb0dbfb42012-02-02 21:08:03 -03002335
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002336 if (err < 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002337 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002338 MGMT_STATUS_NOT_PAIRED, &rp, sizeof(rp));
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002339 goto unlock;
2340 }
2341
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002342 if (cp->disconnect) {
Andre Guedes591f47f2012-04-24 21:02:49 -03002343 if (cp->addr.type == BDADDR_BREDR)
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002344 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002345 &cp->addr.bdaddr);
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002346 else
2347 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002348 &cp->addr.bdaddr);
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002349 } else {
2350 conn = NULL;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002351 }
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002352
Johan Hedberga8a1d192011-11-10 15:54:38 +02002353 if (!conn) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002354 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002355 &rp, sizeof(rp));
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002356 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, sk);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002357 goto unlock;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002358 }
2359
Johan Hedberg124f6e32012-02-09 13:50:12 +02002360 cmd = mgmt_pending_add(sk, MGMT_OP_UNPAIR_DEVICE, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002361 sizeof(*cp));
Johan Hedberga8a1d192011-11-10 15:54:38 +02002362 if (!cmd) {
2363 err = -ENOMEM;
2364 goto unlock;
2365 }
2366
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002367 dc.handle = cpu_to_le16(conn->handle);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002368 dc.reason = 0x13; /* Remote User Terminated Connection */
2369 err = hci_send_cmd(hdev, HCI_OP_DISCONNECT, sizeof(dc), &dc);
2370 if (err < 0)
2371 mgmt_pending_remove(cmd);
2372
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002373unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002374 hci_dev_unlock(hdev);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02002375 return err;
2376}
2377
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002378static int disconnect(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002379 u16 len)
Johan Hedberg8962ee72011-01-20 12:40:27 +02002380{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002381 struct mgmt_cp_disconnect *cp = data;
Johan Hedberg06a63b12013-01-20 14:27:21 +02002382 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002383 struct hci_cp_disconnect dc;
Johan Hedberg366a0332011-02-19 12:05:55 -03002384 struct pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002385 struct hci_conn *conn;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002386 int err;
2387
2388 BT_DBG("");
2389
Johan Hedberg06a63b12013-01-20 14:27:21 +02002390 memset(&rp, 0, sizeof(rp));
2391 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2392 rp.addr.type = cp->addr.type;
2393
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002394 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg06a63b12013-01-20 14:27:21 +02002395 return cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2396 MGMT_STATUS_INVALID_PARAMS,
2397 &rp, sizeof(rp));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002398
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002399 hci_dev_lock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002400
2401 if (!test_bit(HCI_UP, &hdev->flags)) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002402 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2403 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002404 goto failed;
2405 }
2406
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002407 if (mgmt_pending_find(MGMT_OP_DISCONNECT, hdev)) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002408 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2409 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002410 goto failed;
2411 }
2412
Andre Guedes591f47f2012-04-24 21:02:49 -03002413 if (cp->addr.type == BDADDR_BREDR)
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03002414 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
2415 &cp->addr.bdaddr);
Johan Hedberg88c3df12012-02-09 14:27:38 +02002416 else
2417 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &cp->addr.bdaddr);
Vinicius Costa Gomes365227e2011-05-06 18:41:44 -03002418
Vishal Agarwalf9607272012-06-13 05:32:43 +05302419 if (!conn || conn->state == BT_OPEN || conn->state == BT_CLOSED) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002420 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2421 MGMT_STATUS_NOT_CONNECTED, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002422 goto failed;
2423 }
2424
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002425 cmd = mgmt_pending_add(sk, MGMT_OP_DISCONNECT, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002426 if (!cmd) {
2427 err = -ENOMEM;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002428 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002429 }
Johan Hedberg8962ee72011-01-20 12:40:27 +02002430
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002431 dc.handle = cpu_to_le16(conn->handle);
Andre Guedes3701f942012-06-11 18:41:12 -03002432 dc.reason = HCI_ERROR_REMOTE_USER_TERM;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002433
2434 err = hci_send_cmd(hdev, HCI_OP_DISCONNECT, sizeof(dc), &dc);
2435 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002436 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002437
2438failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002439 hci_dev_unlock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002440 return err;
2441}
2442
Andre Guedes57c14772012-04-24 21:02:50 -03002443static u8 link_to_bdaddr(u8 link_type, u8 addr_type)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002444{
2445 switch (link_type) {
2446 case LE_LINK:
Johan Hedberg48264f02011-11-09 13:58:58 +02002447 switch (addr_type) {
2448 case ADDR_LE_DEV_PUBLIC:
Andre Guedes591f47f2012-04-24 21:02:49 -03002449 return BDADDR_LE_PUBLIC;
Andre Guedes0ed09142012-04-03 08:46:54 -03002450
Johan Hedberg48264f02011-11-09 13:58:58 +02002451 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002452 /* Fallback to LE Random address type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002453 return BDADDR_LE_RANDOM;
Johan Hedberg48264f02011-11-09 13:58:58 +02002454 }
Andre Guedes0ed09142012-04-03 08:46:54 -03002455
Johan Hedberg4c659c32011-11-07 23:13:39 +02002456 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002457 /* Fallback to BR/EDR type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002458 return BDADDR_BREDR;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002459 }
2460}
2461
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002462static int get_connections(struct sock *sk, struct hci_dev *hdev, void *data,
2463 u16 data_len)
Johan Hedberg2784eb42011-01-21 13:56:35 +02002464{
Johan Hedberg2784eb42011-01-21 13:56:35 +02002465 struct mgmt_rp_get_connections *rp;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +02002466 struct hci_conn *c;
Johan Hedberga38528f2011-01-22 06:46:43 +02002467 size_t rp_len;
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002468 int err;
2469 u16 i;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002470
2471 BT_DBG("");
2472
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002473 hci_dev_lock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002474
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002475 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002476 err = cmd_status(sk, hdev->id, MGMT_OP_GET_CONNECTIONS,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002477 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002478 goto unlock;
2479 }
2480
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002481 i = 0;
Johan Hedbergb644ba32012-01-17 21:48:47 +02002482 list_for_each_entry(c, &hdev->conn_hash.list, list) {
2483 if (test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002484 i++;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002485 }
2486
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002487 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Andre Guedes92c4c202012-06-07 19:05:44 -03002488 rp = kmalloc(rp_len, GFP_KERNEL);
Johan Hedberga38528f2011-01-22 06:46:43 +02002489 if (!rp) {
Johan Hedberg2784eb42011-01-21 13:56:35 +02002490 err = -ENOMEM;
2491 goto unlock;
2492 }
2493
Johan Hedberg2784eb42011-01-21 13:56:35 +02002494 i = 0;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002495 list_for_each_entry(c, &hdev->conn_hash.list, list) {
Johan Hedbergb644ba32012-01-17 21:48:47 +02002496 if (!test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
2497 continue;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002498 bacpy(&rp->addr[i].bdaddr, &c->dst);
Andre Guedes57c14772012-04-24 21:02:50 -03002499 rp->addr[i].type = link_to_bdaddr(c->type, c->dst_type);
Andre Guedes0ed09142012-04-03 08:46:54 -03002500 if (c->type == SCO_LINK || c->type == ESCO_LINK)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002501 continue;
2502 i++;
2503 }
2504
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002505 rp->conn_count = cpu_to_le16(i);
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002506
Johan Hedberg4c659c32011-11-07 23:13:39 +02002507 /* Recalculate length in case of filtered SCO connections, etc */
2508 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Johan Hedberg2784eb42011-01-21 13:56:35 +02002509
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002510 err = cmd_complete(sk, hdev->id, MGMT_OP_GET_CONNECTIONS, 0, rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002511 rp_len);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002512
Johan Hedberga38528f2011-01-22 06:46:43 +02002513 kfree(rp);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002514
2515unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002516 hci_dev_unlock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002517 return err;
2518}
2519
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002520static int send_pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002521 struct mgmt_cp_pin_code_neg_reply *cp)
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002522{
2523 struct pending_cmd *cmd;
2524 int err;
2525
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002526 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_NEG_REPLY, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002527 sizeof(*cp));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002528 if (!cmd)
2529 return -ENOMEM;
2530
Johan Hedbergd8457692012-02-17 14:24:57 +02002531 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_NEG_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002532 sizeof(cp->addr.bdaddr), &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002533 if (err < 0)
2534 mgmt_pending_remove(cmd);
2535
2536 return err;
2537}
2538
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002539static int pin_code_reply(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002540 u16 len)
Johan Hedberg980e1a52011-01-22 06:10:07 +02002541{
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002542 struct hci_conn *conn;
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002543 struct mgmt_cp_pin_code_reply *cp = data;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002544 struct hci_cp_pin_code_reply reply;
Johan Hedberg366a0332011-02-19 12:05:55 -03002545 struct pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002546 int err;
2547
2548 BT_DBG("");
2549
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002550 hci_dev_lock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002551
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002552 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002553 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002554 MGMT_STATUS_NOT_POWERED);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002555 goto failed;
2556 }
2557
Johan Hedbergd8457692012-02-17 14:24:57 +02002558 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002559 if (!conn) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002560 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002561 MGMT_STATUS_NOT_CONNECTED);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002562 goto failed;
2563 }
2564
2565 if (conn->pending_sec_level == BT_SECURITY_HIGH && cp->pin_len != 16) {
Johan Hedbergd8457692012-02-17 14:24:57 +02002566 struct mgmt_cp_pin_code_neg_reply ncp;
2567
2568 memcpy(&ncp.addr, &cp->addr, sizeof(ncp.addr));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002569
2570 BT_ERR("PIN code is not 16 bytes long");
2571
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002572 err = send_pin_code_neg_reply(sk, hdev, &ncp);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002573 if (err >= 0)
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002574 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002575 MGMT_STATUS_INVALID_PARAMS);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002576
2577 goto failed;
2578 }
2579
Gustavo F. Padovan00abfe42012-03-01 00:37:10 -03002580 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_REPLY, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002581 if (!cmd) {
2582 err = -ENOMEM;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002583 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002584 }
Johan Hedberg980e1a52011-01-22 06:10:07 +02002585
Johan Hedbergd8457692012-02-17 14:24:57 +02002586 bacpy(&reply.bdaddr, &cp->addr.bdaddr);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002587 reply.pin_len = cp->pin_len;
Waldemar Rymarkiewicz24718ca2011-06-01 17:28:47 +02002588 memcpy(reply.pin_code, cp->pin_code, sizeof(reply.pin_code));
Johan Hedberg980e1a52011-01-22 06:10:07 +02002589
2590 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_REPLY, sizeof(reply), &reply);
2591 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002592 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002593
2594failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002595 hci_dev_unlock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002596 return err;
2597}
2598
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002599static int set_io_capability(struct sock *sk, struct hci_dev *hdev, void *data,
2600 u16 len)
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002601{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002602 struct mgmt_cp_set_io_capability *cp = data;
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002603
2604 BT_DBG("");
2605
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002606 hci_dev_lock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002607
2608 hdev->io_capability = cp->io_capability;
2609
2610 BT_DBG("%s IO capability set to 0x%02x", hdev->name,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002611 hdev->io_capability);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002612
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002613 hci_dev_unlock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002614
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002615 return cmd_complete(sk, hdev->id, MGMT_OP_SET_IO_CAPABILITY, 0, NULL,
2616 0);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002617}
2618
Gustavo Padovan6039aa732012-05-23 04:04:18 -03002619static struct pending_cmd *find_pairing(struct hci_conn *conn)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002620{
2621 struct hci_dev *hdev = conn->hdev;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +02002622 struct pending_cmd *cmd;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002623
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002624 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
Johan Hedberge9a416b2011-02-19 12:05:56 -03002625 if (cmd->opcode != MGMT_OP_PAIR_DEVICE)
2626 continue;
2627
Johan Hedberge9a416b2011-02-19 12:05:56 -03002628 if (cmd->user_data != conn)
2629 continue;
2630
2631 return cmd;
2632 }
2633
2634 return NULL;
2635}
2636
2637static void pairing_complete(struct pending_cmd *cmd, u8 status)
2638{
2639 struct mgmt_rp_pair_device rp;
2640 struct hci_conn *conn = cmd->user_data;
2641
Johan Hedbergba4e5642011-11-11 00:07:34 +02002642 bacpy(&rp.addr.bdaddr, &conn->dst);
Andre Guedes57c14772012-04-24 21:02:50 -03002643 rp.addr.type = link_to_bdaddr(conn->type, conn->dst_type);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002644
Johan Hedbergaee9b2182012-02-18 15:07:59 +02002645 cmd_complete(cmd->sk, cmd->index, MGMT_OP_PAIR_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002646 &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002647
2648 /* So we don't get further callbacks for this connection */
2649 conn->connect_cfm_cb = NULL;
2650 conn->security_cfm_cb = NULL;
2651 conn->disconn_cfm_cb = NULL;
2652
David Herrmann76a68ba2013-04-06 20:28:37 +02002653 hci_conn_drop(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002654
Johan Hedberga664b5b2011-02-19 12:06:02 -03002655 mgmt_pending_remove(cmd);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002656}
2657
2658static void pairing_complete_cb(struct hci_conn *conn, u8 status)
2659{
2660 struct pending_cmd *cmd;
2661
2662 BT_DBG("status %u", status);
2663
Johan Hedberg56e5cb82011-11-08 20:40:16 +02002664 cmd = find_pairing(conn);
2665 if (!cmd)
2666 BT_DBG("Unable to find a pending command");
2667 else
Johan Hedberge2113262012-02-18 15:20:03 +02002668 pairing_complete(cmd, mgmt_status(status));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002669}
2670
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302671static void le_connect_complete_cb(struct hci_conn *conn, u8 status)
2672{
2673 struct pending_cmd *cmd;
2674
2675 BT_DBG("status %u", status);
2676
2677 if (!status)
2678 return;
2679
2680 cmd = find_pairing(conn);
2681 if (!cmd)
2682 BT_DBG("Unable to find a pending command");
2683 else
2684 pairing_complete(cmd, mgmt_status(status));
2685}
2686
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002687static int pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002688 u16 len)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002689{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002690 struct mgmt_cp_pair_device *cp = data;
Johan Hedberg1425acb2011-11-11 00:07:35 +02002691 struct mgmt_rp_pair_device rp;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002692 struct pending_cmd *cmd;
2693 u8 sec_level, auth_type;
2694 struct hci_conn *conn;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002695 int err;
2696
2697 BT_DBG("");
2698
Szymon Jancf950a30e2013-01-18 12:48:07 +01002699 memset(&rp, 0, sizeof(rp));
2700 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2701 rp.addr.type = cp->addr.type;
2702
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002703 if (!bdaddr_type_is_valid(cp->addr.type))
2704 return cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2705 MGMT_STATUS_INVALID_PARAMS,
2706 &rp, sizeof(rp));
2707
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002708 hci_dev_lock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002709
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002710 if (!hdev_is_powered(hdev)) {
Szymon Jancf950a30e2013-01-18 12:48:07 +01002711 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2712 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002713 goto unlock;
2714 }
2715
Vinicius Costa Gomesc908df32011-09-02 14:51:22 -03002716 sec_level = BT_SECURITY_MEDIUM;
2717 if (cp->io_cap == 0x03)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002718 auth_type = HCI_AT_DEDICATED_BONDING;
Vinicius Costa Gomesc908df32011-09-02 14:51:22 -03002719 else
Johan Hedberge9a416b2011-02-19 12:05:56 -03002720 auth_type = HCI_AT_DEDICATED_BONDING_MITM;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002721
Andre Guedes591f47f2012-04-24 21:02:49 -03002722 if (cp->addr.type == BDADDR_BREDR)
Andre Guedesb12f62c2012-04-24 21:02:54 -03002723 conn = hci_connect(hdev, ACL_LINK, &cp->addr.bdaddr,
2724 cp->addr.type, sec_level, auth_type);
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002725 else
Andre Guedesb12f62c2012-04-24 21:02:54 -03002726 conn = hci_connect(hdev, LE_LINK, &cp->addr.bdaddr,
2727 cp->addr.type, sec_level, auth_type);
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002728
Ville Tervo30e76272011-02-22 16:10:53 -03002729 if (IS_ERR(conn)) {
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002730 int status;
2731
2732 if (PTR_ERR(conn) == -EBUSY)
2733 status = MGMT_STATUS_BUSY;
2734 else
2735 status = MGMT_STATUS_CONNECT_FAILED;
2736
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002737 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002738 status, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002739 sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002740 goto unlock;
2741 }
2742
2743 if (conn->connect_cfm_cb) {
David Herrmann76a68ba2013-04-06 20:28:37 +02002744 hci_conn_drop(conn);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002745 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002746 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002747 goto unlock;
2748 }
2749
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002750 cmd = mgmt_pending_add(sk, MGMT_OP_PAIR_DEVICE, hdev, data, len);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002751 if (!cmd) {
2752 err = -ENOMEM;
David Herrmann76a68ba2013-04-06 20:28:37 +02002753 hci_conn_drop(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002754 goto unlock;
2755 }
2756
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002757 /* For LE, just connecting isn't a proof that the pairing finished */
Andre Guedes591f47f2012-04-24 21:02:49 -03002758 if (cp->addr.type == BDADDR_BREDR)
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002759 conn->connect_cfm_cb = pairing_complete_cb;
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302760 else
2761 conn->connect_cfm_cb = le_connect_complete_cb;
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002762
Johan Hedberge9a416b2011-02-19 12:05:56 -03002763 conn->security_cfm_cb = pairing_complete_cb;
2764 conn->disconn_cfm_cb = pairing_complete_cb;
2765 conn->io_capability = cp->io_cap;
2766 cmd->user_data = conn;
2767
2768 if (conn->state == BT_CONNECTED &&
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002769 hci_conn_security(conn, sec_level, auth_type))
Johan Hedberge9a416b2011-02-19 12:05:56 -03002770 pairing_complete(cmd, 0);
2771
2772 err = 0;
2773
2774unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002775 hci_dev_unlock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002776 return err;
2777}
2778
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002779static int cancel_pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
2780 u16 len)
Johan Hedberg28424702012-02-02 04:02:29 +02002781{
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02002782 struct mgmt_addr_info *addr = data;
Johan Hedberg28424702012-02-02 04:02:29 +02002783 struct pending_cmd *cmd;
2784 struct hci_conn *conn;
2785 int err;
2786
2787 BT_DBG("");
2788
Johan Hedberg28424702012-02-02 04:02:29 +02002789 hci_dev_lock(hdev);
2790
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002791 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002792 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002793 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002794 goto unlock;
2795 }
2796
Johan Hedberg28424702012-02-02 04:02:29 +02002797 cmd = mgmt_pending_find(MGMT_OP_PAIR_DEVICE, hdev);
2798 if (!cmd) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002799 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002800 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002801 goto unlock;
2802 }
2803
2804 conn = cmd->user_data;
2805
2806 if (bacmp(&addr->bdaddr, &conn->dst) != 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002807 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002808 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002809 goto unlock;
2810 }
2811
2812 pairing_complete(cmd, MGMT_STATUS_CANCELLED);
2813
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002814 err = cmd_complete(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002815 addr, sizeof(*addr));
Johan Hedberg28424702012-02-02 04:02:29 +02002816unlock:
2817 hci_dev_unlock(hdev);
Johan Hedberg28424702012-02-02 04:02:29 +02002818 return err;
2819}
2820
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002821static int user_pairing_resp(struct sock *sk, struct hci_dev *hdev,
Johan Hedberg1707c602013-03-15 17:07:15 -05002822 struct mgmt_addr_info *addr, u16 mgmt_op,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002823 u16 hci_op, __le32 passkey)
Johan Hedberga5c29682011-02-19 12:05:57 -03002824{
Johan Hedberga5c29682011-02-19 12:05:57 -03002825 struct pending_cmd *cmd;
Brian Gix0df4c182011-11-16 13:53:13 -08002826 struct hci_conn *conn;
Johan Hedberga5c29682011-02-19 12:05:57 -03002827 int err;
2828
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002829 hci_dev_lock(hdev);
Johan Hedberg08ba5382011-03-16 14:29:34 +02002830
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002831 if (!hdev_is_powered(hdev)) {
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002832 err = cmd_complete(sk, hdev->id, mgmt_op,
2833 MGMT_STATUS_NOT_POWERED, addr,
2834 sizeof(*addr));
Brian Gix0df4c182011-11-16 13:53:13 -08002835 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002836 }
2837
Johan Hedberg1707c602013-03-15 17:07:15 -05002838 if (addr->type == BDADDR_BREDR)
2839 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &addr->bdaddr);
Johan Hedberg272d90d2012-02-09 15:26:12 +02002840 else
Johan Hedberg1707c602013-03-15 17:07:15 -05002841 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &addr->bdaddr);
Brian Gix47c15e22011-11-16 13:53:14 -08002842
Johan Hedberg272d90d2012-02-09 15:26:12 +02002843 if (!conn) {
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002844 err = cmd_complete(sk, hdev->id, mgmt_op,
2845 MGMT_STATUS_NOT_CONNECTED, addr,
2846 sizeof(*addr));
Johan Hedberg272d90d2012-02-09 15:26:12 +02002847 goto done;
2848 }
2849
Johan Hedberg1707c602013-03-15 17:07:15 -05002850 if (addr->type == BDADDR_LE_PUBLIC || addr->type == BDADDR_LE_RANDOM) {
Brian Gix47c15e22011-11-16 13:53:14 -08002851 /* Continue with pairing via SMP */
Brian Gix5fe57d92011-12-21 16:12:13 -08002852 err = smp_user_confirm_reply(conn, mgmt_op, passkey);
Brian Gix47c15e22011-11-16 13:53:14 -08002853
Brian Gix5fe57d92011-12-21 16:12:13 -08002854 if (!err)
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002855 err = cmd_complete(sk, hdev->id, mgmt_op,
2856 MGMT_STATUS_SUCCESS, addr,
2857 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002858 else
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002859 err = cmd_complete(sk, hdev->id, mgmt_op,
2860 MGMT_STATUS_FAILED, addr,
2861 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002862
Brian Gix47c15e22011-11-16 13:53:14 -08002863 goto done;
2864 }
2865
Johan Hedberg1707c602013-03-15 17:07:15 -05002866 cmd = mgmt_pending_add(sk, mgmt_op, hdev, addr, sizeof(*addr));
Johan Hedberga5c29682011-02-19 12:05:57 -03002867 if (!cmd) {
2868 err = -ENOMEM;
Brian Gix0df4c182011-11-16 13:53:13 -08002869 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002870 }
2871
Brian Gix0df4c182011-11-16 13:53:13 -08002872 /* Continue with pairing via HCI */
Brian Gix604086b2011-11-23 08:28:33 -08002873 if (hci_op == HCI_OP_USER_PASSKEY_REPLY) {
2874 struct hci_cp_user_passkey_reply cp;
2875
Johan Hedberg1707c602013-03-15 17:07:15 -05002876 bacpy(&cp.bdaddr, &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002877 cp.passkey = passkey;
2878 err = hci_send_cmd(hdev, hci_op, sizeof(cp), &cp);
2879 } else
Johan Hedberg1707c602013-03-15 17:07:15 -05002880 err = hci_send_cmd(hdev, hci_op, sizeof(addr->bdaddr),
2881 &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002882
Johan Hedberga664b5b2011-02-19 12:06:02 -03002883 if (err < 0)
2884 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03002885
Brian Gix0df4c182011-11-16 13:53:13 -08002886done:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002887 hci_dev_unlock(hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03002888 return err;
2889}
2890
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05302891static int pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
2892 void *data, u16 len)
2893{
2894 struct mgmt_cp_pin_code_neg_reply *cp = data;
2895
2896 BT_DBG("");
2897
Johan Hedberg1707c602013-03-15 17:07:15 -05002898 return user_pairing_resp(sk, hdev, &cp->addr,
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05302899 MGMT_OP_PIN_CODE_NEG_REPLY,
2900 HCI_OP_PIN_CODE_NEG_REPLY, 0);
2901}
2902
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002903static int user_confirm_reply(struct sock *sk, struct hci_dev *hdev, void *data,
2904 u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08002905{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002906 struct mgmt_cp_user_confirm_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08002907
2908 BT_DBG("");
2909
2910 if (len != sizeof(*cp))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002911 return cmd_status(sk, hdev->id, MGMT_OP_USER_CONFIRM_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002912 MGMT_STATUS_INVALID_PARAMS);
Brian Gix0df4c182011-11-16 13:53:13 -08002913
Johan Hedberg1707c602013-03-15 17:07:15 -05002914 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002915 MGMT_OP_USER_CONFIRM_REPLY,
2916 HCI_OP_USER_CONFIRM_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08002917}
2918
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002919static int user_confirm_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002920 void *data, u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08002921{
Johan Hedbergc9c26592011-12-15 00:47:41 +02002922 struct mgmt_cp_user_confirm_neg_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08002923
2924 BT_DBG("");
2925
Johan Hedberg1707c602013-03-15 17:07:15 -05002926 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002927 MGMT_OP_USER_CONFIRM_NEG_REPLY,
2928 HCI_OP_USER_CONFIRM_NEG_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08002929}
2930
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002931static int user_passkey_reply(struct sock *sk, struct hci_dev *hdev, void *data,
2932 u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08002933{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002934 struct mgmt_cp_user_passkey_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08002935
2936 BT_DBG("");
2937
Johan Hedberg1707c602013-03-15 17:07:15 -05002938 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002939 MGMT_OP_USER_PASSKEY_REPLY,
2940 HCI_OP_USER_PASSKEY_REPLY, cp->passkey);
Brian Gix604086b2011-11-23 08:28:33 -08002941}
2942
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002943static int user_passkey_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002944 void *data, u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08002945{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002946 struct mgmt_cp_user_passkey_neg_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08002947
2948 BT_DBG("");
2949
Johan Hedberg1707c602013-03-15 17:07:15 -05002950 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002951 MGMT_OP_USER_PASSKEY_NEG_REPLY,
2952 HCI_OP_USER_PASSKEY_NEG_REPLY, 0);
Brian Gix604086b2011-11-23 08:28:33 -08002953}
2954
Johan Hedberg13928972013-03-15 17:07:00 -05002955static void update_name(struct hci_request *req)
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002956{
Johan Hedberg13928972013-03-15 17:07:00 -05002957 struct hci_dev *hdev = req->hdev;
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002958 struct hci_cp_write_local_name cp;
2959
Johan Hedberg13928972013-03-15 17:07:00 -05002960 memcpy(cp.name, hdev->dev_name, sizeof(cp.name));
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002961
Johan Hedberg890ea892013-03-15 17:06:52 -05002962 hci_req_add(req, HCI_OP_WRITE_LOCAL_NAME, sizeof(cp), &cp);
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002963}
2964
Johan Hedberg13928972013-03-15 17:07:00 -05002965static void set_name_complete(struct hci_dev *hdev, u8 status)
2966{
2967 struct mgmt_cp_set_local_name *cp;
2968 struct pending_cmd *cmd;
2969
2970 BT_DBG("status 0x%02x", status);
2971
2972 hci_dev_lock(hdev);
2973
2974 cmd = mgmt_pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
2975 if (!cmd)
2976 goto unlock;
2977
2978 cp = cmd->param;
2979
2980 if (status)
2981 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME,
2982 mgmt_status(status));
2983 else
2984 cmd_complete(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
2985 cp, sizeof(*cp));
2986
2987 mgmt_pending_remove(cmd);
2988
2989unlock:
2990 hci_dev_unlock(hdev);
2991}
2992
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002993static int set_local_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002994 u16 len)
Johan Hedbergb312b1612011-03-16 14:29:37 +02002995{
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002996 struct mgmt_cp_set_local_name *cp = data;
Johan Hedbergb312b1612011-03-16 14:29:37 +02002997 struct pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05002998 struct hci_request req;
Johan Hedbergb312b1612011-03-16 14:29:37 +02002999 int err;
3000
3001 BT_DBG("");
3002
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003003 hci_dev_lock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003004
Johan Hedbergb3f2ca92013-03-15 17:07:03 -05003005 /* If the old values are the same as the new ones just return a
3006 * direct command complete event.
3007 */
3008 if (!memcmp(hdev->dev_name, cp->name, sizeof(hdev->dev_name)) &&
3009 !memcmp(hdev->short_name, cp->short_name,
3010 sizeof(hdev->short_name))) {
3011 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
3012 data, len);
3013 goto failed;
3014 }
3015
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003016 memcpy(hdev->short_name, cp->short_name, sizeof(hdev->short_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003017
Johan Hedbergb5235a62012-02-21 14:32:24 +02003018 if (!hdev_is_powered(hdev)) {
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003019 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003020
3021 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003022 data, len);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003023 if (err < 0)
3024 goto failed;
3025
3026 err = mgmt_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, data, len,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003027 sk);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003028
Johan Hedbergb5235a62012-02-21 14:32:24 +02003029 goto failed;
3030 }
3031
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003032 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LOCAL_NAME, hdev, data, len);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003033 if (!cmd) {
3034 err = -ENOMEM;
3035 goto failed;
3036 }
3037
Johan Hedberg13928972013-03-15 17:07:00 -05003038 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
3039
Johan Hedberg890ea892013-03-15 17:06:52 -05003040 hci_req_init(&req, hdev);
Johan Hedberg3f985052013-03-15 17:07:02 -05003041
3042 if (lmp_bredr_capable(hdev)) {
3043 update_name(&req);
3044 update_eir(&req);
3045 }
3046
Marcel Holtmann7a5f4992013-10-16 00:16:49 -07003047 /* The name is stored in the scan response data and so
3048 * no need to udpate the advertising data here.
3049 */
Johan Hedberg3f985052013-03-15 17:07:02 -05003050 if (lmp_le_capable(hdev))
Marcel Holtmann7a5f4992013-10-16 00:16:49 -07003051 update_scan_rsp_data(&req);
Johan Hedberg3f985052013-03-15 17:07:02 -05003052
Johan Hedberg13928972013-03-15 17:07:00 -05003053 err = hci_req_run(&req, set_name_complete);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003054 if (err < 0)
3055 mgmt_pending_remove(cmd);
3056
3057failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003058 hci_dev_unlock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003059 return err;
3060}
3061
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02003062static int read_local_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003063 void *data, u16 data_len)
Szymon Jancc35938b2011-03-22 13:12:21 +01003064{
Szymon Jancc35938b2011-03-22 13:12:21 +01003065 struct pending_cmd *cmd;
3066 int err;
3067
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003068 BT_DBG("%s", hdev->name);
Szymon Jancc35938b2011-03-22 13:12:21 +01003069
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003070 hci_dev_lock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003071
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003072 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003073 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003074 MGMT_STATUS_NOT_POWERED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003075 goto unlock;
3076 }
3077
Andre Guedes9a1a1992012-07-24 15:03:48 -03003078 if (!lmp_ssp_capable(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003079 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003080 MGMT_STATUS_NOT_SUPPORTED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003081 goto unlock;
3082 }
3083
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003084 if (mgmt_pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003085 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003086 MGMT_STATUS_BUSY);
Szymon Jancc35938b2011-03-22 13:12:21 +01003087 goto unlock;
3088 }
3089
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003090 cmd = mgmt_pending_add(sk, MGMT_OP_READ_LOCAL_OOB_DATA, hdev, NULL, 0);
Szymon Jancc35938b2011-03-22 13:12:21 +01003091 if (!cmd) {
3092 err = -ENOMEM;
3093 goto unlock;
3094 }
3095
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08003096 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags))
3097 err = hci_send_cmd(hdev, HCI_OP_READ_LOCAL_OOB_EXT_DATA,
3098 0, NULL);
3099 else
3100 err = hci_send_cmd(hdev, HCI_OP_READ_LOCAL_OOB_DATA, 0, NULL);
3101
Szymon Jancc35938b2011-03-22 13:12:21 +01003102 if (err < 0)
3103 mgmt_pending_remove(cmd);
3104
3105unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003106 hci_dev_unlock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003107 return err;
3108}
3109
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003110static int add_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003111 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003112{
Szymon Janc2763eda2011-03-22 13:12:22 +01003113 int err;
3114
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003115 BT_DBG("%s ", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003116
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003117 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003118
Marcel Holtmannec109112014-01-10 02:07:30 -08003119 if (len == MGMT_ADD_REMOTE_OOB_DATA_SIZE) {
3120 struct mgmt_cp_add_remote_oob_data *cp = data;
3121 u8 status;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003122
Marcel Holtmannec109112014-01-10 02:07:30 -08003123 err = hci_add_remote_oob_data(hdev, &cp->addr.bdaddr,
3124 cp->hash, cp->randomizer);
3125 if (err < 0)
3126 status = MGMT_STATUS_FAILED;
3127 else
3128 status = MGMT_STATUS_SUCCESS;
3129
3130 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3131 status, &cp->addr, sizeof(cp->addr));
3132 } else if (len == MGMT_ADD_REMOTE_OOB_EXT_DATA_SIZE) {
3133 struct mgmt_cp_add_remote_oob_ext_data *cp = data;
3134 u8 status;
3135
3136 err = hci_add_remote_oob_ext_data(hdev, &cp->addr.bdaddr,
3137 cp->hash192,
3138 cp->randomizer192,
3139 cp->hash256,
3140 cp->randomizer256);
3141 if (err < 0)
3142 status = MGMT_STATUS_FAILED;
3143 else
3144 status = MGMT_STATUS_SUCCESS;
3145
3146 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3147 status, &cp->addr, sizeof(cp->addr));
3148 } else {
3149 BT_ERR("add_remote_oob_data: invalid length of %u bytes", len);
3150 err = cmd_status(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3151 MGMT_STATUS_INVALID_PARAMS);
3152 }
Szymon Janc2763eda2011-03-22 13:12:22 +01003153
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003154 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003155 return err;
3156}
3157
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003158static int remove_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003159 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003160{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003161 struct mgmt_cp_remove_remote_oob_data *cp = data;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003162 u8 status;
Szymon Janc2763eda2011-03-22 13:12:22 +01003163 int err;
3164
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003165 BT_DBG("%s", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003166
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003167 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003168
Johan Hedberg664ce4c2012-02-09 15:44:09 +02003169 err = hci_remove_remote_oob_data(hdev, &cp->addr.bdaddr);
Szymon Janc2763eda2011-03-22 13:12:22 +01003170 if (err < 0)
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003171 status = MGMT_STATUS_INVALID_PARAMS;
Szymon Janc2763eda2011-03-22 13:12:22 +01003172 else
Szymon Janca6785be2012-12-13 15:11:21 +01003173 status = MGMT_STATUS_SUCCESS;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003174
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003175 err = cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_REMOTE_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003176 status, &cp->addr, sizeof(cp->addr));
Szymon Janc2763eda2011-03-22 13:12:22 +01003177
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003178 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003179 return err;
3180}
3181
Andre Guedes41dc2bd2013-04-30 15:29:30 -03003182static int mgmt_start_discovery_failed(struct hci_dev *hdev, u8 status)
3183{
3184 struct pending_cmd *cmd;
3185 u8 type;
3186 int err;
3187
3188 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3189
3190 cmd = mgmt_pending_find(MGMT_OP_START_DISCOVERY, hdev);
3191 if (!cmd)
3192 return -ENOENT;
3193
3194 type = hdev->discovery.type;
3195
3196 err = cmd_complete(cmd->sk, hdev->id, cmd->opcode, mgmt_status(status),
3197 &type, sizeof(type));
3198 mgmt_pending_remove(cmd);
3199
3200 return err;
3201}
3202
Andre Guedes7c307722013-04-30 15:29:28 -03003203static void start_discovery_complete(struct hci_dev *hdev, u8 status)
3204{
3205 BT_DBG("status %d", status);
3206
3207 if (status) {
3208 hci_dev_lock(hdev);
3209 mgmt_start_discovery_failed(hdev, status);
3210 hci_dev_unlock(hdev);
3211 return;
3212 }
3213
3214 hci_dev_lock(hdev);
3215 hci_discovery_set_state(hdev, DISCOVERY_FINDING);
3216 hci_dev_unlock(hdev);
3217
3218 switch (hdev->discovery.type) {
3219 case DISCOV_TYPE_LE:
3220 queue_delayed_work(hdev->workqueue, &hdev->le_scan_disable,
Andre Guedes0d8cc932013-04-30 15:29:31 -03003221 DISCOV_LE_TIMEOUT);
Andre Guedes7c307722013-04-30 15:29:28 -03003222 break;
3223
3224 case DISCOV_TYPE_INTERLEAVED:
3225 queue_delayed_work(hdev->workqueue, &hdev->le_scan_disable,
Andre Guedes0d8cc932013-04-30 15:29:31 -03003226 DISCOV_INTERLEAVED_TIMEOUT);
Andre Guedes7c307722013-04-30 15:29:28 -03003227 break;
3228
3229 case DISCOV_TYPE_BREDR:
3230 break;
3231
3232 default:
3233 BT_ERR("Invalid discovery type %d", hdev->discovery.type);
3234 }
3235}
3236
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003237static int start_discovery(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003238 void *data, u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003239{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003240 struct mgmt_cp_start_discovery *cp = data;
Johan Hedberg14a53662011-04-27 10:29:56 -04003241 struct pending_cmd *cmd;
Andre Guedes7c307722013-04-30 15:29:28 -03003242 struct hci_cp_le_set_scan_param param_cp;
3243 struct hci_cp_le_set_scan_enable enable_cp;
3244 struct hci_cp_inquiry inq_cp;
3245 struct hci_request req;
3246 /* General inquiry access code (GIAC) */
3247 u8 lap[3] = { 0x33, 0x8b, 0x9e };
Johan Hedberge6fe7982013-10-02 15:45:22 +03003248 u8 status;
Johan Hedberg14a53662011-04-27 10:29:56 -04003249 int err;
3250
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003251 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003252
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003253 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003254
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003255 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003256 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003257 MGMT_STATUS_NOT_POWERED);
Johan Hedbergbd2d1332011-11-07 23:13:37 +02003258 goto failed;
3259 }
3260
Andre Guedes642be6c2012-03-21 00:03:37 -03003261 if (test_bit(HCI_PERIODIC_INQ, &hdev->dev_flags)) {
3262 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3263 MGMT_STATUS_BUSY);
3264 goto failed;
3265 }
3266
Johan Hedbergff9ef572012-01-04 14:23:45 +02003267 if (hdev->discovery.state != DISCOVERY_STOPPED) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003268 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003269 MGMT_STATUS_BUSY);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003270 goto failed;
3271 }
3272
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003273 cmd = mgmt_pending_add(sk, MGMT_OP_START_DISCOVERY, hdev, NULL, 0);
Johan Hedberg14a53662011-04-27 10:29:56 -04003274 if (!cmd) {
3275 err = -ENOMEM;
3276 goto failed;
3277 }
3278
Andre Guedes4aab14e2012-02-17 20:39:36 -03003279 hdev->discovery.type = cp->type;
3280
Andre Guedes7c307722013-04-30 15:29:28 -03003281 hci_req_init(&req, hdev);
3282
Andre Guedes4aab14e2012-02-17 20:39:36 -03003283 switch (hdev->discovery.type) {
Andre Guedesf39799f2012-02-17 20:39:35 -03003284 case DISCOV_TYPE_BREDR:
Johan Hedberge6fe7982013-10-02 15:45:22 +03003285 status = mgmt_bredr_support(hdev);
3286 if (status) {
Johan Hedberg04106752013-01-10 14:54:09 +02003287 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003288 status);
Johan Hedberg04106752013-01-10 14:54:09 +02003289 mgmt_pending_remove(cmd);
3290 goto failed;
3291 }
3292
Andre Guedes7c307722013-04-30 15:29:28 -03003293 if (test_bit(HCI_INQUIRY, &hdev->flags)) {
3294 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3295 MGMT_STATUS_BUSY);
3296 mgmt_pending_remove(cmd);
3297 goto failed;
3298 }
3299
3300 hci_inquiry_cache_flush(hdev);
3301
3302 memset(&inq_cp, 0, sizeof(inq_cp));
3303 memcpy(&inq_cp.lap, lap, sizeof(inq_cp.lap));
Andre Guedes0d8cc932013-04-30 15:29:31 -03003304 inq_cp.length = DISCOV_BREDR_INQUIRY_LEN;
Andre Guedes7c307722013-04-30 15:29:28 -03003305 hci_req_add(&req, HCI_OP_INQUIRY, sizeof(inq_cp), &inq_cp);
Andre Guedesf39799f2012-02-17 20:39:35 -03003306 break;
3307
3308 case DISCOV_TYPE_LE:
Andre Guedes7c307722013-04-30 15:29:28 -03003309 case DISCOV_TYPE_INTERLEAVED:
Johan Hedberge6fe7982013-10-02 15:45:22 +03003310 status = mgmt_le_support(hdev);
3311 if (status) {
Johan Hedberg04106752013-01-10 14:54:09 +02003312 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003313 status);
Johan Hedberg04106752013-01-10 14:54:09 +02003314 mgmt_pending_remove(cmd);
3315 goto failed;
3316 }
3317
Andre Guedes7c307722013-04-30 15:29:28 -03003318 if (hdev->discovery.type == DISCOV_TYPE_INTERLEAVED &&
Johan Hedberg56f87902013-10-02 13:43:13 +03003319 !test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
Johan Hedberg04106752013-01-10 14:54:09 +02003320 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3321 MGMT_STATUS_NOT_SUPPORTED);
3322 mgmt_pending_remove(cmd);
3323 goto failed;
3324 }
3325
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003326 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
Andre Guedes7c307722013-04-30 15:29:28 -03003327 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3328 MGMT_STATUS_REJECTED);
3329 mgmt_pending_remove(cmd);
3330 goto failed;
3331 }
3332
3333 if (test_bit(HCI_LE_SCAN, &hdev->dev_flags)) {
3334 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3335 MGMT_STATUS_BUSY);
3336 mgmt_pending_remove(cmd);
3337 goto failed;
3338 }
3339
3340 memset(&param_cp, 0, sizeof(param_cp));
3341 param_cp.type = LE_SCAN_ACTIVE;
Andre Guedes0d8cc932013-04-30 15:29:31 -03003342 param_cp.interval = cpu_to_le16(DISCOV_LE_SCAN_INT);
3343 param_cp.window = cpu_to_le16(DISCOV_LE_SCAN_WIN);
Marcel Holtmann79830f62013-10-18 16:38:09 -07003344 param_cp.own_address_type = hdev->own_addr_type;
Andre Guedes7c307722013-04-30 15:29:28 -03003345 hci_req_add(&req, HCI_OP_LE_SET_SCAN_PARAM, sizeof(param_cp),
3346 &param_cp);
3347
3348 memset(&enable_cp, 0, sizeof(enable_cp));
3349 enable_cp.enable = LE_SCAN_ENABLE;
3350 enable_cp.filter_dup = LE_SCAN_FILTER_DUP_ENABLE;
3351 hci_req_add(&req, HCI_OP_LE_SET_SCAN_ENABLE, sizeof(enable_cp),
3352 &enable_cp);
Andre Guedes5e0452c2012-02-17 20:39:38 -03003353 break;
3354
Andre Guedesf39799f2012-02-17 20:39:35 -03003355 default:
Johan Hedberg04106752013-01-10 14:54:09 +02003356 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3357 MGMT_STATUS_INVALID_PARAMS);
3358 mgmt_pending_remove(cmd);
3359 goto failed;
Andre Guedesf39799f2012-02-17 20:39:35 -03003360 }
Andre Guedes3fd24152012-02-03 17:48:01 -03003361
Andre Guedes7c307722013-04-30 15:29:28 -03003362 err = hci_req_run(&req, start_discovery_complete);
Johan Hedberg14a53662011-04-27 10:29:56 -04003363 if (err < 0)
3364 mgmt_pending_remove(cmd);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003365 else
3366 hci_discovery_set_state(hdev, DISCOVERY_STARTING);
Johan Hedberg14a53662011-04-27 10:29:56 -04003367
3368failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003369 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003370 return err;
3371}
3372
Andre Guedes1183fdc2013-04-30 15:29:35 -03003373static int mgmt_stop_discovery_failed(struct hci_dev *hdev, u8 status)
3374{
3375 struct pending_cmd *cmd;
3376 int err;
3377
3378 cmd = mgmt_pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
3379 if (!cmd)
3380 return -ENOENT;
3381
3382 err = cmd_complete(cmd->sk, hdev->id, cmd->opcode, mgmt_status(status),
3383 &hdev->discovery.type, sizeof(hdev->discovery.type));
3384 mgmt_pending_remove(cmd);
3385
3386 return err;
3387}
3388
Andre Guedes0e05bba2013-04-30 15:29:33 -03003389static void stop_discovery_complete(struct hci_dev *hdev, u8 status)
3390{
3391 BT_DBG("status %d", status);
3392
3393 hci_dev_lock(hdev);
3394
3395 if (status) {
3396 mgmt_stop_discovery_failed(hdev, status);
3397 goto unlock;
3398 }
3399
3400 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3401
3402unlock:
3403 hci_dev_unlock(hdev);
3404}
3405
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003406static int stop_discovery(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003407 u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003408{
Johan Hedbergd9306502012-02-20 23:25:18 +02003409 struct mgmt_cp_stop_discovery *mgmt_cp = data;
Johan Hedberg14a53662011-04-27 10:29:56 -04003410 struct pending_cmd *cmd;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003411 struct hci_cp_remote_name_req_cancel cp;
3412 struct inquiry_entry *e;
Andre Guedes0e05bba2013-04-30 15:29:33 -03003413 struct hci_request req;
3414 struct hci_cp_le_set_scan_enable enable_cp;
Johan Hedberg14a53662011-04-27 10:29:56 -04003415 int err;
3416
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003417 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003418
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003419 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003420
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003421 if (!hci_discovery_active(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003422 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003423 MGMT_STATUS_REJECTED, &mgmt_cp->type,
3424 sizeof(mgmt_cp->type));
Johan Hedbergd9306502012-02-20 23:25:18 +02003425 goto unlock;
3426 }
3427
3428 if (hdev->discovery.type != mgmt_cp->type) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003429 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003430 MGMT_STATUS_INVALID_PARAMS, &mgmt_cp->type,
3431 sizeof(mgmt_cp->type));
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003432 goto unlock;
Johan Hedbergff9ef572012-01-04 14:23:45 +02003433 }
3434
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003435 cmd = mgmt_pending_add(sk, MGMT_OP_STOP_DISCOVERY, hdev, NULL, 0);
Johan Hedberg14a53662011-04-27 10:29:56 -04003436 if (!cmd) {
3437 err = -ENOMEM;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003438 goto unlock;
Johan Hedberg14a53662011-04-27 10:29:56 -04003439 }
3440
Andre Guedes0e05bba2013-04-30 15:29:33 -03003441 hci_req_init(&req, hdev);
3442
Andre Guedese0d9727e2012-03-20 15:15:36 -03003443 switch (hdev->discovery.state) {
3444 case DISCOVERY_FINDING:
Andre Guedes0e05bba2013-04-30 15:29:33 -03003445 if (test_bit(HCI_INQUIRY, &hdev->flags)) {
3446 hci_req_add(&req, HCI_OP_INQUIRY_CANCEL, 0, NULL);
3447 } else {
3448 cancel_delayed_work(&hdev->le_scan_disable);
3449
3450 memset(&enable_cp, 0, sizeof(enable_cp));
3451 enable_cp.enable = LE_SCAN_DISABLE;
3452 hci_req_add(&req, HCI_OP_LE_SET_SCAN_ENABLE,
3453 sizeof(enable_cp), &enable_cp);
3454 }
Andre Guedesc9ecc482012-03-15 16:52:08 -03003455
Andre Guedese0d9727e2012-03-20 15:15:36 -03003456 break;
3457
3458 case DISCOVERY_RESOLVING:
3459 e = hci_inquiry_cache_lookup_resolve(hdev, BDADDR_ANY,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003460 NAME_PENDING);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003461 if (!e) {
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003462 mgmt_pending_remove(cmd);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003463 err = cmd_complete(sk, hdev->id,
3464 MGMT_OP_STOP_DISCOVERY, 0,
3465 &mgmt_cp->type,
3466 sizeof(mgmt_cp->type));
3467 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3468 goto unlock;
3469 }
3470
3471 bacpy(&cp.bdaddr, &e->data.bdaddr);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003472 hci_req_add(&req, HCI_OP_REMOTE_NAME_REQ_CANCEL, sizeof(cp),
3473 &cp);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003474
3475 break;
3476
3477 default:
3478 BT_DBG("unknown discovery state %u", hdev->discovery.state);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003479
3480 mgmt_pending_remove(cmd);
3481 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
3482 MGMT_STATUS_FAILED, &mgmt_cp->type,
3483 sizeof(mgmt_cp->type));
3484 goto unlock;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003485 }
3486
Andre Guedes0e05bba2013-04-30 15:29:33 -03003487 err = hci_req_run(&req, stop_discovery_complete);
Johan Hedberg14a53662011-04-27 10:29:56 -04003488 if (err < 0)
3489 mgmt_pending_remove(cmd);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003490 else
3491 hci_discovery_set_state(hdev, DISCOVERY_STOPPING);
Johan Hedberg14a53662011-04-27 10:29:56 -04003492
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003493unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003494 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003495 return err;
3496}
3497
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003498static int confirm_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003499 u16 len)
Johan Hedberg561aafb2012-01-04 13:31:59 +02003500{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003501 struct mgmt_cp_confirm_name *cp = data;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003502 struct inquiry_entry *e;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003503 int err;
3504
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003505 BT_DBG("%s", hdev->name);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003506
Johan Hedberg561aafb2012-01-04 13:31:59 +02003507 hci_dev_lock(hdev);
3508
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003509 if (!hci_discovery_active(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003510 err = cmd_status(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003511 MGMT_STATUS_FAILED);
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003512 goto failed;
3513 }
3514
Johan Hedberga198e7b2012-02-17 14:27:06 +02003515 e = hci_inquiry_cache_lookup_unknown(hdev, &cp->addr.bdaddr);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003516 if (!e) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003517 err = cmd_status(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003518 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003519 goto failed;
3520 }
3521
3522 if (cp->name_known) {
3523 e->name_state = NAME_KNOWN;
3524 list_del(&e->list);
3525 } else {
3526 e->name_state = NAME_NEEDED;
Johan Hedberga3d4e202012-01-09 00:53:02 +02003527 hci_inquiry_cache_update_resolve(hdev, e);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003528 }
3529
Johan Hedberge3846622013-01-09 15:29:33 +02003530 err = cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME, 0, &cp->addr,
3531 sizeof(cp->addr));
Johan Hedberg561aafb2012-01-04 13:31:59 +02003532
3533failed:
3534 hci_dev_unlock(hdev);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003535 return err;
3536}
3537
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003538static int block_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003539 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003540{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003541 struct mgmt_cp_block_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003542 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003543 int err;
3544
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003545 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003546
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003547 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg5d0846d2013-01-20 14:27:22 +02003548 return cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE,
3549 MGMT_STATUS_INVALID_PARAMS,
3550 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003551
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003552 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003553
Johan Hedberg88c1fe42012-02-09 15:56:11 +02003554 err = hci_blacklist_add(hdev, &cp->addr.bdaddr, cp->addr.type);
Antti Julku7fbec222011-06-15 12:01:15 +03003555 if (err < 0)
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003556 status = MGMT_STATUS_FAILED;
Antti Julku7fbec222011-06-15 12:01:15 +03003557 else
Szymon Janca6785be2012-12-13 15:11:21 +01003558 status = MGMT_STATUS_SUCCESS;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003559
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003560 err = cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003561 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003562
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003563 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003564
3565 return err;
3566}
3567
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003568static int unblock_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003569 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003570{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003571 struct mgmt_cp_unblock_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003572 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003573 int err;
3574
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003575 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003576
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003577 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg5d0846d2013-01-20 14:27:22 +02003578 return cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE,
3579 MGMT_STATUS_INVALID_PARAMS,
3580 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003581
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003582 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003583
Johan Hedberg88c1fe42012-02-09 15:56:11 +02003584 err = hci_blacklist_del(hdev, &cp->addr.bdaddr, cp->addr.type);
Antti Julku7fbec222011-06-15 12:01:15 +03003585 if (err < 0)
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003586 status = MGMT_STATUS_INVALID_PARAMS;
Antti Julku7fbec222011-06-15 12:01:15 +03003587 else
Szymon Janca6785be2012-12-13 15:11:21 +01003588 status = MGMT_STATUS_SUCCESS;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003589
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003590 err = cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003591 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003592
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003593 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003594
3595 return err;
3596}
3597
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003598static int set_device_id(struct sock *sk, struct hci_dev *hdev, void *data,
3599 u16 len)
3600{
3601 struct mgmt_cp_set_device_id *cp = data;
Johan Hedberg890ea892013-03-15 17:06:52 -05003602 struct hci_request req;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003603 int err;
Szymon Jancc72d4b82012-03-16 16:02:57 +01003604 __u16 source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003605
3606 BT_DBG("%s", hdev->name);
3607
Szymon Jancc72d4b82012-03-16 16:02:57 +01003608 source = __le16_to_cpu(cp->source);
3609
3610 if (source > 0x0002)
3611 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEVICE_ID,
3612 MGMT_STATUS_INVALID_PARAMS);
3613
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003614 hci_dev_lock(hdev);
3615
Szymon Jancc72d4b82012-03-16 16:02:57 +01003616 hdev->devid_source = source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003617 hdev->devid_vendor = __le16_to_cpu(cp->vendor);
3618 hdev->devid_product = __le16_to_cpu(cp->product);
3619 hdev->devid_version = __le16_to_cpu(cp->version);
3620
3621 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_DEVICE_ID, 0, NULL, 0);
3622
Johan Hedberg890ea892013-03-15 17:06:52 -05003623 hci_req_init(&req, hdev);
3624 update_eir(&req);
3625 hci_req_run(&req, NULL);
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003626
3627 hci_dev_unlock(hdev);
3628
3629 return err;
3630}
3631
Johan Hedberg4375f102013-09-25 13:26:10 +03003632static void set_advertising_complete(struct hci_dev *hdev, u8 status)
3633{
3634 struct cmd_lookup match = { NULL, hdev };
3635
3636 if (status) {
3637 u8 mgmt_err = mgmt_status(status);
3638
3639 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev,
3640 cmd_status_rsp, &mgmt_err);
3641 return;
3642 }
3643
3644 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev, settings_rsp,
3645 &match);
3646
3647 new_settings(hdev, match.sk);
3648
3649 if (match.sk)
3650 sock_put(match.sk);
3651}
3652
Marcel Holtmann21b51872013-10-10 09:47:53 -07003653static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
3654 u16 len)
Johan Hedberg4375f102013-09-25 13:26:10 +03003655{
3656 struct mgmt_mode *cp = data;
3657 struct pending_cmd *cmd;
3658 struct hci_request req;
Johan Hedberge6fe7982013-10-02 15:45:22 +03003659 u8 val, enabled, status;
Johan Hedberg4375f102013-09-25 13:26:10 +03003660 int err;
3661
3662 BT_DBG("request for %s", hdev->name);
3663
Johan Hedberge6fe7982013-10-02 15:45:22 +03003664 status = mgmt_le_support(hdev);
3665 if (status)
Johan Hedberg4375f102013-09-25 13:26:10 +03003666 return cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003667 status);
Johan Hedberg4375f102013-09-25 13:26:10 +03003668
3669 if (cp->val != 0x00 && cp->val != 0x01)
3670 return cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3671 MGMT_STATUS_INVALID_PARAMS);
3672
3673 hci_dev_lock(hdev);
3674
3675 val = !!cp->val;
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003676 enabled = test_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedberg4375f102013-09-25 13:26:10 +03003677
Johan Hedbergf74ca9b2013-10-08 15:52:18 +02003678 /* The following conditions are ones which mean that we should
3679 * not do any HCI communication but directly send a mgmt
3680 * response to user space (after toggling the flag if
3681 * necessary).
3682 */
3683 if (!hdev_is_powered(hdev) || val == enabled ||
Marcel Holtmannb145edc2013-10-10 09:47:54 -07003684 hci_conn_num(hdev, LE_LINK) > 0) {
Johan Hedberg4375f102013-09-25 13:26:10 +03003685 bool changed = false;
3686
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003687 if (val != test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
3688 change_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedberg4375f102013-09-25 13:26:10 +03003689 changed = true;
3690 }
3691
3692 err = send_settings_rsp(sk, MGMT_OP_SET_ADVERTISING, hdev);
3693 if (err < 0)
3694 goto unlock;
3695
3696 if (changed)
3697 err = new_settings(hdev, sk);
3698
3699 goto unlock;
3700 }
3701
3702 if (mgmt_pending_find(MGMT_OP_SET_ADVERTISING, hdev) ||
3703 mgmt_pending_find(MGMT_OP_SET_LE, hdev)) {
3704 err = cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3705 MGMT_STATUS_BUSY);
3706 goto unlock;
3707 }
3708
3709 cmd = mgmt_pending_add(sk, MGMT_OP_SET_ADVERTISING, hdev, data, len);
3710 if (!cmd) {
3711 err = -ENOMEM;
3712 goto unlock;
3713 }
3714
3715 hci_req_init(&req, hdev);
3716
Marcel Holtmannbba3aa52013-10-06 02:55:21 -07003717 if (val)
3718 enable_advertising(&req);
3719 else
3720 disable_advertising(&req);
Johan Hedberg4375f102013-09-25 13:26:10 +03003721
3722 err = hci_req_run(&req, set_advertising_complete);
3723 if (err < 0)
3724 mgmt_pending_remove(cmd);
3725
3726unlock:
3727 hci_dev_unlock(hdev);
3728 return err;
3729}
3730
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003731static int set_static_address(struct sock *sk, struct hci_dev *hdev,
3732 void *data, u16 len)
3733{
3734 struct mgmt_cp_set_static_address *cp = data;
3735 int err;
3736
3737 BT_DBG("%s", hdev->name);
3738
Marcel Holtmann62af4442013-10-02 22:10:32 -07003739 if (!lmp_le_capable(hdev))
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003740 return cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
Marcel Holtmann62af4442013-10-02 22:10:32 -07003741 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003742
3743 if (hdev_is_powered(hdev))
3744 return cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
3745 MGMT_STATUS_REJECTED);
3746
3747 if (bacmp(&cp->bdaddr, BDADDR_ANY)) {
3748 if (!bacmp(&cp->bdaddr, BDADDR_NONE))
3749 return cmd_status(sk, hdev->id,
3750 MGMT_OP_SET_STATIC_ADDRESS,
3751 MGMT_STATUS_INVALID_PARAMS);
3752
3753 /* Two most significant bits shall be set */
3754 if ((cp->bdaddr.b[5] & 0xc0) != 0xc0)
3755 return cmd_status(sk, hdev->id,
3756 MGMT_OP_SET_STATIC_ADDRESS,
3757 MGMT_STATUS_INVALID_PARAMS);
3758 }
3759
3760 hci_dev_lock(hdev);
3761
3762 bacpy(&hdev->static_addr, &cp->bdaddr);
3763
3764 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS, 0, NULL, 0);
3765
3766 hci_dev_unlock(hdev);
3767
3768 return err;
3769}
3770
Marcel Holtmann14b49b92013-10-11 08:23:20 -07003771static int set_scan_params(struct sock *sk, struct hci_dev *hdev,
3772 void *data, u16 len)
3773{
3774 struct mgmt_cp_set_scan_params *cp = data;
3775 __u16 interval, window;
3776 int err;
3777
3778 BT_DBG("%s", hdev->name);
3779
3780 if (!lmp_le_capable(hdev))
3781 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3782 MGMT_STATUS_NOT_SUPPORTED);
3783
3784 interval = __le16_to_cpu(cp->interval);
3785
3786 if (interval < 0x0004 || interval > 0x4000)
3787 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3788 MGMT_STATUS_INVALID_PARAMS);
3789
3790 window = __le16_to_cpu(cp->window);
3791
3792 if (window < 0x0004 || window > 0x4000)
3793 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3794 MGMT_STATUS_INVALID_PARAMS);
3795
Marcel Holtmann899e1072013-10-14 09:55:32 -07003796 if (window > interval)
3797 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3798 MGMT_STATUS_INVALID_PARAMS);
3799
Marcel Holtmann14b49b92013-10-11 08:23:20 -07003800 hci_dev_lock(hdev);
3801
3802 hdev->le_scan_interval = interval;
3803 hdev->le_scan_window = window;
3804
3805 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS, 0, NULL, 0);
3806
3807 hci_dev_unlock(hdev);
3808
3809 return err;
3810}
3811
Johan Hedberg33e38b32013-03-15 17:07:05 -05003812static void fast_connectable_complete(struct hci_dev *hdev, u8 status)
3813{
3814 struct pending_cmd *cmd;
3815
3816 BT_DBG("status 0x%02x", status);
3817
3818 hci_dev_lock(hdev);
3819
3820 cmd = mgmt_pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev);
3821 if (!cmd)
3822 goto unlock;
3823
3824 if (status) {
3825 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3826 mgmt_status(status));
3827 } else {
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05003828 struct mgmt_mode *cp = cmd->param;
3829
3830 if (cp->val)
3831 set_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3832 else
3833 clear_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3834
Johan Hedberg33e38b32013-03-15 17:07:05 -05003835 send_settings_rsp(cmd->sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev);
3836 new_settings(hdev, cmd->sk);
3837 }
3838
3839 mgmt_pending_remove(cmd);
3840
3841unlock:
3842 hci_dev_unlock(hdev);
3843}
3844
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003845static int set_fast_connectable(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003846 void *data, u16 len)
Antti Julkuf6422ec2011-06-22 13:11:56 +03003847{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003848 struct mgmt_mode *cp = data;
Johan Hedberg33e38b32013-03-15 17:07:05 -05003849 struct pending_cmd *cmd;
3850 struct hci_request req;
Antti Julkuf6422ec2011-06-22 13:11:56 +03003851 int err;
3852
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003853 BT_DBG("%s", hdev->name);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003854
Johan Hedberg56f87902013-10-02 13:43:13 +03003855 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags) ||
3856 hdev->hci_ver < BLUETOOTH_VER_1_2)
Johan Hedberg33c525c2012-10-24 21:11:58 +03003857 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3858 MGMT_STATUS_NOT_SUPPORTED);
3859
Johan Hedberga7e80f22013-01-09 16:05:19 +02003860 if (cp->val != 0x00 && cp->val != 0x01)
3861 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3862 MGMT_STATUS_INVALID_PARAMS);
3863
Johan Hedberg5400c042012-02-21 16:40:33 +02003864 if (!hdev_is_powered(hdev))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003865 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003866 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5400c042012-02-21 16:40:33 +02003867
3868 if (!test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003869 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003870 MGMT_STATUS_REJECTED);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003871
3872 hci_dev_lock(hdev);
3873
Johan Hedberg05cbf292013-03-15 17:07:07 -05003874 if (mgmt_pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev)) {
3875 err = cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3876 MGMT_STATUS_BUSY);
3877 goto unlock;
3878 }
3879
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05003880 if (!!cp->val == test_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags)) {
3881 err = send_settings_rsp(sk, MGMT_OP_SET_FAST_CONNECTABLE,
3882 hdev);
3883 goto unlock;
3884 }
3885
Johan Hedberg33e38b32013-03-15 17:07:05 -05003886 cmd = mgmt_pending_add(sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev,
3887 data, len);
3888 if (!cmd) {
3889 err = -ENOMEM;
3890 goto unlock;
3891 }
3892
3893 hci_req_init(&req, hdev);
3894
Johan Hedberg406d7802013-03-15 17:07:09 -05003895 write_fast_connectable(&req, cp->val);
Johan Hedberg33e38b32013-03-15 17:07:05 -05003896
3897 err = hci_req_run(&req, fast_connectable_complete);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003898 if (err < 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003899 err = cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003900 MGMT_STATUS_FAILED);
Johan Hedberg33e38b32013-03-15 17:07:05 -05003901 mgmt_pending_remove(cmd);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003902 }
3903
Johan Hedberg33e38b32013-03-15 17:07:05 -05003904unlock:
Antti Julkuf6422ec2011-06-22 13:11:56 +03003905 hci_dev_unlock(hdev);
Johan Hedberg33e38b32013-03-15 17:07:05 -05003906
Antti Julkuf6422ec2011-06-22 13:11:56 +03003907 return err;
3908}
3909
Johan Hedberg67e5a7a2013-10-14 21:15:25 +03003910static void set_bredr_scan(struct hci_request *req)
3911{
3912 struct hci_dev *hdev = req->hdev;
3913 u8 scan = 0;
3914
3915 /* Ensure that fast connectable is disabled. This function will
3916 * not do anything if the page scan parameters are already what
3917 * they should be.
3918 */
3919 write_fast_connectable(req, false);
3920
3921 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
3922 scan |= SCAN_PAGE;
3923 if (test_bit(HCI_DISCOVERABLE, &hdev->dev_flags))
3924 scan |= SCAN_INQUIRY;
3925
3926 if (scan)
3927 hci_req_add(req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
3928}
3929
Johan Hedberg0663ca22013-10-02 13:43:14 +03003930static void set_bredr_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_BREDR, hdev);
3939 if (!cmd)
3940 goto unlock;
3941
3942 if (status) {
3943 u8 mgmt_err = mgmt_status(status);
3944
3945 /* We need to restore the flag if related HCI commands
3946 * failed.
3947 */
3948 clear_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
3949
3950 cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
3951 } else {
3952 send_settings_rsp(cmd->sk, MGMT_OP_SET_BREDR, hdev);
3953 new_settings(hdev, cmd->sk);
3954 }
3955
3956 mgmt_pending_remove(cmd);
3957
3958unlock:
3959 hci_dev_unlock(hdev);
3960}
3961
3962static int set_bredr(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
3963{
3964 struct mgmt_mode *cp = data;
3965 struct pending_cmd *cmd;
3966 struct hci_request req;
3967 int err;
3968
3969 BT_DBG("request for %s", hdev->name);
3970
3971 if (!lmp_bredr_capable(hdev) || !lmp_le_capable(hdev))
3972 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
3973 MGMT_STATUS_NOT_SUPPORTED);
3974
3975 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
3976 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
3977 MGMT_STATUS_REJECTED);
3978
3979 if (cp->val != 0x00 && cp->val != 0x01)
3980 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
3981 MGMT_STATUS_INVALID_PARAMS);
3982
3983 hci_dev_lock(hdev);
3984
3985 if (cp->val == test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
3986 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
3987 goto unlock;
3988 }
3989
3990 if (!hdev_is_powered(hdev)) {
3991 if (!cp->val) {
Johan Hedberg0663ca22013-10-02 13:43:14 +03003992 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
3993 clear_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
3994 clear_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
3995 clear_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3996 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
3997 }
3998
3999 change_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
4000
4001 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4002 if (err < 0)
4003 goto unlock;
4004
4005 err = new_settings(hdev, sk);
4006 goto unlock;
4007 }
4008
4009 /* Reject disabling when powered on */
4010 if (!cp->val) {
4011 err = cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4012 MGMT_STATUS_REJECTED);
4013 goto unlock;
4014 }
4015
4016 if (mgmt_pending_find(MGMT_OP_SET_BREDR, hdev)) {
4017 err = cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4018 MGMT_STATUS_BUSY);
4019 goto unlock;
4020 }
4021
4022 cmd = mgmt_pending_add(sk, MGMT_OP_SET_BREDR, hdev, data, len);
4023 if (!cmd) {
4024 err = -ENOMEM;
4025 goto unlock;
4026 }
4027
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004028 /* We need to flip the bit already here so that update_adv_data
Johan Hedberg0663ca22013-10-02 13:43:14 +03004029 * generates the correct flags.
4030 */
4031 set_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
4032
4033 hci_req_init(&req, hdev);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004034
4035 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
4036 set_bredr_scan(&req);
4037
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004038 /* Since only the advertising data flags will change, there
4039 * is no need to update the scan response data.
4040 */
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004041 update_adv_data(&req);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004042
Johan Hedberg0663ca22013-10-02 13:43:14 +03004043 err = hci_req_run(&req, set_bredr_complete);
4044 if (err < 0)
4045 mgmt_pending_remove(cmd);
4046
4047unlock:
4048 hci_dev_unlock(hdev);
4049 return err;
4050}
4051
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004052static int set_secure_conn(struct sock *sk, struct hci_dev *hdev,
4053 void *data, u16 len)
4054{
4055 struct mgmt_mode *cp = data;
4056 struct pending_cmd *cmd;
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004057 u8 val, status;
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004058 int err;
4059
4060 BT_DBG("request for %s", hdev->name);
4061
4062 status = mgmt_bredr_support(hdev);
4063 if (status)
4064 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4065 status);
4066
Marcel Holtmann5afeac142014-01-10 02:07:27 -08004067 if (!lmp_sc_capable(hdev) &&
4068 !test_bit(HCI_FORCE_SC, &hdev->dev_flags))
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004069 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4070 MGMT_STATUS_NOT_SUPPORTED);
4071
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004072 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004073 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4074 MGMT_STATUS_INVALID_PARAMS);
4075
4076 hci_dev_lock(hdev);
4077
4078 if (!hdev_is_powered(hdev)) {
4079 bool changed;
4080
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004081 if (cp->val) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004082 changed = !test_and_set_bit(HCI_SC_ENABLED,
4083 &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004084 if (cp->val == 0x02)
4085 set_bit(HCI_SC_ONLY, &hdev->dev_flags);
4086 else
4087 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4088 } else {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004089 changed = test_and_clear_bit(HCI_SC_ENABLED,
4090 &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004091 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4092 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004093
4094 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4095 if (err < 0)
4096 goto failed;
4097
4098 if (changed)
4099 err = new_settings(hdev, sk);
4100
4101 goto failed;
4102 }
4103
4104 if (mgmt_pending_find(MGMT_OP_SET_SECURE_CONN, hdev)) {
4105 err = cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4106 MGMT_STATUS_BUSY);
4107 goto failed;
4108 }
4109
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004110 val = !!cp->val;
4111
4112 if (val == test_bit(HCI_SC_ENABLED, &hdev->dev_flags) &&
4113 (cp->val == 0x02) == test_bit(HCI_SC_ONLY, &hdev->dev_flags)) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004114 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4115 goto failed;
4116 }
4117
4118 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SECURE_CONN, hdev, data, len);
4119 if (!cmd) {
4120 err = -ENOMEM;
4121 goto failed;
4122 }
4123
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004124 err = hci_send_cmd(hdev, HCI_OP_WRITE_SC_SUPPORT, 1, &val);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004125 if (err < 0) {
4126 mgmt_pending_remove(cmd);
4127 goto failed;
4128 }
4129
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004130 if (cp->val == 0x02)
4131 set_bit(HCI_SC_ONLY, &hdev->dev_flags);
4132 else
4133 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4134
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004135failed:
4136 hci_dev_unlock(hdev);
4137 return err;
4138}
4139
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004140static int set_debug_keys(struct sock *sk, struct hci_dev *hdev,
4141 void *data, u16 len)
4142{
4143 struct mgmt_mode *cp = data;
4144 bool changed;
4145 int err;
4146
4147 BT_DBG("request for %s", hdev->name);
4148
4149 if (cp->val != 0x00 && cp->val != 0x01)
4150 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEBUG_KEYS,
4151 MGMT_STATUS_INVALID_PARAMS);
4152
4153 hci_dev_lock(hdev);
4154
4155 if (cp->val)
4156 changed = !test_and_set_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
4157 else
4158 changed = test_and_clear_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
4159
4160 err = send_settings_rsp(sk, MGMT_OP_SET_DEBUG_KEYS, hdev);
4161 if (err < 0)
4162 goto unlock;
4163
4164 if (changed)
4165 err = new_settings(hdev, sk);
4166
4167unlock:
4168 hci_dev_unlock(hdev);
4169 return err;
4170}
4171
Johan Hedberg41edf162014-02-18 10:19:35 +02004172static bool irk_is_valid(struct mgmt_irk_info *irk)
4173{
4174 switch (irk->addr.type) {
4175 case BDADDR_LE_PUBLIC:
4176 return true;
4177
4178 case BDADDR_LE_RANDOM:
4179 /* Two most significant bits shall be set */
4180 if ((irk->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4181 return false;
4182 return true;
4183 }
4184
4185 return false;
4186}
4187
4188static int load_irks(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4189 u16 len)
4190{
4191 struct mgmt_cp_load_irks *cp = cp_data;
4192 u16 irk_count, expected_len;
4193 int i, err;
4194
4195 BT_DBG("request for %s", hdev->name);
4196
4197 if (!lmp_le_capable(hdev))
4198 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4199 MGMT_STATUS_NOT_SUPPORTED);
4200
4201 irk_count = __le16_to_cpu(cp->irk_count);
4202
4203 expected_len = sizeof(*cp) + irk_count * sizeof(struct mgmt_irk_info);
4204 if (expected_len != len) {
4205 BT_ERR("load_irks: expected %u bytes, got %u bytes",
4206 len, expected_len);
4207 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4208 MGMT_STATUS_INVALID_PARAMS);
4209 }
4210
4211 BT_DBG("%s irk_count %u", hdev->name, irk_count);
4212
4213 for (i = 0; i < irk_count; i++) {
4214 struct mgmt_irk_info *key = &cp->irks[i];
4215
4216 if (!irk_is_valid(key))
4217 return cmd_status(sk, hdev->id,
4218 MGMT_OP_LOAD_IRKS,
4219 MGMT_STATUS_INVALID_PARAMS);
4220 }
4221
4222 hci_dev_lock(hdev);
4223
4224 hci_smp_irks_clear(hdev);
4225
4226 for (i = 0; i < irk_count; i++) {
4227 struct mgmt_irk_info *irk = &cp->irks[i];
4228 u8 addr_type;
4229
4230 if (irk->addr.type == BDADDR_LE_PUBLIC)
4231 addr_type = ADDR_LE_DEV_PUBLIC;
4232 else
4233 addr_type = ADDR_LE_DEV_RANDOM;
4234
4235 hci_add_irk(hdev, &irk->addr.bdaddr, addr_type, irk->val,
4236 BDADDR_ANY);
4237 }
4238
4239 set_bit(HCI_RPA_RESOLVING, &hdev->dev_flags);
4240
4241 err = cmd_complete(sk, hdev->id, MGMT_OP_LOAD_IRKS, 0, NULL, 0);
4242
4243 hci_dev_unlock(hdev);
4244
4245 return err;
4246}
4247
Johan Hedberg3f706b72013-01-20 14:27:16 +02004248static bool ltk_is_valid(struct mgmt_ltk_info *key)
4249{
4250 if (key->master != 0x00 && key->master != 0x01)
4251 return false;
Marcel Holtmann490cb0b2014-02-16 12:59:05 -08004252
4253 switch (key->addr.type) {
4254 case BDADDR_LE_PUBLIC:
4255 return true;
4256
4257 case BDADDR_LE_RANDOM:
4258 /* Two most significant bits shall be set */
4259 if ((key->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4260 return false;
4261 return true;
4262 }
4263
4264 return false;
Johan Hedberg3f706b72013-01-20 14:27:16 +02004265}
4266
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004267static int load_long_term_keys(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004268 void *cp_data, u16 len)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004269{
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004270 struct mgmt_cp_load_long_term_keys *cp = cp_data;
4271 u16 key_count, expected_len;
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004272 int i, err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004273
Marcel Holtmanncf99ba12013-10-02 21:16:08 -07004274 BT_DBG("request for %s", hdev->name);
4275
4276 if (!lmp_le_capable(hdev))
4277 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4278 MGMT_STATUS_NOT_SUPPORTED);
4279
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004280 key_count = __le16_to_cpu(cp->key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004281
4282 expected_len = sizeof(*cp) + key_count *
4283 sizeof(struct mgmt_ltk_info);
4284 if (expected_len != len) {
4285 BT_ERR("load_keys: expected %u bytes, got %u bytes",
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004286 len, expected_len);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004287 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
Johan Hedberge57e6192013-01-20 14:27:14 +02004288 MGMT_STATUS_INVALID_PARAMS);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004289 }
4290
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004291 BT_DBG("%s key_count %u", hdev->name, key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004292
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004293 for (i = 0; i < key_count; i++) {
4294 struct mgmt_ltk_info *key = &cp->keys[i];
4295
Johan Hedberg3f706b72013-01-20 14:27:16 +02004296 if (!ltk_is_valid(key))
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004297 return cmd_status(sk, hdev->id,
4298 MGMT_OP_LOAD_LONG_TERM_KEYS,
4299 MGMT_STATUS_INVALID_PARAMS);
4300 }
4301
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004302 hci_dev_lock(hdev);
4303
4304 hci_smp_ltks_clear(hdev);
4305
4306 for (i = 0; i < key_count; i++) {
4307 struct mgmt_ltk_info *key = &cp->keys[i];
Marcel Holtmann79d95a12013-10-13 03:57:38 -07004308 u8 type, addr_type;
4309
4310 if (key->addr.type == BDADDR_LE_PUBLIC)
4311 addr_type = ADDR_LE_DEV_PUBLIC;
4312 else
4313 addr_type = ADDR_LE_DEV_RANDOM;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004314
4315 if (key->master)
4316 type = HCI_SMP_LTK;
4317 else
4318 type = HCI_SMP_LTK_SLAVE;
4319
Marcel Holtmann79d95a12013-10-13 03:57:38 -07004320 hci_add_ltk(hdev, &key->addr.bdaddr, addr_type,
Marcel Holtmannd40f3ee2014-01-31 18:42:17 -08004321 type, 0, key->type, key->val,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004322 key->enc_size, key->ediv, key->rand);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004323 }
4324
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004325 err = cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS, 0,
4326 NULL, 0);
4327
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004328 hci_dev_unlock(hdev);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004329
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004330 return err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004331}
4332
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +02004333static const struct mgmt_handler {
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004334 int (*func) (struct sock *sk, struct hci_dev *hdev, void *data,
4335 u16 data_len);
Johan Hedbergbe22b542012-03-01 22:24:41 +02004336 bool var_len;
4337 size_t data_len;
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004338} mgmt_handlers[] = {
4339 { NULL }, /* 0x0000 (no command) */
Johan Hedbergbe22b542012-03-01 22:24:41 +02004340 { read_version, false, MGMT_READ_VERSION_SIZE },
4341 { read_commands, false, MGMT_READ_COMMANDS_SIZE },
4342 { read_index_list, false, MGMT_READ_INDEX_LIST_SIZE },
4343 { read_controller_info, false, MGMT_READ_INFO_SIZE },
4344 { set_powered, false, MGMT_SETTING_SIZE },
4345 { set_discoverable, false, MGMT_SET_DISCOVERABLE_SIZE },
4346 { set_connectable, false, MGMT_SETTING_SIZE },
4347 { set_fast_connectable, false, MGMT_SETTING_SIZE },
4348 { set_pairable, false, MGMT_SETTING_SIZE },
4349 { set_link_security, false, MGMT_SETTING_SIZE },
4350 { set_ssp, false, MGMT_SETTING_SIZE },
4351 { set_hs, false, MGMT_SETTING_SIZE },
4352 { set_le, false, MGMT_SETTING_SIZE },
4353 { set_dev_class, false, MGMT_SET_DEV_CLASS_SIZE },
4354 { set_local_name, false, MGMT_SET_LOCAL_NAME_SIZE },
4355 { add_uuid, false, MGMT_ADD_UUID_SIZE },
4356 { remove_uuid, false, MGMT_REMOVE_UUID_SIZE },
4357 { load_link_keys, true, MGMT_LOAD_LINK_KEYS_SIZE },
4358 { load_long_term_keys, true, MGMT_LOAD_LONG_TERM_KEYS_SIZE },
4359 { disconnect, false, MGMT_DISCONNECT_SIZE },
4360 { get_connections, false, MGMT_GET_CONNECTIONS_SIZE },
4361 { pin_code_reply, false, MGMT_PIN_CODE_REPLY_SIZE },
4362 { pin_code_neg_reply, false, MGMT_PIN_CODE_NEG_REPLY_SIZE },
4363 { set_io_capability, false, MGMT_SET_IO_CAPABILITY_SIZE },
4364 { pair_device, false, MGMT_PAIR_DEVICE_SIZE },
4365 { cancel_pair_device, false, MGMT_CANCEL_PAIR_DEVICE_SIZE },
4366 { unpair_device, false, MGMT_UNPAIR_DEVICE_SIZE },
4367 { user_confirm_reply, false, MGMT_USER_CONFIRM_REPLY_SIZE },
4368 { user_confirm_neg_reply, false, MGMT_USER_CONFIRM_NEG_REPLY_SIZE },
4369 { user_passkey_reply, false, MGMT_USER_PASSKEY_REPLY_SIZE },
4370 { user_passkey_neg_reply, false, MGMT_USER_PASSKEY_NEG_REPLY_SIZE },
4371 { read_local_oob_data, false, MGMT_READ_LOCAL_OOB_DATA_SIZE },
Marcel Holtmannec109112014-01-10 02:07:30 -08004372 { add_remote_oob_data, true, MGMT_ADD_REMOTE_OOB_DATA_SIZE },
Johan Hedbergbe22b542012-03-01 22:24:41 +02004373 { remove_remote_oob_data, false, MGMT_REMOVE_REMOTE_OOB_DATA_SIZE },
4374 { start_discovery, false, MGMT_START_DISCOVERY_SIZE },
4375 { stop_discovery, false, MGMT_STOP_DISCOVERY_SIZE },
4376 { confirm_name, false, MGMT_CONFIRM_NAME_SIZE },
4377 { block_device, false, MGMT_BLOCK_DEVICE_SIZE },
4378 { unblock_device, false, MGMT_UNBLOCK_DEVICE_SIZE },
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07004379 { set_device_id, false, MGMT_SET_DEVICE_ID_SIZE },
Johan Hedberg4375f102013-09-25 13:26:10 +03004380 { set_advertising, false, MGMT_SETTING_SIZE },
Johan Hedberg0663ca22013-10-02 13:43:14 +03004381 { set_bredr, false, MGMT_SETTING_SIZE },
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004382 { set_static_address, false, MGMT_SET_STATIC_ADDRESS_SIZE },
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004383 { set_scan_params, false, MGMT_SET_SCAN_PARAMS_SIZE },
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004384 { set_secure_conn, false, MGMT_SETTING_SIZE },
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004385 { set_debug_keys, false, MGMT_SETTING_SIZE },
Johan Hedberg41edf162014-02-18 10:19:35 +02004386 { },
4387 { load_irks, true, MGMT_LOAD_IRKS_SIZE },
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004388};
4389
4390
Johan Hedberg03811012010-12-08 00:21:06 +02004391int mgmt_control(struct sock *sk, struct msghdr *msg, size_t msglen)
4392{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004393 void *buf;
4394 u8 *cp;
Johan Hedberg03811012010-12-08 00:21:06 +02004395 struct mgmt_hdr *hdr;
Szymon Janc4e51eae2011-02-25 19:05:48 +01004396 u16 opcode, index, len;
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004397 struct hci_dev *hdev = NULL;
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +02004398 const struct mgmt_handler *handler;
Johan Hedberg03811012010-12-08 00:21:06 +02004399 int err;
4400
4401 BT_DBG("got %zu bytes", msglen);
4402
4403 if (msglen < sizeof(*hdr))
4404 return -EINVAL;
4405
Gustavo F. Padovane63a15e2011-04-04 18:56:53 -03004406 buf = kmalloc(msglen, GFP_KERNEL);
Johan Hedberg03811012010-12-08 00:21:06 +02004407 if (!buf)
4408 return -ENOMEM;
4409
4410 if (memcpy_fromiovec(buf, msg->msg_iov, msglen)) {
4411 err = -EFAULT;
4412 goto done;
4413 }
4414
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004415 hdr = buf;
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004416 opcode = __le16_to_cpu(hdr->opcode);
4417 index = __le16_to_cpu(hdr->index);
4418 len = __le16_to_cpu(hdr->len);
Johan Hedberg03811012010-12-08 00:21:06 +02004419
4420 if (len != msglen - sizeof(*hdr)) {
4421 err = -EINVAL;
4422 goto done;
4423 }
4424
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004425 if (index != MGMT_INDEX_NONE) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004426 hdev = hci_dev_get(index);
4427 if (!hdev) {
4428 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004429 MGMT_STATUS_INVALID_INDEX);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004430 goto done;
4431 }
Marcel Holtmann0736cfa2013-08-26 21:40:51 -07004432
Johan Hedbergcebf4cf2013-10-10 18:06:04 +02004433 if (test_bit(HCI_SETUP, &hdev->dev_flags) ||
4434 test_bit(HCI_USER_CHANNEL, &hdev->dev_flags)) {
Marcel Holtmann0736cfa2013-08-26 21:40:51 -07004435 err = cmd_status(sk, index, opcode,
4436 MGMT_STATUS_INVALID_INDEX);
4437 goto done;
4438 }
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004439 }
4440
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004441 if (opcode >= ARRAY_SIZE(mgmt_handlers) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004442 mgmt_handlers[opcode].func == NULL) {
Johan Hedberg03811012010-12-08 00:21:06 +02004443 BT_DBG("Unknown op %u", opcode);
Johan Hedbergca69b792011-11-11 18:10:00 +02004444 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004445 MGMT_STATUS_UNKNOWN_COMMAND);
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004446 goto done;
Johan Hedberg03811012010-12-08 00:21:06 +02004447 }
4448
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004449 if ((hdev && opcode < MGMT_OP_READ_INFO) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004450 (!hdev && opcode >= MGMT_OP_READ_INFO)) {
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004451 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004452 MGMT_STATUS_INVALID_INDEX);
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004453 goto done;
4454 }
4455
Johan Hedbergbe22b542012-03-01 22:24:41 +02004456 handler = &mgmt_handlers[opcode];
4457
4458 if ((handler->var_len && len < handler->data_len) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004459 (!handler->var_len && len != handler->data_len)) {
Johan Hedbergbe22b542012-03-01 22:24:41 +02004460 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004461 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergbe22b542012-03-01 22:24:41 +02004462 goto done;
4463 }
4464
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004465 if (hdev)
4466 mgmt_init_hdev(sk, hdev);
4467
4468 cp = buf + sizeof(*hdr);
4469
Johan Hedbergbe22b542012-03-01 22:24:41 +02004470 err = handler->func(sk, hdev, cp, len);
Johan Hedberge41d8b42010-12-13 21:07:03 +02004471 if (err < 0)
4472 goto done;
4473
Johan Hedberg03811012010-12-08 00:21:06 +02004474 err = msglen;
4475
4476done:
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004477 if (hdev)
4478 hci_dev_put(hdev);
4479
Johan Hedberg03811012010-12-08 00:21:06 +02004480 kfree(buf);
4481 return err;
4482}
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004483
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004484void mgmt_index_added(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004485{
Marcel Holtmann1514b892013-10-06 08:25:01 -07004486 if (hdev->dev_type != HCI_BREDR)
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004487 return;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03004488
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004489 mgmt_event(MGMT_EV_INDEX_ADDED, hdev, NULL, 0, NULL);
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004490}
4491
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004492void mgmt_index_removed(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004493{
Johan Hedberg5f159032012-03-02 03:13:19 +02004494 u8 status = MGMT_STATUS_INVALID_INDEX;
Johan Hedbergb24752f2011-11-03 14:40:33 +02004495
Marcel Holtmann1514b892013-10-06 08:25:01 -07004496 if (hdev->dev_type != HCI_BREDR)
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004497 return;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03004498
Johan Hedberg744cf192011-11-08 20:40:14 +02004499 mgmt_pending_foreach(0, hdev, cmd_status_rsp, &status);
Johan Hedbergb24752f2011-11-03 14:40:33 +02004500
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004501 mgmt_event(MGMT_EV_INDEX_REMOVED, hdev, NULL, 0, NULL);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02004502}
4503
Johan Hedberg229ab392013-03-15 17:06:53 -05004504static void powered_complete(struct hci_dev *hdev, u8 status)
4505{
4506 struct cmd_lookup match = { NULL, hdev };
4507
4508 BT_DBG("status 0x%02x", status);
4509
4510 hci_dev_lock(hdev);
4511
4512 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
4513
4514 new_settings(hdev, match.sk);
4515
4516 hci_dev_unlock(hdev);
4517
4518 if (match.sk)
4519 sock_put(match.sk);
4520}
4521
Johan Hedberg70da6242013-03-15 17:06:51 -05004522static int powered_update_hci(struct hci_dev *hdev)
4523{
Johan Hedberg890ea892013-03-15 17:06:52 -05004524 struct hci_request req;
Johan Hedberg70da6242013-03-15 17:06:51 -05004525 u8 link_sec;
4526
Johan Hedberg890ea892013-03-15 17:06:52 -05004527 hci_req_init(&req, hdev);
4528
Johan Hedberg70da6242013-03-15 17:06:51 -05004529 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags) &&
4530 !lmp_host_ssp_capable(hdev)) {
4531 u8 ssp = 1;
4532
Johan Hedberg890ea892013-03-15 17:06:52 -05004533 hci_req_add(&req, HCI_OP_WRITE_SSP_MODE, 1, &ssp);
Johan Hedberg70da6242013-03-15 17:06:51 -05004534 }
4535
Johan Hedbergc73eee92013-04-19 18:35:21 +03004536 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags) &&
4537 lmp_bredr_capable(hdev)) {
Johan Hedberg70da6242013-03-15 17:06:51 -05004538 struct hci_cp_write_le_host_supported cp;
4539
4540 cp.le = 1;
4541 cp.simul = lmp_le_br_capable(hdev);
4542
4543 /* Check first if we already have the right
4544 * host state (host features set)
4545 */
4546 if (cp.le != lmp_host_le_capable(hdev) ||
4547 cp.simul != lmp_host_le_br_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05004548 hci_req_add(&req, HCI_OP_WRITE_LE_HOST_SUPPORTED,
4549 sizeof(cp), &cp);
Johan Hedberg70da6242013-03-15 17:06:51 -05004550 }
4551
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004552 if (lmp_le_capable(hdev)) {
4553 /* Set random address to static address if configured */
4554 if (bacmp(&hdev->static_addr, BDADDR_ANY))
4555 hci_req_add(&req, HCI_OP_LE_SET_RANDOM_ADDR, 6,
4556 &hdev->static_addr);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004557
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07004558 /* Make sure the controller has a good default for
4559 * advertising data. This also applies to the case
4560 * where BR/EDR was toggled during the AUTO_OFF phase.
4561 */
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004562 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004563 update_adv_data(&req);
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004564 update_scan_rsp_data(&req);
4565 }
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07004566
Marcel Holtmannbba3aa52013-10-06 02:55:21 -07004567 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
4568 enable_advertising(&req);
Johan Hedbergeeca6f82013-09-25 13:26:09 +03004569 }
4570
Johan Hedberg70da6242013-03-15 17:06:51 -05004571 link_sec = test_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
4572 if (link_sec != test_bit(HCI_AUTH, &hdev->flags))
Johan Hedberg890ea892013-03-15 17:06:52 -05004573 hci_req_add(&req, HCI_OP_WRITE_AUTH_ENABLE,
4574 sizeof(link_sec), &link_sec);
Johan Hedberg70da6242013-03-15 17:06:51 -05004575
4576 if (lmp_bredr_capable(hdev)) {
Johan Hedberg56f87902013-10-02 13:43:13 +03004577 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
4578 set_bredr_scan(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05004579 update_class(&req);
Johan Hedberg13928972013-03-15 17:07:00 -05004580 update_name(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05004581 update_eir(&req);
Johan Hedberg70da6242013-03-15 17:06:51 -05004582 }
4583
Johan Hedberg229ab392013-03-15 17:06:53 -05004584 return hci_req_run(&req, powered_complete);
Johan Hedberg70da6242013-03-15 17:06:51 -05004585}
4586
Johan Hedberg744cf192011-11-08 20:40:14 +02004587int mgmt_powered(struct hci_dev *hdev, u8 powered)
Johan Hedberg5add6af2010-12-16 10:00:37 +02004588{
Johan Hedberg76a7f3a2012-02-17 00:34:40 +02004589 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg229ab392013-03-15 17:06:53 -05004590 u8 status_not_powered = MGMT_STATUS_NOT_POWERED;
4591 u8 zero_cod[] = { 0, 0, 0 };
Johan Hedberg7bb895d2012-02-17 01:20:00 +02004592 int err;
Johan Hedberg5add6af2010-12-16 10:00:37 +02004593
Johan Hedberg5e5282b2012-02-21 16:01:30 +02004594 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
4595 return 0;
4596
Johan Hedberg5e5282b2012-02-21 16:01:30 +02004597 if (powered) {
Johan Hedberg229ab392013-03-15 17:06:53 -05004598 if (powered_update_hci(hdev) == 0)
4599 return 0;
Johan Hedbergfe038882013-01-16 16:15:34 +02004600
Johan Hedberg229ab392013-03-15 17:06:53 -05004601 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp,
4602 &match);
4603 goto new_settings;
Johan Hedbergb24752f2011-11-03 14:40:33 +02004604 }
4605
Johan Hedberg229ab392013-03-15 17:06:53 -05004606 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
4607 mgmt_pending_foreach(0, hdev, cmd_status_rsp, &status_not_powered);
4608
4609 if (memcmp(hdev->dev_class, zero_cod, sizeof(zero_cod)) != 0)
4610 mgmt_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev,
4611 zero_cod, sizeof(zero_cod), NULL);
4612
4613new_settings:
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02004614 err = new_settings(hdev, match.sk);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02004615
4616 if (match.sk)
4617 sock_put(match.sk);
4618
Johan Hedberg7bb895d2012-02-17 01:20:00 +02004619 return err;
Johan Hedberg5add6af2010-12-16 10:00:37 +02004620}
Johan Hedberg73f22f62010-12-29 16:00:25 +02004621
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004622void mgmt_set_powered_failed(struct hci_dev *hdev, int err)
Johan Hedberg96570ff2013-05-29 09:51:29 +03004623{
4624 struct pending_cmd *cmd;
4625 u8 status;
4626
4627 cmd = mgmt_pending_find(MGMT_OP_SET_POWERED, hdev);
4628 if (!cmd)
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004629 return;
Johan Hedberg96570ff2013-05-29 09:51:29 +03004630
4631 if (err == -ERFKILL)
4632 status = MGMT_STATUS_RFKILLED;
4633 else
4634 status = MGMT_STATUS_FAILED;
4635
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004636 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_POWERED, status);
Johan Hedberg96570ff2013-05-29 09:51:29 +03004637
4638 mgmt_pending_remove(cmd);
Johan Hedberg96570ff2013-05-29 09:51:29 +03004639}
4640
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004641void mgmt_discoverable_timeout(struct hci_dev *hdev)
4642{
4643 struct hci_request req;
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004644
4645 hci_dev_lock(hdev);
4646
4647 /* When discoverable timeout triggers, then just make sure
4648 * the limited discoverable flag is cleared. Even in the case
4649 * of a timeout triggered from general discoverable, it is
4650 * safe to unconditionally clear the flag.
4651 */
4652 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004653 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004654
4655 hci_req_init(&req, hdev);
Johan Hedberg4b580612013-10-19 23:38:21 +03004656 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
4657 u8 scan = SCAN_PAGE;
4658 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE,
4659 sizeof(scan), &scan);
4660 }
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004661 update_class(&req);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004662 update_adv_data(&req);
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004663 hci_req_run(&req, NULL);
4664
4665 hdev->discov_timeout = 0;
4666
Johan Hedberg9a43e252013-10-20 19:00:07 +03004667 new_settings(hdev, NULL);
4668
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004669 hci_dev_unlock(hdev);
4670}
4671
Marcel Holtmann86a75642013-10-15 06:33:54 -07004672void mgmt_discoverable(struct hci_dev *hdev, u8 discoverable)
Johan Hedberg73f22f62010-12-29 16:00:25 +02004673{
Marcel Holtmann86a75642013-10-15 06:33:54 -07004674 bool changed;
Johan Hedberg73f22f62010-12-29 16:00:25 +02004675
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03004676 /* Nothing needed here if there's a pending command since that
4677 * commands request completion callback takes care of everything
4678 * necessary.
4679 */
4680 if (mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev))
Marcel Holtmann86a75642013-10-15 06:33:54 -07004681 return;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03004682
Johan Hedberg9a43e252013-10-20 19:00:07 +03004683 if (discoverable) {
Marcel Holtmann86a75642013-10-15 06:33:54 -07004684 changed = !test_and_set_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004685 } else {
4686 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Marcel Holtmann86a75642013-10-15 06:33:54 -07004687 changed = test_and_clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004688 }
Johan Hedberg73f22f62010-12-29 16:00:25 +02004689
Johan Hedberg9a43e252013-10-20 19:00:07 +03004690 if (changed) {
4691 struct hci_request req;
4692
4693 /* In case this change in discoverable was triggered by
4694 * a disabling of connectable there could be a need to
4695 * update the advertising flags.
4696 */
4697 hci_req_init(&req, hdev);
4698 update_adv_data(&req);
4699 hci_req_run(&req, NULL);
4700
Marcel Holtmann86a75642013-10-15 06:33:54 -07004701 new_settings(hdev, NULL);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004702 }
Johan Hedberg73f22f62010-12-29 16:00:25 +02004703}
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004704
Marcel Holtmanna3309162013-10-15 06:33:55 -07004705void mgmt_connectable(struct hci_dev *hdev, u8 connectable)
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004706{
Marcel Holtmanna3309162013-10-15 06:33:55 -07004707 bool changed;
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004708
Johan Hedbergd7b856f2013-10-14 16:20:04 +03004709 /* Nothing needed here if there's a pending command since that
4710 * commands request completion callback takes care of everything
4711 * necessary.
4712 */
4713 if (mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev))
Marcel Holtmanna3309162013-10-15 06:33:55 -07004714 return;
Johan Hedbergd7b856f2013-10-14 16:20:04 +03004715
Marcel Holtmanna3309162013-10-15 06:33:55 -07004716 if (connectable)
4717 changed = !test_and_set_bit(HCI_CONNECTABLE, &hdev->dev_flags);
4718 else
4719 changed = test_and_clear_bit(HCI_CONNECTABLE, &hdev->dev_flags);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004720
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02004721 if (changed)
Marcel Holtmanna3309162013-10-15 06:33:55 -07004722 new_settings(hdev, NULL);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004723}
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004724
Marcel Holtmann4796e8a2013-10-15 06:33:56 -07004725void mgmt_write_scan_failed(struct hci_dev *hdev, u8 scan, u8 status)
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004726{
Johan Hedbergca69b792011-11-11 18:10:00 +02004727 u8 mgmt_err = mgmt_status(status);
4728
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004729 if (scan & SCAN_PAGE)
Johan Hedberg744cf192011-11-08 20:40:14 +02004730 mgmt_pending_foreach(MGMT_OP_SET_CONNECTABLE, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004731 cmd_status_rsp, &mgmt_err);
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004732
4733 if (scan & SCAN_INQUIRY)
Johan Hedberg744cf192011-11-08 20:40:14 +02004734 mgmt_pending_foreach(MGMT_OP_SET_DISCOVERABLE, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004735 cmd_status_rsp, &mgmt_err);
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004736}
4737
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07004738void mgmt_new_link_key(struct hci_dev *hdev, struct link_key *key,
4739 bool persistent)
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004740{
Johan Hedberg86742e12011-11-07 23:13:38 +02004741 struct mgmt_ev_new_link_key ev;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004742
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004743 memset(&ev, 0, sizeof(ev));
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004744
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004745 ev.store_hint = persistent;
Johan Hedbergd753fdc2012-02-17 14:06:34 +02004746 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004747 ev.key.addr.type = BDADDR_BREDR;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004748 ev.key.type = key->type;
Andrei Emeltchenko9b3b4462012-05-23 11:31:20 +03004749 memcpy(ev.key.val, key->val, HCI_LINK_KEY_SIZE);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004750 ev.key.pin_len = key->pin_len;
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004751
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07004752 mgmt_event(MGMT_EV_NEW_LINK_KEY, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg55ed8ca2011-01-17 14:41:05 +02004753}
Johan Hedbergf7520542011-01-20 12:34:39 +02004754
Marcel Holtmann083368f2013-10-15 14:26:29 -07004755void mgmt_new_ltk(struct hci_dev *hdev, struct smp_ltk *key, u8 persistent)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004756{
4757 struct mgmt_ev_new_long_term_key ev;
4758
4759 memset(&ev, 0, sizeof(ev));
4760
4761 ev.store_hint = persistent;
4762 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004763 ev.key.addr.type = link_to_bdaddr(LE_LINK, key->bdaddr_type);
Marcel Holtmannd40f3ee2014-01-31 18:42:17 -08004764 ev.key.type = key->authenticated;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004765 ev.key.enc_size = key->enc_size;
4766 ev.key.ediv = key->ediv;
4767
4768 if (key->type == HCI_SMP_LTK)
4769 ev.key.master = 1;
4770
4771 memcpy(ev.key.rand, key->rand, sizeof(key->rand));
4772 memcpy(ev.key.val, key->val, sizeof(key->val));
4773
Marcel Holtmann083368f2013-10-15 14:26:29 -07004774 mgmt_event(MGMT_EV_NEW_LONG_TERM_KEY, hdev, &ev, sizeof(ev), NULL);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004775}
4776
Marcel Holtmann94933992013-10-15 10:26:39 -07004777static inline u16 eir_append_data(u8 *eir, u16 eir_len, u8 type, u8 *data,
4778 u8 data_len)
4779{
4780 eir[eir_len++] = sizeof(type) + data_len;
4781 eir[eir_len++] = type;
4782 memcpy(&eir[eir_len], data, data_len);
4783 eir_len += data_len;
4784
4785 return eir_len;
4786}
4787
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07004788void mgmt_device_connected(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
4789 u8 addr_type, u32 flags, u8 *name, u8 name_len,
4790 u8 *dev_class)
Johan Hedbergf7520542011-01-20 12:34:39 +02004791{
Johan Hedbergb644ba32012-01-17 21:48:47 +02004792 char buf[512];
4793 struct mgmt_ev_device_connected *ev = (void *) buf;
4794 u16 eir_len = 0;
Johan Hedbergf7520542011-01-20 12:34:39 +02004795
Johan Hedbergb644ba32012-01-17 21:48:47 +02004796 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004797 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergf7520542011-01-20 12:34:39 +02004798
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02004799 ev->flags = __cpu_to_le32(flags);
Johan Hedberg08c79b62012-02-23 22:31:51 +02004800
Johan Hedbergb644ba32012-01-17 21:48:47 +02004801 if (name_len > 0)
4802 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004803 name, name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02004804
4805 if (dev_class && memcmp(dev_class, "\0\0\0", 3) != 0)
Brian Gix53156382012-03-09 14:07:03 -08004806 eir_len = eir_append_data(ev->eir, eir_len,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004807 EIR_CLASS_OF_DEV, dev_class, 3);
Johan Hedbergb644ba32012-01-17 21:48:47 +02004808
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02004809 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02004810
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07004811 mgmt_event(MGMT_EV_DEVICE_CONNECTED, hdev, buf,
4812 sizeof(*ev) + eir_len, NULL);
Johan Hedbergf7520542011-01-20 12:34:39 +02004813}
4814
Johan Hedberg8962ee72011-01-20 12:40:27 +02004815static void disconnect_rsp(struct pending_cmd *cmd, void *data)
4816{
Szymon Jancc68fb7f2011-03-22 13:12:19 +01004817 struct mgmt_cp_disconnect *cp = cmd->param;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004818 struct sock **sk = data;
Johan Hedberga38528f2011-01-22 06:46:43 +02004819 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004820
Johan Hedberg88c3df12012-02-09 14:27:38 +02004821 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
4822 rp.addr.type = cp->addr.type;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004823
Johan Hedbergaee9b2182012-02-18 15:07:59 +02004824 cmd_complete(cmd->sk, cmd->index, MGMT_OP_DISCONNECT, 0, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004825 sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02004826
4827 *sk = cmd->sk;
4828 sock_hold(*sk);
4829
Johan Hedberga664b5b2011-02-19 12:06:02 -03004830 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004831}
4832
Johan Hedberg124f6e32012-02-09 13:50:12 +02004833static void unpair_device_rsp(struct pending_cmd *cmd, void *data)
Johan Hedberga8a1d192011-11-10 15:54:38 +02004834{
Johan Hedbergb1078ad2012-02-09 17:21:16 +02004835 struct hci_dev *hdev = data;
Johan Hedberg124f6e32012-02-09 13:50:12 +02004836 struct mgmt_cp_unpair_device *cp = cmd->param;
4837 struct mgmt_rp_unpair_device rp;
Johan Hedberga8a1d192011-11-10 15:54:38 +02004838
4839 memset(&rp, 0, sizeof(rp));
Johan Hedberg124f6e32012-02-09 13:50:12 +02004840 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
4841 rp.addr.type = cp->addr.type;
Johan Hedberga8a1d192011-11-10 15:54:38 +02004842
Johan Hedbergb1078ad2012-02-09 17:21:16 +02004843 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, cmd->sk);
4844
Johan Hedbergaee9b2182012-02-18 15:07:59 +02004845 cmd_complete(cmd->sk, cmd->index, cmd->opcode, 0, &rp, sizeof(rp));
Johan Hedberga8a1d192011-11-10 15:54:38 +02004846
4847 mgmt_pending_remove(cmd);
4848}
4849
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07004850void mgmt_device_disconnected(struct hci_dev *hdev, bdaddr_t *bdaddr,
4851 u8 link_type, u8 addr_type, u8 reason)
Johan Hedbergf7520542011-01-20 12:34:39 +02004852{
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02004853 struct mgmt_ev_device_disconnected ev;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004854 struct sock *sk = NULL;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004855
Andre Guedes57eb7762013-10-30 19:01:41 -03004856 if (link_type != ACL_LINK && link_type != LE_LINK)
4857 return;
4858
Johan Hedberg744cf192011-11-08 20:40:14 +02004859 mgmt_pending_foreach(MGMT_OP_DISCONNECT, hdev, disconnect_rsp, &sk);
Johan Hedbergf7520542011-01-20 12:34:39 +02004860
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02004861 bacpy(&ev.addr.bdaddr, bdaddr);
4862 ev.addr.type = link_to_bdaddr(link_type, addr_type);
4863 ev.reason = reason;
Johan Hedbergf7520542011-01-20 12:34:39 +02004864
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07004865 mgmt_event(MGMT_EV_DEVICE_DISCONNECTED, hdev, &ev, sizeof(ev), sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004866
4867 if (sk)
Szymon Jancd97dcb62012-03-16 16:02:56 +01004868 sock_put(sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004869
Johan Hedberg124f6e32012-02-09 13:50:12 +02004870 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004871 hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004872}
4873
Marcel Holtmann78929242013-10-06 23:55:47 -07004874void mgmt_disconnect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr,
4875 u8 link_type, u8 addr_type, u8 status)
Johan Hedberg8962ee72011-01-20 12:40:27 +02004876{
Andre Guedes3655bba2013-10-30 19:01:40 -03004877 u8 bdaddr_type = link_to_bdaddr(link_type, addr_type);
4878 struct mgmt_cp_disconnect *cp;
Johan Hedberg88c3df12012-02-09 14:27:38 +02004879 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004880 struct pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004881
Jefferson Delfes36a75f12012-09-18 13:36:54 -04004882 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
4883 hdev);
4884
Johan Hedberg2e58ef32011-11-08 20:40:15 +02004885 cmd = mgmt_pending_find(MGMT_OP_DISCONNECT, hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004886 if (!cmd)
Marcel Holtmann78929242013-10-06 23:55:47 -07004887 return;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004888
Andre Guedes3655bba2013-10-30 19:01:40 -03004889 cp = cmd->param;
4890
4891 if (bacmp(bdaddr, &cp->addr.bdaddr))
4892 return;
4893
4894 if (cp->addr.type != bdaddr_type)
4895 return;
4896
Johan Hedberg88c3df12012-02-09 14:27:38 +02004897 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes3655bba2013-10-30 19:01:40 -03004898 rp.addr.type = bdaddr_type;
Johan Hedberg37d9ef72011-11-10 15:54:39 +02004899
Marcel Holtmann78929242013-10-06 23:55:47 -07004900 cmd_complete(cmd->sk, cmd->index, MGMT_OP_DISCONNECT,
4901 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02004902
Johan Hedberga664b5b2011-02-19 12:06:02 -03004903 mgmt_pending_remove(cmd);
Johan Hedbergf7520542011-01-20 12:34:39 +02004904}
Johan Hedberg17d5c042011-01-22 06:09:08 +02004905
Marcel Holtmann445608d2013-10-06 23:55:48 -07004906void mgmt_connect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
4907 u8 addr_type, u8 status)
Johan Hedberg17d5c042011-01-22 06:09:08 +02004908{
4909 struct mgmt_ev_connect_failed ev;
4910
Johan Hedberg4c659c32011-11-07 23:13:39 +02004911 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004912 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02004913 ev.status = mgmt_status(status);
Johan Hedberg17d5c042011-01-22 06:09:08 +02004914
Marcel Holtmann445608d2013-10-06 23:55:48 -07004915 mgmt_event(MGMT_EV_CONNECT_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg17d5c042011-01-22 06:09:08 +02004916}
Johan Hedberg980e1a52011-01-22 06:10:07 +02004917
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07004918void mgmt_pin_code_request(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 secure)
Johan Hedberg980e1a52011-01-22 06:10:07 +02004919{
4920 struct mgmt_ev_pin_code_request ev;
4921
Johan Hedbergd8457692012-02-17 14:24:57 +02004922 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004923 ev.addr.type = BDADDR_BREDR;
Waldemar Rymarkiewicza770bb52011-04-28 12:07:59 +02004924 ev.secure = secure;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004925
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07004926 mgmt_event(MGMT_EV_PIN_CODE_REQUEST, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004927}
4928
Marcel Holtmanne669cf82013-10-15 14:26:21 -07004929void mgmt_pin_code_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
4930 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02004931{
4932 struct pending_cmd *cmd;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004933 struct mgmt_rp_pin_code_reply rp;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004934
Johan Hedberg2e58ef32011-11-08 20:40:15 +02004935 cmd = mgmt_pending_find(MGMT_OP_PIN_CODE_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004936 if (!cmd)
Marcel Holtmanne669cf82013-10-15 14:26:21 -07004937 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004938
Johan Hedbergd8457692012-02-17 14:24:57 +02004939 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004940 rp.addr.type = BDADDR_BREDR;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004941
Marcel Holtmanne669cf82013-10-15 14:26:21 -07004942 cmd_complete(cmd->sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
4943 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg980e1a52011-01-22 06:10:07 +02004944
Johan Hedberga664b5b2011-02-19 12:06:02 -03004945 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004946}
4947
Marcel Holtmann3eb38522013-10-15 14:26:22 -07004948void mgmt_pin_code_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
4949 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02004950{
4951 struct pending_cmd *cmd;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004952 struct mgmt_rp_pin_code_reply rp;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004953
Johan Hedberg2e58ef32011-11-08 20:40:15 +02004954 cmd = mgmt_pending_find(MGMT_OP_PIN_CODE_NEG_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004955 if (!cmd)
Marcel Holtmann3eb38522013-10-15 14:26:22 -07004956 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004957
Johan Hedbergd8457692012-02-17 14:24:57 +02004958 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004959 rp.addr.type = BDADDR_BREDR;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004960
Marcel Holtmann3eb38522013-10-15 14:26:22 -07004961 cmd_complete(cmd->sk, hdev->id, MGMT_OP_PIN_CODE_NEG_REPLY,
4962 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg980e1a52011-01-22 06:10:07 +02004963
Johan Hedberga664b5b2011-02-19 12:06:02 -03004964 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004965}
Johan Hedberga5c29682011-02-19 12:05:57 -03004966
Johan Hedberg744cf192011-11-08 20:40:14 +02004967int mgmt_user_confirm_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004968 u8 link_type, u8 addr_type, __le32 value,
4969 u8 confirm_hint)
Johan Hedberga5c29682011-02-19 12:05:57 -03004970{
4971 struct mgmt_ev_user_confirm_request ev;
4972
Johan Hedberg744cf192011-11-08 20:40:14 +02004973 BT_DBG("%s", hdev->name);
Johan Hedberga5c29682011-02-19 12:05:57 -03004974
Johan Hedberg272d90d2012-02-09 15:26:12 +02004975 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004976 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberg55bc1a32011-04-28 11:28:56 -07004977 ev.confirm_hint = confirm_hint;
Andrei Emeltchenko78e8098e2012-03-09 13:00:50 +02004978 ev.value = value;
Johan Hedberga5c29682011-02-19 12:05:57 -03004979
Johan Hedberg744cf192011-11-08 20:40:14 +02004980 return mgmt_event(MGMT_EV_USER_CONFIRM_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004981 NULL);
Johan Hedberga5c29682011-02-19 12:05:57 -03004982}
4983
Johan Hedberg272d90d2012-02-09 15:26:12 +02004984int mgmt_user_passkey_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004985 u8 link_type, u8 addr_type)
Brian Gix604086b2011-11-23 08:28:33 -08004986{
4987 struct mgmt_ev_user_passkey_request ev;
4988
4989 BT_DBG("%s", hdev->name);
4990
Johan Hedberg272d90d2012-02-09 15:26:12 +02004991 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004992 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Brian Gix604086b2011-11-23 08:28:33 -08004993
4994 return mgmt_event(MGMT_EV_USER_PASSKEY_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004995 NULL);
Brian Gix604086b2011-11-23 08:28:33 -08004996}
4997
Brian Gix0df4c182011-11-16 13:53:13 -08004998static int user_pairing_resp_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004999 u8 link_type, u8 addr_type, u8 status,
5000 u8 opcode)
Johan Hedberga5c29682011-02-19 12:05:57 -03005001{
5002 struct pending_cmd *cmd;
5003 struct mgmt_rp_user_confirm_reply rp;
5004 int err;
5005
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005006 cmd = mgmt_pending_find(opcode, hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03005007 if (!cmd)
5008 return -ENOENT;
5009
Johan Hedberg272d90d2012-02-09 15:26:12 +02005010 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005011 rp.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergaee9b2182012-02-18 15:07:59 +02005012 err = cmd_complete(cmd->sk, hdev->id, opcode, mgmt_status(status),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005013 &rp, sizeof(rp));
Johan Hedberga5c29682011-02-19 12:05:57 -03005014
Johan Hedberga664b5b2011-02-19 12:06:02 -03005015 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03005016
5017 return err;
5018}
5019
Johan Hedberg744cf192011-11-08 20:40:14 +02005020int mgmt_user_confirm_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005021 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03005022{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005023 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005024 status, MGMT_OP_USER_CONFIRM_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03005025}
5026
Johan Hedberg272d90d2012-02-09 15:26:12 +02005027int mgmt_user_confirm_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005028 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03005029{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005030 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03005031 status,
5032 MGMT_OP_USER_CONFIRM_NEG_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03005033}
Johan Hedberg2a611692011-02-19 12:06:00 -03005034
Brian Gix604086b2011-11-23 08:28:33 -08005035int mgmt_user_passkey_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005036 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08005037{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005038 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005039 status, MGMT_OP_USER_PASSKEY_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08005040}
5041
Johan Hedberg272d90d2012-02-09 15:26:12 +02005042int mgmt_user_passkey_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005043 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08005044{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005045 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03005046 status,
5047 MGMT_OP_USER_PASSKEY_NEG_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08005048}
5049
Johan Hedberg92a25252012-09-06 18:39:26 +03005050int mgmt_user_passkey_notify(struct hci_dev *hdev, bdaddr_t *bdaddr,
5051 u8 link_type, u8 addr_type, u32 passkey,
5052 u8 entered)
5053{
5054 struct mgmt_ev_passkey_notify ev;
5055
5056 BT_DBG("%s", hdev->name);
5057
5058 bacpy(&ev.addr.bdaddr, bdaddr);
5059 ev.addr.type = link_to_bdaddr(link_type, addr_type);
5060 ev.passkey = __cpu_to_le32(passkey);
5061 ev.entered = entered;
5062
5063 return mgmt_event(MGMT_EV_PASSKEY_NOTIFY, hdev, &ev, sizeof(ev), NULL);
5064}
5065
Marcel Holtmanne5460992013-10-15 14:26:23 -07005066void mgmt_auth_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5067 u8 addr_type, u8 status)
Johan Hedberg2a611692011-02-19 12:06:00 -03005068{
5069 struct mgmt_ev_auth_failed ev;
5070
Johan Hedbergbab73cb2012-02-09 16:07:29 +02005071 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005072 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02005073 ev.status = mgmt_status(status);
Johan Hedberg2a611692011-02-19 12:06:00 -03005074
Marcel Holtmanne5460992013-10-15 14:26:23 -07005075 mgmt_event(MGMT_EV_AUTH_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg2a611692011-02-19 12:06:00 -03005076}
Johan Hedbergb312b1612011-03-16 14:29:37 +02005077
Marcel Holtmann464996a2013-10-15 14:26:24 -07005078void mgmt_auth_enable_complete(struct hci_dev *hdev, u8 status)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005079{
5080 struct cmd_lookup match = { NULL, hdev };
Marcel Holtmann464996a2013-10-15 14:26:24 -07005081 bool changed;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005082
5083 if (status) {
5084 u8 mgmt_err = mgmt_status(status);
5085 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005086 cmd_status_rsp, &mgmt_err);
Marcel Holtmann464996a2013-10-15 14:26:24 -07005087 return;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005088 }
5089
Marcel Holtmann464996a2013-10-15 14:26:24 -07005090 if (test_bit(HCI_AUTH, &hdev->flags))
5091 changed = !test_and_set_bit(HCI_LINK_SECURITY,
5092 &hdev->dev_flags);
5093 else
5094 changed = test_and_clear_bit(HCI_LINK_SECURITY,
5095 &hdev->dev_flags);
Johan Hedberg47990ea2012-02-22 11:58:37 +02005096
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005097 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev, settings_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005098 &match);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005099
Johan Hedberg47990ea2012-02-22 11:58:37 +02005100 if (changed)
Marcel Holtmann464996a2013-10-15 14:26:24 -07005101 new_settings(hdev, match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005102
5103 if (match.sk)
5104 sock_put(match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005105}
5106
Johan Hedberg890ea892013-03-15 17:06:52 -05005107static void clear_eir(struct hci_request *req)
Johan Hedbergcacaf522012-02-21 00:52:42 +02005108{
Johan Hedberg890ea892013-03-15 17:06:52 -05005109 struct hci_dev *hdev = req->hdev;
Johan Hedbergcacaf522012-02-21 00:52:42 +02005110 struct hci_cp_write_eir cp;
5111
Johan Hedberg976eb202012-10-24 21:12:01 +03005112 if (!lmp_ext_inq_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05005113 return;
Johan Hedbergcacaf522012-02-21 00:52:42 +02005114
Johan Hedbergc80da272012-02-22 15:38:48 +02005115 memset(hdev->eir, 0, sizeof(hdev->eir));
5116
Johan Hedbergcacaf522012-02-21 00:52:42 +02005117 memset(&cp, 0, sizeof(cp));
5118
Johan Hedberg890ea892013-03-15 17:06:52 -05005119 hci_req_add(req, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
Johan Hedbergcacaf522012-02-21 00:52:42 +02005120}
5121
Marcel Holtmann3e248562013-10-15 14:26:25 -07005122void mgmt_ssp_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005123{
5124 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg890ea892013-03-15 17:06:52 -05005125 struct hci_request req;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005126 bool changed = false;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005127
5128 if (status) {
5129 u8 mgmt_err = mgmt_status(status);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005130
5131 if (enable && test_and_clear_bit(HCI_SSP_ENABLED,
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005132 &hdev->dev_flags)) {
5133 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Marcel Holtmann3e248562013-10-15 14:26:25 -07005134 new_settings(hdev, NULL);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005135 }
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005136
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005137 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, cmd_status_rsp,
5138 &mgmt_err);
Marcel Holtmann3e248562013-10-15 14:26:25 -07005139 return;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005140 }
5141
5142 if (enable) {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005143 changed = !test_and_set_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005144 } else {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005145 changed = test_and_clear_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
5146 if (!changed)
5147 changed = test_and_clear_bit(HCI_HS_ENABLED,
5148 &hdev->dev_flags);
5149 else
5150 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005151 }
5152
5153 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, settings_rsp, &match);
5154
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005155 if (changed)
Marcel Holtmann3e248562013-10-15 14:26:25 -07005156 new_settings(hdev, match.sk);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005157
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005158 if (match.sk)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005159 sock_put(match.sk);
5160
Johan Hedberg890ea892013-03-15 17:06:52 -05005161 hci_req_init(&req, hdev);
5162
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005163 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -05005164 update_eir(&req);
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005165 else
Johan Hedberg890ea892013-03-15 17:06:52 -05005166 clear_eir(&req);
5167
5168 hci_req_run(&req, NULL);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005169}
5170
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005171void mgmt_sc_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
5172{
5173 struct cmd_lookup match = { NULL, hdev };
5174 bool changed = false;
5175
5176 if (status) {
5177 u8 mgmt_err = mgmt_status(status);
5178
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005179 if (enable) {
5180 if (test_and_clear_bit(HCI_SC_ENABLED,
5181 &hdev->dev_flags))
5182 new_settings(hdev, NULL);
5183 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
5184 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005185
5186 mgmt_pending_foreach(MGMT_OP_SET_SECURE_CONN, hdev,
5187 cmd_status_rsp, &mgmt_err);
5188 return;
5189 }
5190
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005191 if (enable) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005192 changed = !test_and_set_bit(HCI_SC_ENABLED, &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005193 } else {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005194 changed = test_and_clear_bit(HCI_SC_ENABLED, &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005195 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
5196 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005197
5198 mgmt_pending_foreach(MGMT_OP_SET_SECURE_CONN, hdev,
5199 settings_rsp, &match);
5200
5201 if (changed)
5202 new_settings(hdev, match.sk);
5203
5204 if (match.sk)
5205 sock_put(match.sk);
5206}
5207
Johan Hedberg92da6092013-03-15 17:06:55 -05005208static void sk_lookup(struct pending_cmd *cmd, void *data)
Johan Hedberg90e70452012-02-23 23:09:40 +02005209{
5210 struct cmd_lookup *match = data;
5211
Johan Hedberg90e70452012-02-23 23:09:40 +02005212 if (match->sk == NULL) {
5213 match->sk = cmd->sk;
5214 sock_hold(match->sk);
5215 }
Johan Hedberg90e70452012-02-23 23:09:40 +02005216}
5217
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07005218void mgmt_set_class_of_dev_complete(struct hci_dev *hdev, u8 *dev_class,
5219 u8 status)
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005220{
Johan Hedberg90e70452012-02-23 23:09:40 +02005221 struct cmd_lookup match = { NULL, hdev, mgmt_status(status) };
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005222
Johan Hedberg92da6092013-03-15 17:06:55 -05005223 mgmt_pending_foreach(MGMT_OP_SET_DEV_CLASS, hdev, sk_lookup, &match);
5224 mgmt_pending_foreach(MGMT_OP_ADD_UUID, hdev, sk_lookup, &match);
5225 mgmt_pending_foreach(MGMT_OP_REMOVE_UUID, hdev, sk_lookup, &match);
Johan Hedberg90e70452012-02-23 23:09:40 +02005226
5227 if (!status)
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07005228 mgmt_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev, dev_class, 3,
5229 NULL);
Johan Hedberg90e70452012-02-23 23:09:40 +02005230
5231 if (match.sk)
5232 sock_put(match.sk);
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005233}
5234
Marcel Holtmann7667da32013-10-15 14:26:27 -07005235void mgmt_set_local_name_complete(struct hci_dev *hdev, u8 *name, u8 status)
Johan Hedbergb312b1612011-03-16 14:29:37 +02005236{
Johan Hedbergb312b1612011-03-16 14:29:37 +02005237 struct mgmt_cp_set_local_name ev;
Johan Hedberg13928972013-03-15 17:07:00 -05005238 struct pending_cmd *cmd;
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02005239
Johan Hedberg13928972013-03-15 17:07:00 -05005240 if (status)
Marcel Holtmann7667da32013-10-15 14:26:27 -07005241 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02005242
5243 memset(&ev, 0, sizeof(ev));
5244 memcpy(ev.name, name, HCI_MAX_NAME_LENGTH);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02005245 memcpy(ev.short_name, hdev->short_name, HCI_MAX_SHORT_NAME_LENGTH);
Johan Hedbergb312b1612011-03-16 14:29:37 +02005246
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005247 cmd = mgmt_pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
Johan Hedberg13928972013-03-15 17:07:00 -05005248 if (!cmd) {
5249 memcpy(hdev->dev_name, name, sizeof(hdev->dev_name));
Johan Hedbergb312b1612011-03-16 14:29:37 +02005250
Johan Hedberg13928972013-03-15 17:07:00 -05005251 /* If this is a HCI command related to powering on the
5252 * HCI dev don't send any mgmt signals.
5253 */
5254 if (mgmt_pending_find(MGMT_OP_SET_POWERED, hdev))
Marcel Holtmann7667da32013-10-15 14:26:27 -07005255 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02005256 }
5257
Marcel Holtmann7667da32013-10-15 14:26:27 -07005258 mgmt_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, &ev, sizeof(ev),
5259 cmd ? cmd->sk : NULL);
Johan Hedbergb312b1612011-03-16 14:29:37 +02005260}
Szymon Jancc35938b2011-03-22 13:12:21 +01005261
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005262void mgmt_read_local_oob_data_complete(struct hci_dev *hdev, u8 *hash192,
5263 u8 *randomizer192, u8 *hash256,
5264 u8 *randomizer256, u8 status)
Szymon Jancc35938b2011-03-22 13:12:21 +01005265{
5266 struct pending_cmd *cmd;
Szymon Jancc35938b2011-03-22 13:12:21 +01005267
Johan Hedberg744cf192011-11-08 20:40:14 +02005268 BT_DBG("%s status %u", hdev->name, status);
Szymon Jancc35938b2011-03-22 13:12:21 +01005269
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005270 cmd = mgmt_pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01005271 if (!cmd)
Marcel Holtmann3edaf092013-10-15 14:26:28 -07005272 return;
Szymon Jancc35938b2011-03-22 13:12:21 +01005273
5274 if (status) {
Marcel Holtmann3edaf092013-10-15 14:26:28 -07005275 cmd_status(cmd->sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
5276 mgmt_status(status));
Szymon Jancc35938b2011-03-22 13:12:21 +01005277 } else {
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005278 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags) &&
5279 hash256 && randomizer256) {
5280 struct mgmt_rp_read_local_oob_ext_data rp;
Szymon Jancc35938b2011-03-22 13:12:21 +01005281
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005282 memcpy(rp.hash192, hash192, sizeof(rp.hash192));
5283 memcpy(rp.randomizer192, randomizer192,
5284 sizeof(rp.randomizer192));
Szymon Jancc35938b2011-03-22 13:12:21 +01005285
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005286 memcpy(rp.hash256, hash256, sizeof(rp.hash256));
5287 memcpy(rp.randomizer256, randomizer256,
5288 sizeof(rp.randomizer256));
5289
5290 cmd_complete(cmd->sk, hdev->id,
5291 MGMT_OP_READ_LOCAL_OOB_DATA, 0,
5292 &rp, sizeof(rp));
5293 } else {
5294 struct mgmt_rp_read_local_oob_data rp;
5295
5296 memcpy(rp.hash, hash192, sizeof(rp.hash));
5297 memcpy(rp.randomizer, randomizer192,
5298 sizeof(rp.randomizer));
5299
5300 cmd_complete(cmd->sk, hdev->id,
5301 MGMT_OP_READ_LOCAL_OOB_DATA, 0,
5302 &rp, sizeof(rp));
5303 }
Szymon Jancc35938b2011-03-22 13:12:21 +01005304 }
5305
5306 mgmt_pending_remove(cmd);
Szymon Jancc35938b2011-03-22 13:12:21 +01005307}
Johan Hedberge17acd42011-03-30 23:57:16 +03005308
Marcel Holtmann901801b2013-10-06 23:55:51 -07005309void mgmt_device_found(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5310 u8 addr_type, u8 *dev_class, s8 rssi, u8 cfm_name, u8
5311 ssp, u8 *eir, u16 eir_len)
Johan Hedberge17acd42011-03-30 23:57:16 +03005312{
Johan Hedberge319d2e2012-01-15 19:51:59 +02005313 char buf[512];
5314 struct mgmt_ev_device_found *ev = (void *) buf;
Johan Hedberg1dc06092012-01-15 21:01:23 +02005315 size_t ev_size;
Johan Hedberge17acd42011-03-30 23:57:16 +03005316
Andre Guedes12602d02013-04-30 15:29:40 -03005317 if (!hci_discovery_active(hdev))
Marcel Holtmann901801b2013-10-06 23:55:51 -07005318 return;
Andre Guedes12602d02013-04-30 15:29:40 -03005319
Johan Hedberg1dc06092012-01-15 21:01:23 +02005320 /* Leave 5 bytes for a potential CoD field */
5321 if (sizeof(*ev) + eir_len + 5 > sizeof(buf))
Marcel Holtmann901801b2013-10-06 23:55:51 -07005322 return;
Andre Guedes7d262f82012-01-10 18:20:49 -03005323
Johan Hedberg1dc06092012-01-15 21:01:23 +02005324 memset(buf, 0, sizeof(buf));
5325
Johan Hedberge319d2e2012-01-15 19:51:59 +02005326 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005327 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberge319d2e2012-01-15 19:51:59 +02005328 ev->rssi = rssi;
Johan Hedberg9a395a82012-02-23 00:00:32 +02005329 if (cfm_name)
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05305330 ev->flags |= __constant_cpu_to_le32(MGMT_DEV_FOUND_CONFIRM_NAME);
Johan Hedberg388fc8f2012-02-23 00:38:59 +02005331 if (!ssp)
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05305332 ev->flags |= __constant_cpu_to_le32(MGMT_DEV_FOUND_LEGACY_PAIRING);
Johan Hedberge17acd42011-03-30 23:57:16 +03005333
Johan Hedberg1dc06092012-01-15 21:01:23 +02005334 if (eir_len > 0)
Johan Hedberge319d2e2012-01-15 19:51:59 +02005335 memcpy(ev->eir, eir, eir_len);
Johan Hedberge17acd42011-03-30 23:57:16 +03005336
Johan Hedberg1dc06092012-01-15 21:01:23 +02005337 if (dev_class && !eir_has_data_type(ev->eir, eir_len, EIR_CLASS_OF_DEV))
5338 eir_len = eir_append_data(ev->eir, eir_len, EIR_CLASS_OF_DEV,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005339 dev_class, 3);
Johan Hedberg1dc06092012-01-15 21:01:23 +02005340
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02005341 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedberg1dc06092012-01-15 21:01:23 +02005342 ev_size = sizeof(*ev) + eir_len;
Andre Guedesf8523592011-09-09 18:56:26 -03005343
Marcel Holtmann901801b2013-10-06 23:55:51 -07005344 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, ev_size, NULL);
Johan Hedberge17acd42011-03-30 23:57:16 +03005345}
Johan Hedberga88a9652011-03-30 13:18:12 +03005346
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07005347void mgmt_remote_name(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5348 u8 addr_type, s8 rssi, u8 *name, u8 name_len)
Johan Hedberga88a9652011-03-30 13:18:12 +03005349{
Johan Hedbergb644ba32012-01-17 21:48:47 +02005350 struct mgmt_ev_device_found *ev;
5351 char buf[sizeof(*ev) + HCI_MAX_NAME_LENGTH + 2];
5352 u16 eir_len;
Johan Hedberga88a9652011-03-30 13:18:12 +03005353
Johan Hedbergb644ba32012-01-17 21:48:47 +02005354 ev = (struct mgmt_ev_device_found *) buf;
Johan Hedberga88a9652011-03-30 13:18:12 +03005355
Johan Hedbergb644ba32012-01-17 21:48:47 +02005356 memset(buf, 0, sizeof(buf));
Johan Hedberga88a9652011-03-30 13:18:12 +03005357
Johan Hedbergb644ba32012-01-17 21:48:47 +02005358 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005359 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005360 ev->rssi = rssi;
5361
5362 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE, name,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005363 name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005364
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02005365 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005366
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07005367 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, sizeof(*ev) + eir_len, NULL);
Johan Hedberga88a9652011-03-30 13:18:12 +03005368}
Johan Hedberg314b2382011-04-27 10:29:57 -04005369
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07005370void mgmt_discovering(struct hci_dev *hdev, u8 discovering)
Johan Hedberg314b2382011-04-27 10:29:57 -04005371{
Johan Hedbergf963e8e2012-02-20 23:30:44 +02005372 struct mgmt_ev_discovering ev;
Johan Hedberg164a6e72011-11-01 17:06:44 +02005373 struct pending_cmd *cmd;
5374
Andre Guedes343fb142011-11-22 17:14:19 -03005375 BT_DBG("%s discovering %u", hdev->name, discovering);
5376
Johan Hedberg164a6e72011-11-01 17:06:44 +02005377 if (discovering)
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005378 cmd = mgmt_pending_find(MGMT_OP_START_DISCOVERY, hdev);
Johan Hedberg164a6e72011-11-01 17:06:44 +02005379 else
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005380 cmd = mgmt_pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
Johan Hedberg164a6e72011-11-01 17:06:44 +02005381
5382 if (cmd != NULL) {
Johan Hedbergf808e162012-02-19 12:52:07 +02005383 u8 type = hdev->discovery.type;
5384
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005385 cmd_complete(cmd->sk, hdev->id, cmd->opcode, 0, &type,
5386 sizeof(type));
Johan Hedberg164a6e72011-11-01 17:06:44 +02005387 mgmt_pending_remove(cmd);
5388 }
5389
Johan Hedbergf963e8e2012-02-20 23:30:44 +02005390 memset(&ev, 0, sizeof(ev));
5391 ev.type = hdev->discovery.type;
5392 ev.discovering = discovering;
5393
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07005394 mgmt_event(MGMT_EV_DISCOVERING, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg314b2382011-04-27 10:29:57 -04005395}
Antti Julku5e762442011-08-25 16:48:02 +03005396
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005397int mgmt_device_blocked(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 type)
Antti Julku5e762442011-08-25 16:48:02 +03005398{
5399 struct pending_cmd *cmd;
5400 struct mgmt_ev_device_blocked ev;
5401
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005402 cmd = mgmt_pending_find(MGMT_OP_BLOCK_DEVICE, hdev);
Antti Julku5e762442011-08-25 16:48:02 +03005403
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005404 bacpy(&ev.addr.bdaddr, bdaddr);
5405 ev.addr.type = type;
Antti Julku5e762442011-08-25 16:48:02 +03005406
Johan Hedberg744cf192011-11-08 20:40:14 +02005407 return mgmt_event(MGMT_EV_DEVICE_BLOCKED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005408 cmd ? cmd->sk : NULL);
Antti Julku5e762442011-08-25 16:48:02 +03005409}
5410
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005411int mgmt_device_unblocked(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 type)
Antti Julku5e762442011-08-25 16:48:02 +03005412{
5413 struct pending_cmd *cmd;
5414 struct mgmt_ev_device_unblocked ev;
5415
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005416 cmd = mgmt_pending_find(MGMT_OP_UNBLOCK_DEVICE, hdev);
Antti Julku5e762442011-08-25 16:48:02 +03005417
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005418 bacpy(&ev.addr.bdaddr, bdaddr);
5419 ev.addr.type = type;
Antti Julku5e762442011-08-25 16:48:02 +03005420
Johan Hedberg744cf192011-11-08 20:40:14 +02005421 return mgmt_event(MGMT_EV_DEVICE_UNBLOCKED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005422 cmd ? cmd->sk : NULL);
Antti Julku5e762442011-08-25 16:48:02 +03005423}
Marcel Holtmann5976e602013-10-06 04:08:14 -07005424
5425static void adv_enable_complete(struct hci_dev *hdev, u8 status)
5426{
5427 BT_DBG("%s status %u", hdev->name, status);
5428
5429 /* Clear the advertising mgmt setting if we failed to re-enable it */
5430 if (status) {
5431 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Marcel Holtmanna6d811e2013-10-06 04:11:12 -07005432 new_settings(hdev, NULL);
Marcel Holtmann5976e602013-10-06 04:08:14 -07005433 }
5434}
5435
5436void mgmt_reenable_advertising(struct hci_dev *hdev)
5437{
5438 struct hci_request req;
5439
Marcel Holtmannb145edc2013-10-10 09:47:54 -07005440 if (hci_conn_num(hdev, LE_LINK) > 0)
Marcel Holtmann5976e602013-10-06 04:08:14 -07005441 return;
5442
5443 if (!test_bit(HCI_ADVERTISING, &hdev->dev_flags))
5444 return;
5445
5446 hci_req_init(&req, hdev);
5447 enable_advertising(&req);
5448
5449 /* If this fails we have no option but to let user space know
5450 * that we've disabled advertising.
5451 */
5452 if (hci_req_run(&req, adv_enable_complete) < 0) {
5453 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Marcel Holtmanna6d811e2013-10-06 04:11:12 -07005454 new_settings(hdev, NULL);
Marcel Holtmann5976e602013-10-06 04:08:14 -07005455 }
5456}