blob: fbb76a0de580a5fd0b7ae8904db76a02cd6345a7 [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 Holtmann40456642014-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-12-13 21:07:06 +0200206
Andre Guedes790eff42012-06-07 19:05:46 -0300207 skb = alloc_skb(sizeof(*hdr) + sizeof(*ev), GFP_KERNEL);
Johan Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-12-13 21:07:06 +0200223 kfree_skb(skb);
224
Gustavo F. Padovan56b7d132011-10-14 19:20:01 -0300225 return err;
Johan Hedbergf7b64e692010-12-13 21:07:06 +0200226}
227
Johan Hedbergaee9b212012-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 Hedbergaee9b212012-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 Hedbergaee9b212012-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 Hedbergaee9b212012-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 Hedbergaee9b212012-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 Holtmann5afeac12014-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergaee9b212012-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 Hedbergf7b64e692010-12-13 21:07:06 +02001177
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001178 hci_dev_lock(hdev);
Johan Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-12-13 21:07:06 +02001304 if (err < 0)
1305 mgmt_pending_remove(cmd);
1306
Johan Hedbergf7b64e692010-12-13 21:07:06 +02001307failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001308 hci_dev_unlock(hdev);
Johan Hedbergf7b64e692010-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 Hedberge8ba3a1f2013-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 Hedbergf7b64e692010-12-13 21:07:06 +02001457{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001458 struct mgmt_mode *cp = data;
Johan Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedberge8ba3a1f2013-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-01-17 14:41:05 +02002234 }
2235
Johan Hedberg4ae143012013-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-01-17 14:41:05 +02002252
2253 hci_link_keys_clear(hdev);
2254
Johan Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-01-17 14:41:05 +02002273
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002274 return 0;
Johan Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-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 Hedberg55ed8ca12011-01-17 14:41:05 +02002296 struct hci_conn *conn;
Johan Hedberg55ed8ca12011-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
2331 err = hci_remove_ltk(hdev, &cp->addr.bdaddr, addr_type);
2332 }
Vinicius Costa Gomesb0dbfb42012-02-02 21:08:03 -03002333
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002334 if (err < 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002335 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002336 MGMT_STATUS_NOT_PAIRED, &rp, sizeof(rp));
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002337 goto unlock;
2338 }
2339
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002340 if (cp->disconnect) {
Andre Guedes591f47f2012-04-24 21:02:49 -03002341 if (cp->addr.type == BDADDR_BREDR)
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002342 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002343 &cp->addr.bdaddr);
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002344 else
2345 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002346 &cp->addr.bdaddr);
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002347 } else {
2348 conn = NULL;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002349 }
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002350
Johan Hedberga8a1d192011-11-10 15:54:38 +02002351 if (!conn) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002352 err = cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002353 &rp, sizeof(rp));
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002354 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, sk);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002355 goto unlock;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002356 }
2357
Johan Hedberg124f6e32012-02-09 13:50:12 +02002358 cmd = mgmt_pending_add(sk, MGMT_OP_UNPAIR_DEVICE, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002359 sizeof(*cp));
Johan Hedberga8a1d192011-11-10 15:54:38 +02002360 if (!cmd) {
2361 err = -ENOMEM;
2362 goto unlock;
2363 }
2364
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002365 dc.handle = cpu_to_le16(conn->handle);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002366 dc.reason = 0x13; /* Remote User Terminated Connection */
2367 err = hci_send_cmd(hdev, HCI_OP_DISCONNECT, sizeof(dc), &dc);
2368 if (err < 0)
2369 mgmt_pending_remove(cmd);
2370
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002371unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002372 hci_dev_unlock(hdev);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002373 return err;
2374}
2375
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002376static int disconnect(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002377 u16 len)
Johan Hedberg8962ee72011-01-20 12:40:27 +02002378{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002379 struct mgmt_cp_disconnect *cp = data;
Johan Hedberg06a63b12013-01-20 14:27:21 +02002380 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002381 struct hci_cp_disconnect dc;
Johan Hedberg366a0332011-02-19 12:05:55 -03002382 struct pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002383 struct hci_conn *conn;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002384 int err;
2385
2386 BT_DBG("");
2387
Johan Hedberg06a63b12013-01-20 14:27:21 +02002388 memset(&rp, 0, sizeof(rp));
2389 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2390 rp.addr.type = cp->addr.type;
2391
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002392 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg06a63b12013-01-20 14:27:21 +02002393 return cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2394 MGMT_STATUS_INVALID_PARAMS,
2395 &rp, sizeof(rp));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002396
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002397 hci_dev_lock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002398
2399 if (!test_bit(HCI_UP, &hdev->flags)) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002400 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2401 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002402 goto failed;
2403 }
2404
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002405 if (mgmt_pending_find(MGMT_OP_DISCONNECT, hdev)) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002406 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2407 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002408 goto failed;
2409 }
2410
Andre Guedes591f47f2012-04-24 21:02:49 -03002411 if (cp->addr.type == BDADDR_BREDR)
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03002412 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
2413 &cp->addr.bdaddr);
Johan Hedberg88c3df12012-02-09 14:27:38 +02002414 else
2415 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &cp->addr.bdaddr);
Vinicius Costa Gomes365227e2011-05-06 18:41:44 -03002416
Vishal Agarwalf9607272012-06-13 05:32:43 +05302417 if (!conn || conn->state == BT_OPEN || conn->state == BT_CLOSED) {
Johan Hedberg06a63b12013-01-20 14:27:21 +02002418 err = cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2419 MGMT_STATUS_NOT_CONNECTED, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002420 goto failed;
2421 }
2422
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002423 cmd = mgmt_pending_add(sk, MGMT_OP_DISCONNECT, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002424 if (!cmd) {
2425 err = -ENOMEM;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002426 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002427 }
Johan Hedberg8962ee72011-01-20 12:40:27 +02002428
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002429 dc.handle = cpu_to_le16(conn->handle);
Andre Guedes3701f942012-06-11 18:41:12 -03002430 dc.reason = HCI_ERROR_REMOTE_USER_TERM;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002431
2432 err = hci_send_cmd(hdev, HCI_OP_DISCONNECT, sizeof(dc), &dc);
2433 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002434 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002435
2436failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002437 hci_dev_unlock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002438 return err;
2439}
2440
Andre Guedes57c14772012-04-24 21:02:50 -03002441static u8 link_to_bdaddr(u8 link_type, u8 addr_type)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002442{
2443 switch (link_type) {
2444 case LE_LINK:
Johan Hedberg48264f02011-11-09 13:58:58 +02002445 switch (addr_type) {
2446 case ADDR_LE_DEV_PUBLIC:
Andre Guedes591f47f2012-04-24 21:02:49 -03002447 return BDADDR_LE_PUBLIC;
Andre Guedes0ed09142012-04-03 08:46:54 -03002448
Johan Hedberg48264f02011-11-09 13:58:58 +02002449 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002450 /* Fallback to LE Random address type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002451 return BDADDR_LE_RANDOM;
Johan Hedberg48264f02011-11-09 13:58:58 +02002452 }
Andre Guedes0ed09142012-04-03 08:46:54 -03002453
Johan Hedberg4c659c32011-11-07 23:13:39 +02002454 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002455 /* Fallback to BR/EDR type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002456 return BDADDR_BREDR;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002457 }
2458}
2459
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002460static int get_connections(struct sock *sk, struct hci_dev *hdev, void *data,
2461 u16 data_len)
Johan Hedberg2784eb42011-01-21 13:56:35 +02002462{
Johan Hedberg2784eb42011-01-21 13:56:35 +02002463 struct mgmt_rp_get_connections *rp;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +02002464 struct hci_conn *c;
Johan Hedberga38528f2011-01-22 06:46:43 +02002465 size_t rp_len;
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002466 int err;
2467 u16 i;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002468
2469 BT_DBG("");
2470
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002471 hci_dev_lock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002472
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002473 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002474 err = cmd_status(sk, hdev->id, MGMT_OP_GET_CONNECTIONS,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002475 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002476 goto unlock;
2477 }
2478
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002479 i = 0;
Johan Hedbergb644ba32012-01-17 21:48:47 +02002480 list_for_each_entry(c, &hdev->conn_hash.list, list) {
2481 if (test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002482 i++;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002483 }
2484
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002485 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Andre Guedes92c4c202012-06-07 19:05:44 -03002486 rp = kmalloc(rp_len, GFP_KERNEL);
Johan Hedberga38528f2011-01-22 06:46:43 +02002487 if (!rp) {
Johan Hedberg2784eb42011-01-21 13:56:35 +02002488 err = -ENOMEM;
2489 goto unlock;
2490 }
2491
Johan Hedberg2784eb42011-01-21 13:56:35 +02002492 i = 0;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002493 list_for_each_entry(c, &hdev->conn_hash.list, list) {
Johan Hedbergb644ba32012-01-17 21:48:47 +02002494 if (!test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
2495 continue;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002496 bacpy(&rp->addr[i].bdaddr, &c->dst);
Andre Guedes57c14772012-04-24 21:02:50 -03002497 rp->addr[i].type = link_to_bdaddr(c->type, c->dst_type);
Andre Guedes0ed09142012-04-03 08:46:54 -03002498 if (c->type == SCO_LINK || c->type == ESCO_LINK)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002499 continue;
2500 i++;
2501 }
2502
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002503 rp->conn_count = cpu_to_le16(i);
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002504
Johan Hedberg4c659c32011-11-07 23:13:39 +02002505 /* Recalculate length in case of filtered SCO connections, etc */
2506 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Johan Hedberg2784eb42011-01-21 13:56:35 +02002507
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002508 err = cmd_complete(sk, hdev->id, MGMT_OP_GET_CONNECTIONS, 0, rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002509 rp_len);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002510
Johan Hedberga38528f2011-01-22 06:46:43 +02002511 kfree(rp);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002512
2513unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002514 hci_dev_unlock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002515 return err;
2516}
2517
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002518static int send_pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002519 struct mgmt_cp_pin_code_neg_reply *cp)
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002520{
2521 struct pending_cmd *cmd;
2522 int err;
2523
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002524 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_NEG_REPLY, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002525 sizeof(*cp));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002526 if (!cmd)
2527 return -ENOMEM;
2528
Johan Hedbergd8457692012-02-17 14:24:57 +02002529 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_NEG_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002530 sizeof(cp->addr.bdaddr), &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002531 if (err < 0)
2532 mgmt_pending_remove(cmd);
2533
2534 return err;
2535}
2536
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002537static int pin_code_reply(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002538 u16 len)
Johan Hedberg980e1a52011-01-22 06:10:07 +02002539{
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002540 struct hci_conn *conn;
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002541 struct mgmt_cp_pin_code_reply *cp = data;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002542 struct hci_cp_pin_code_reply reply;
Johan Hedberg366a0332011-02-19 12:05:55 -03002543 struct pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002544 int err;
2545
2546 BT_DBG("");
2547
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002548 hci_dev_lock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002549
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002550 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002551 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002552 MGMT_STATUS_NOT_POWERED);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002553 goto failed;
2554 }
2555
Johan Hedbergd8457692012-02-17 14:24:57 +02002556 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002557 if (!conn) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002558 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002559 MGMT_STATUS_NOT_CONNECTED);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002560 goto failed;
2561 }
2562
2563 if (conn->pending_sec_level == BT_SECURITY_HIGH && cp->pin_len != 16) {
Johan Hedbergd8457692012-02-17 14:24:57 +02002564 struct mgmt_cp_pin_code_neg_reply ncp;
2565
2566 memcpy(&ncp.addr, &cp->addr, sizeof(ncp.addr));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002567
2568 BT_ERR("PIN code is not 16 bytes long");
2569
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002570 err = send_pin_code_neg_reply(sk, hdev, &ncp);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002571 if (err >= 0)
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002572 err = cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002573 MGMT_STATUS_INVALID_PARAMS);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002574
2575 goto failed;
2576 }
2577
Gustavo F. Padovan00abfe42012-03-01 00:37:10 -03002578 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_REPLY, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002579 if (!cmd) {
2580 err = -ENOMEM;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002581 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002582 }
Johan Hedberg980e1a52011-01-22 06:10:07 +02002583
Johan Hedbergd8457692012-02-17 14:24:57 +02002584 bacpy(&reply.bdaddr, &cp->addr.bdaddr);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002585 reply.pin_len = cp->pin_len;
Waldemar Rymarkiewicz24718ca2011-06-01 17:28:47 +02002586 memcpy(reply.pin_code, cp->pin_code, sizeof(reply.pin_code));
Johan Hedberg980e1a52011-01-22 06:10:07 +02002587
2588 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_REPLY, sizeof(reply), &reply);
2589 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002590 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002591
2592failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002593 hci_dev_unlock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002594 return err;
2595}
2596
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002597static int set_io_capability(struct sock *sk, struct hci_dev *hdev, void *data,
2598 u16 len)
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002599{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002600 struct mgmt_cp_set_io_capability *cp = data;
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002601
2602 BT_DBG("");
2603
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002604 hci_dev_lock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002605
2606 hdev->io_capability = cp->io_capability;
2607
2608 BT_DBG("%s IO capability set to 0x%02x", hdev->name,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002609 hdev->io_capability);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002610
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002611 hci_dev_unlock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002612
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002613 return cmd_complete(sk, hdev->id, MGMT_OP_SET_IO_CAPABILITY, 0, NULL,
2614 0);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002615}
2616
Gustavo Padovan6039aa72012-05-23 04:04:18 -03002617static struct pending_cmd *find_pairing(struct hci_conn *conn)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002618{
2619 struct hci_dev *hdev = conn->hdev;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +02002620 struct pending_cmd *cmd;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002621
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002622 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
Johan Hedberge9a416b2011-02-19 12:05:56 -03002623 if (cmd->opcode != MGMT_OP_PAIR_DEVICE)
2624 continue;
2625
Johan Hedberge9a416b2011-02-19 12:05:56 -03002626 if (cmd->user_data != conn)
2627 continue;
2628
2629 return cmd;
2630 }
2631
2632 return NULL;
2633}
2634
2635static void pairing_complete(struct pending_cmd *cmd, u8 status)
2636{
2637 struct mgmt_rp_pair_device rp;
2638 struct hci_conn *conn = cmd->user_data;
2639
Johan Hedbergba4e5642011-11-11 00:07:34 +02002640 bacpy(&rp.addr.bdaddr, &conn->dst);
Andre Guedes57c14772012-04-24 21:02:50 -03002641 rp.addr.type = link_to_bdaddr(conn->type, conn->dst_type);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002642
Johan Hedbergaee9b212012-02-18 15:07:59 +02002643 cmd_complete(cmd->sk, cmd->index, MGMT_OP_PAIR_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002644 &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002645
2646 /* So we don't get further callbacks for this connection */
2647 conn->connect_cfm_cb = NULL;
2648 conn->security_cfm_cb = NULL;
2649 conn->disconn_cfm_cb = NULL;
2650
David Herrmann76a68ba2013-04-06 20:28:37 +02002651 hci_conn_drop(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002652
Johan Hedberga664b5b2011-02-19 12:06:02 -03002653 mgmt_pending_remove(cmd);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002654}
2655
2656static void pairing_complete_cb(struct hci_conn *conn, u8 status)
2657{
2658 struct pending_cmd *cmd;
2659
2660 BT_DBG("status %u", status);
2661
Johan Hedberg56e5cb82011-11-08 20:40:16 +02002662 cmd = find_pairing(conn);
2663 if (!cmd)
2664 BT_DBG("Unable to find a pending command");
2665 else
Johan Hedberge2113262012-02-18 15:20:03 +02002666 pairing_complete(cmd, mgmt_status(status));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002667}
2668
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302669static void le_connect_complete_cb(struct hci_conn *conn, u8 status)
2670{
2671 struct pending_cmd *cmd;
2672
2673 BT_DBG("status %u", status);
2674
2675 if (!status)
2676 return;
2677
2678 cmd = find_pairing(conn);
2679 if (!cmd)
2680 BT_DBG("Unable to find a pending command");
2681 else
2682 pairing_complete(cmd, mgmt_status(status));
2683}
2684
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002685static int pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002686 u16 len)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002687{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002688 struct mgmt_cp_pair_device *cp = data;
Johan Hedberg1425acb2011-11-11 00:07:35 +02002689 struct mgmt_rp_pair_device rp;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002690 struct pending_cmd *cmd;
2691 u8 sec_level, auth_type;
2692 struct hci_conn *conn;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002693 int err;
2694
2695 BT_DBG("");
2696
Szymon Jancf950a30e2013-01-18 12:48:07 +01002697 memset(&rp, 0, sizeof(rp));
2698 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2699 rp.addr.type = cp->addr.type;
2700
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002701 if (!bdaddr_type_is_valid(cp->addr.type))
2702 return cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2703 MGMT_STATUS_INVALID_PARAMS,
2704 &rp, sizeof(rp));
2705
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002706 hci_dev_lock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002707
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002708 if (!hdev_is_powered(hdev)) {
Szymon Jancf950a30e2013-01-18 12:48:07 +01002709 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2710 MGMT_STATUS_NOT_POWERED, &rp, sizeof(rp));
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002711 goto unlock;
2712 }
2713
Vinicius Costa Gomesc908df32011-09-02 14:51:22 -03002714 sec_level = BT_SECURITY_MEDIUM;
2715 if (cp->io_cap == 0x03)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002716 auth_type = HCI_AT_DEDICATED_BONDING;
Vinicius Costa Gomesc908df32011-09-02 14:51:22 -03002717 else
Johan Hedberge9a416b2011-02-19 12:05:56 -03002718 auth_type = HCI_AT_DEDICATED_BONDING_MITM;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002719
Andre Guedes591f47f2012-04-24 21:02:49 -03002720 if (cp->addr.type == BDADDR_BREDR)
Andre Guedesb12f62c2012-04-24 21:02:54 -03002721 conn = hci_connect(hdev, ACL_LINK, &cp->addr.bdaddr,
2722 cp->addr.type, sec_level, auth_type);
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002723 else
Andre Guedesb12f62c2012-04-24 21:02:54 -03002724 conn = hci_connect(hdev, LE_LINK, &cp->addr.bdaddr,
2725 cp->addr.type, sec_level, auth_type);
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002726
Ville Tervo30e76272011-02-22 16:10:53 -03002727 if (IS_ERR(conn)) {
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002728 int status;
2729
2730 if (PTR_ERR(conn) == -EBUSY)
2731 status = MGMT_STATUS_BUSY;
2732 else
2733 status = MGMT_STATUS_CONNECT_FAILED;
2734
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002735 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002736 status, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002737 sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002738 goto unlock;
2739 }
2740
2741 if (conn->connect_cfm_cb) {
David Herrmann76a68ba2013-04-06 20:28:37 +02002742 hci_conn_drop(conn);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002743 err = cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002744 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002745 goto unlock;
2746 }
2747
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002748 cmd = mgmt_pending_add(sk, MGMT_OP_PAIR_DEVICE, hdev, data, len);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002749 if (!cmd) {
2750 err = -ENOMEM;
David Herrmann76a68ba2013-04-06 20:28:37 +02002751 hci_conn_drop(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002752 goto unlock;
2753 }
2754
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002755 /* For LE, just connecting isn't a proof that the pairing finished */
Andre Guedes591f47f2012-04-24 21:02:49 -03002756 if (cp->addr.type == BDADDR_BREDR)
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002757 conn->connect_cfm_cb = pairing_complete_cb;
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302758 else
2759 conn->connect_cfm_cb = le_connect_complete_cb;
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002760
Johan Hedberge9a416b2011-02-19 12:05:56 -03002761 conn->security_cfm_cb = pairing_complete_cb;
2762 conn->disconn_cfm_cb = pairing_complete_cb;
2763 conn->io_capability = cp->io_cap;
2764 cmd->user_data = conn;
2765
2766 if (conn->state == BT_CONNECTED &&
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002767 hci_conn_security(conn, sec_level, auth_type))
Johan Hedberge9a416b2011-02-19 12:05:56 -03002768 pairing_complete(cmd, 0);
2769
2770 err = 0;
2771
2772unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002773 hci_dev_unlock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002774 return err;
2775}
2776
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002777static int cancel_pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
2778 u16 len)
Johan Hedberg28424702012-02-02 04:02:29 +02002779{
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02002780 struct mgmt_addr_info *addr = data;
Johan Hedberg28424702012-02-02 04:02:29 +02002781 struct pending_cmd *cmd;
2782 struct hci_conn *conn;
2783 int err;
2784
2785 BT_DBG("");
2786
Johan Hedberg28424702012-02-02 04:02:29 +02002787 hci_dev_lock(hdev);
2788
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002789 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002790 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002791 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002792 goto unlock;
2793 }
2794
Johan Hedberg28424702012-02-02 04:02:29 +02002795 cmd = mgmt_pending_find(MGMT_OP_PAIR_DEVICE, hdev);
2796 if (!cmd) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002797 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002798 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002799 goto unlock;
2800 }
2801
2802 conn = cmd->user_data;
2803
2804 if (bacmp(&addr->bdaddr, &conn->dst) != 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002805 err = cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002806 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002807 goto unlock;
2808 }
2809
2810 pairing_complete(cmd, MGMT_STATUS_CANCELLED);
2811
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002812 err = cmd_complete(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002813 addr, sizeof(*addr));
Johan Hedberg28424702012-02-02 04:02:29 +02002814unlock:
2815 hci_dev_unlock(hdev);
Johan Hedberg28424702012-02-02 04:02:29 +02002816 return err;
2817}
2818
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002819static int user_pairing_resp(struct sock *sk, struct hci_dev *hdev,
Johan Hedberg1707c602013-03-15 17:07:15 -05002820 struct mgmt_addr_info *addr, u16 mgmt_op,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002821 u16 hci_op, __le32 passkey)
Johan Hedberga5c29682011-02-19 12:05:57 -03002822{
Johan Hedberga5c29682011-02-19 12:05:57 -03002823 struct pending_cmd *cmd;
Brian Gix0df4c182011-11-16 13:53:13 -08002824 struct hci_conn *conn;
Johan Hedberga5c29682011-02-19 12:05:57 -03002825 int err;
2826
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002827 hci_dev_lock(hdev);
Johan Hedberg08ba5382011-03-16 14:29:34 +02002828
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002829 if (!hdev_is_powered(hdev)) {
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002830 err = cmd_complete(sk, hdev->id, mgmt_op,
2831 MGMT_STATUS_NOT_POWERED, addr,
2832 sizeof(*addr));
Brian Gix0df4c182011-11-16 13:53:13 -08002833 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002834 }
2835
Johan Hedberg1707c602013-03-15 17:07:15 -05002836 if (addr->type == BDADDR_BREDR)
2837 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &addr->bdaddr);
Johan Hedberg272d90d2012-02-09 15:26:12 +02002838 else
Johan Hedberg1707c602013-03-15 17:07:15 -05002839 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &addr->bdaddr);
Brian Gix47c15e22011-11-16 13:53:14 -08002840
Johan Hedberg272d90d2012-02-09 15:26:12 +02002841 if (!conn) {
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002842 err = cmd_complete(sk, hdev->id, mgmt_op,
2843 MGMT_STATUS_NOT_CONNECTED, addr,
2844 sizeof(*addr));
Johan Hedberg272d90d2012-02-09 15:26:12 +02002845 goto done;
2846 }
2847
Johan Hedberg1707c602013-03-15 17:07:15 -05002848 if (addr->type == BDADDR_LE_PUBLIC || addr->type == BDADDR_LE_RANDOM) {
Brian Gix47c15e22011-11-16 13:53:14 -08002849 /* Continue with pairing via SMP */
Brian Gix5fe57d92011-12-21 16:12:13 -08002850 err = smp_user_confirm_reply(conn, mgmt_op, passkey);
Brian Gix47c15e22011-11-16 13:53:14 -08002851
Brian Gix5fe57d92011-12-21 16:12:13 -08002852 if (!err)
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002853 err = cmd_complete(sk, hdev->id, mgmt_op,
2854 MGMT_STATUS_SUCCESS, addr,
2855 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002856 else
Johan Hedbergfeb94d32013-03-15 17:07:16 -05002857 err = cmd_complete(sk, hdev->id, mgmt_op,
2858 MGMT_STATUS_FAILED, addr,
2859 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002860
Brian Gix47c15e22011-11-16 13:53:14 -08002861 goto done;
2862 }
2863
Johan Hedberg1707c602013-03-15 17:07:15 -05002864 cmd = mgmt_pending_add(sk, mgmt_op, hdev, addr, sizeof(*addr));
Johan Hedberga5c29682011-02-19 12:05:57 -03002865 if (!cmd) {
2866 err = -ENOMEM;
Brian Gix0df4c182011-11-16 13:53:13 -08002867 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002868 }
2869
Brian Gix0df4c182011-11-16 13:53:13 -08002870 /* Continue with pairing via HCI */
Brian Gix604086b2011-11-23 08:28:33 -08002871 if (hci_op == HCI_OP_USER_PASSKEY_REPLY) {
2872 struct hci_cp_user_passkey_reply cp;
2873
Johan Hedberg1707c602013-03-15 17:07:15 -05002874 bacpy(&cp.bdaddr, &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002875 cp.passkey = passkey;
2876 err = hci_send_cmd(hdev, hci_op, sizeof(cp), &cp);
2877 } else
Johan Hedberg1707c602013-03-15 17:07:15 -05002878 err = hci_send_cmd(hdev, hci_op, sizeof(addr->bdaddr),
2879 &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002880
Johan Hedberga664b5b2011-02-19 12:06:02 -03002881 if (err < 0)
2882 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03002883
Brian Gix0df4c182011-11-16 13:53:13 -08002884done:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002885 hci_dev_unlock(hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03002886 return err;
2887}
2888
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05302889static int pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
2890 void *data, u16 len)
2891{
2892 struct mgmt_cp_pin_code_neg_reply *cp = data;
2893
2894 BT_DBG("");
2895
Johan Hedberg1707c602013-03-15 17:07:15 -05002896 return user_pairing_resp(sk, hdev, &cp->addr,
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05302897 MGMT_OP_PIN_CODE_NEG_REPLY,
2898 HCI_OP_PIN_CODE_NEG_REPLY, 0);
2899}
2900
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002901static int user_confirm_reply(struct sock *sk, struct hci_dev *hdev, void *data,
2902 u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08002903{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002904 struct mgmt_cp_user_confirm_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08002905
2906 BT_DBG("");
2907
2908 if (len != sizeof(*cp))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002909 return cmd_status(sk, hdev->id, MGMT_OP_USER_CONFIRM_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002910 MGMT_STATUS_INVALID_PARAMS);
Brian Gix0df4c182011-11-16 13:53:13 -08002911
Johan Hedberg1707c602013-03-15 17:07:15 -05002912 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002913 MGMT_OP_USER_CONFIRM_REPLY,
2914 HCI_OP_USER_CONFIRM_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08002915}
2916
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002917static int user_confirm_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002918 void *data, u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08002919{
Johan Hedbergc9c26592011-12-15 00:47:41 +02002920 struct mgmt_cp_user_confirm_neg_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08002921
2922 BT_DBG("");
2923
Johan Hedberg1707c602013-03-15 17:07:15 -05002924 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002925 MGMT_OP_USER_CONFIRM_NEG_REPLY,
2926 HCI_OP_USER_CONFIRM_NEG_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08002927}
2928
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002929static int user_passkey_reply(struct sock *sk, struct hci_dev *hdev, void *data,
2930 u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08002931{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002932 struct mgmt_cp_user_passkey_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08002933
2934 BT_DBG("");
2935
Johan Hedberg1707c602013-03-15 17:07:15 -05002936 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002937 MGMT_OP_USER_PASSKEY_REPLY,
2938 HCI_OP_USER_PASSKEY_REPLY, cp->passkey);
Brian Gix604086b2011-11-23 08:28:33 -08002939}
2940
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002941static int user_passkey_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002942 void *data, u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08002943{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002944 struct mgmt_cp_user_passkey_neg_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08002945
2946 BT_DBG("");
2947
Johan Hedberg1707c602013-03-15 17:07:15 -05002948 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002949 MGMT_OP_USER_PASSKEY_NEG_REPLY,
2950 HCI_OP_USER_PASSKEY_NEG_REPLY, 0);
Brian Gix604086b2011-11-23 08:28:33 -08002951}
2952
Johan Hedberg13928972013-03-15 17:07:00 -05002953static void update_name(struct hci_request *req)
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002954{
Johan Hedberg13928972013-03-15 17:07:00 -05002955 struct hci_dev *hdev = req->hdev;
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002956 struct hci_cp_write_local_name cp;
2957
Johan Hedberg13928972013-03-15 17:07:00 -05002958 memcpy(cp.name, hdev->dev_name, sizeof(cp.name));
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002959
Johan Hedberg890ea892013-03-15 17:06:52 -05002960 hci_req_add(req, HCI_OP_WRITE_LOCAL_NAME, sizeof(cp), &cp);
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002961}
2962
Johan Hedberg13928972013-03-15 17:07:00 -05002963static void set_name_complete(struct hci_dev *hdev, u8 status)
2964{
2965 struct mgmt_cp_set_local_name *cp;
2966 struct pending_cmd *cmd;
2967
2968 BT_DBG("status 0x%02x", status);
2969
2970 hci_dev_lock(hdev);
2971
2972 cmd = mgmt_pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
2973 if (!cmd)
2974 goto unlock;
2975
2976 cp = cmd->param;
2977
2978 if (status)
2979 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME,
2980 mgmt_status(status));
2981 else
2982 cmd_complete(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
2983 cp, sizeof(*cp));
2984
2985 mgmt_pending_remove(cmd);
2986
2987unlock:
2988 hci_dev_unlock(hdev);
2989}
2990
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002991static int set_local_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002992 u16 len)
Johan Hedbergb312b1612011-03-16 14:29:37 +02002993{
Johan Hedberg2b4bf392012-03-03 00:19:06 +02002994 struct mgmt_cp_set_local_name *cp = data;
Johan Hedbergb312b1612011-03-16 14:29:37 +02002995 struct pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05002996 struct hci_request req;
Johan Hedbergb312b1612011-03-16 14:29:37 +02002997 int err;
2998
2999 BT_DBG("");
3000
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003001 hci_dev_lock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003002
Johan Hedbergb3f2ca92013-03-15 17:07:03 -05003003 /* If the old values are the same as the new ones just return a
3004 * direct command complete event.
3005 */
3006 if (!memcmp(hdev->dev_name, cp->name, sizeof(hdev->dev_name)) &&
3007 !memcmp(hdev->short_name, cp->short_name,
3008 sizeof(hdev->short_name))) {
3009 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
3010 data, len);
3011 goto failed;
3012 }
3013
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003014 memcpy(hdev->short_name, cp->short_name, sizeof(hdev->short_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003015
Johan Hedbergb5235a62012-02-21 14:32:24 +02003016 if (!hdev_is_powered(hdev)) {
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003017 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003018
3019 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003020 data, len);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003021 if (err < 0)
3022 goto failed;
3023
3024 err = mgmt_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, data, len,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003025 sk);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003026
Johan Hedbergb5235a62012-02-21 14:32:24 +02003027 goto failed;
3028 }
3029
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003030 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LOCAL_NAME, hdev, data, len);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003031 if (!cmd) {
3032 err = -ENOMEM;
3033 goto failed;
3034 }
3035
Johan Hedberg13928972013-03-15 17:07:00 -05003036 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
3037
Johan Hedberg890ea892013-03-15 17:06:52 -05003038 hci_req_init(&req, hdev);
Johan Hedberg3f985052013-03-15 17:07:02 -05003039
3040 if (lmp_bredr_capable(hdev)) {
3041 update_name(&req);
3042 update_eir(&req);
3043 }
3044
Marcel Holtmann7a5f4992013-10-16 00:16:49 -07003045 /* The name is stored in the scan response data and so
3046 * no need to udpate the advertising data here.
3047 */
Johan Hedberg3f985052013-03-15 17:07:02 -05003048 if (lmp_le_capable(hdev))
Marcel Holtmann7a5f4992013-10-16 00:16:49 -07003049 update_scan_rsp_data(&req);
Johan Hedberg3f985052013-03-15 17:07:02 -05003050
Johan Hedberg13928972013-03-15 17:07:00 -05003051 err = hci_req_run(&req, set_name_complete);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003052 if (err < 0)
3053 mgmt_pending_remove(cmd);
3054
3055failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003056 hci_dev_unlock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003057 return err;
3058}
3059
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02003060static int read_local_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003061 void *data, u16 data_len)
Szymon Jancc35938b2011-03-22 13:12:21 +01003062{
Szymon Jancc35938b2011-03-22 13:12:21 +01003063 struct pending_cmd *cmd;
3064 int err;
3065
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003066 BT_DBG("%s", hdev->name);
Szymon Jancc35938b2011-03-22 13:12:21 +01003067
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003068 hci_dev_lock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003069
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003070 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003071 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003072 MGMT_STATUS_NOT_POWERED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003073 goto unlock;
3074 }
3075
Andre Guedes9a1a1992012-07-24 15:03:48 -03003076 if (!lmp_ssp_capable(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003077 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003078 MGMT_STATUS_NOT_SUPPORTED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003079 goto unlock;
3080 }
3081
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003082 if (mgmt_pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003083 err = cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003084 MGMT_STATUS_BUSY);
Szymon Jancc35938b2011-03-22 13:12:21 +01003085 goto unlock;
3086 }
3087
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003088 cmd = mgmt_pending_add(sk, MGMT_OP_READ_LOCAL_OOB_DATA, hdev, NULL, 0);
Szymon Jancc35938b2011-03-22 13:12:21 +01003089 if (!cmd) {
3090 err = -ENOMEM;
3091 goto unlock;
3092 }
3093
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08003094 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags))
3095 err = hci_send_cmd(hdev, HCI_OP_READ_LOCAL_OOB_EXT_DATA,
3096 0, NULL);
3097 else
3098 err = hci_send_cmd(hdev, HCI_OP_READ_LOCAL_OOB_DATA, 0, NULL);
3099
Szymon Jancc35938b2011-03-22 13:12:21 +01003100 if (err < 0)
3101 mgmt_pending_remove(cmd);
3102
3103unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003104 hci_dev_unlock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003105 return err;
3106}
3107
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003108static int add_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003109 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003110{
Szymon Janc2763eda2011-03-22 13:12:22 +01003111 int err;
3112
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003113 BT_DBG("%s ", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003114
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003115 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003116
Marcel Holtmannec109112014-01-10 02:07:30 -08003117 if (len == MGMT_ADD_REMOTE_OOB_DATA_SIZE) {
3118 struct mgmt_cp_add_remote_oob_data *cp = data;
3119 u8 status;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003120
Marcel Holtmannec109112014-01-10 02:07:30 -08003121 err = hci_add_remote_oob_data(hdev, &cp->addr.bdaddr,
3122 cp->hash, cp->randomizer);
3123 if (err < 0)
3124 status = MGMT_STATUS_FAILED;
3125 else
3126 status = MGMT_STATUS_SUCCESS;
3127
3128 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3129 status, &cp->addr, sizeof(cp->addr));
3130 } else if (len == MGMT_ADD_REMOTE_OOB_EXT_DATA_SIZE) {
3131 struct mgmt_cp_add_remote_oob_ext_data *cp = data;
3132 u8 status;
3133
3134 err = hci_add_remote_oob_ext_data(hdev, &cp->addr.bdaddr,
3135 cp->hash192,
3136 cp->randomizer192,
3137 cp->hash256,
3138 cp->randomizer256);
3139 if (err < 0)
3140 status = MGMT_STATUS_FAILED;
3141 else
3142 status = MGMT_STATUS_SUCCESS;
3143
3144 err = cmd_complete(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3145 status, &cp->addr, sizeof(cp->addr));
3146 } else {
3147 BT_ERR("add_remote_oob_data: invalid length of %u bytes", len);
3148 err = cmd_status(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3149 MGMT_STATUS_INVALID_PARAMS);
3150 }
Szymon Janc2763eda2011-03-22 13:12:22 +01003151
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003152 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003153 return err;
3154}
3155
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003156static int remove_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003157 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003158{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003159 struct mgmt_cp_remove_remote_oob_data *cp = data;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003160 u8 status;
Szymon Janc2763eda2011-03-22 13:12:22 +01003161 int err;
3162
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003163 BT_DBG("%s", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003164
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003165 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003166
Johan Hedberg664ce4c2012-02-09 15:44:09 +02003167 err = hci_remove_remote_oob_data(hdev, &cp->addr.bdaddr);
Szymon Janc2763eda2011-03-22 13:12:22 +01003168 if (err < 0)
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003169 status = MGMT_STATUS_INVALID_PARAMS;
Szymon Janc2763eda2011-03-22 13:12:22 +01003170 else
Szymon Janca6785be2012-12-13 15:11:21 +01003171 status = MGMT_STATUS_SUCCESS;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003172
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003173 err = cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_REMOTE_OOB_DATA,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003174 status, &cp->addr, sizeof(cp->addr));
Szymon Janc2763eda2011-03-22 13:12:22 +01003175
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003176 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003177 return err;
3178}
3179
Andre Guedes41dc2bd2013-04-30 15:29:30 -03003180static int mgmt_start_discovery_failed(struct hci_dev *hdev, u8 status)
3181{
3182 struct pending_cmd *cmd;
3183 u8 type;
3184 int err;
3185
3186 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3187
3188 cmd = mgmt_pending_find(MGMT_OP_START_DISCOVERY, hdev);
3189 if (!cmd)
3190 return -ENOENT;
3191
3192 type = hdev->discovery.type;
3193
3194 err = cmd_complete(cmd->sk, hdev->id, cmd->opcode, mgmt_status(status),
3195 &type, sizeof(type));
3196 mgmt_pending_remove(cmd);
3197
3198 return err;
3199}
3200
Andre Guedes7c307722013-04-30 15:29:28 -03003201static void start_discovery_complete(struct hci_dev *hdev, u8 status)
3202{
3203 BT_DBG("status %d", status);
3204
3205 if (status) {
3206 hci_dev_lock(hdev);
3207 mgmt_start_discovery_failed(hdev, status);
3208 hci_dev_unlock(hdev);
3209 return;
3210 }
3211
3212 hci_dev_lock(hdev);
3213 hci_discovery_set_state(hdev, DISCOVERY_FINDING);
3214 hci_dev_unlock(hdev);
3215
3216 switch (hdev->discovery.type) {
3217 case DISCOV_TYPE_LE:
3218 queue_delayed_work(hdev->workqueue, &hdev->le_scan_disable,
Andre Guedes0d8cc932013-04-30 15:29:31 -03003219 DISCOV_LE_TIMEOUT);
Andre Guedes7c307722013-04-30 15:29:28 -03003220 break;
3221
3222 case DISCOV_TYPE_INTERLEAVED:
3223 queue_delayed_work(hdev->workqueue, &hdev->le_scan_disable,
Andre Guedes0d8cc932013-04-30 15:29:31 -03003224 DISCOV_INTERLEAVED_TIMEOUT);
Andre Guedes7c307722013-04-30 15:29:28 -03003225 break;
3226
3227 case DISCOV_TYPE_BREDR:
3228 break;
3229
3230 default:
3231 BT_ERR("Invalid discovery type %d", hdev->discovery.type);
3232 }
3233}
3234
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003235static int start_discovery(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003236 void *data, u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003237{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003238 struct mgmt_cp_start_discovery *cp = data;
Johan Hedberg14a53662011-04-27 10:29:56 -04003239 struct pending_cmd *cmd;
Andre Guedes7c307722013-04-30 15:29:28 -03003240 struct hci_cp_le_set_scan_param param_cp;
3241 struct hci_cp_le_set_scan_enable enable_cp;
3242 struct hci_cp_inquiry inq_cp;
3243 struct hci_request req;
3244 /* General inquiry access code (GIAC) */
3245 u8 lap[3] = { 0x33, 0x8b, 0x9e };
Johan Hedberge6fe7982013-10-02 15:45:22 +03003246 u8 status;
Johan Hedberg14a53662011-04-27 10:29:56 -04003247 int err;
3248
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003249 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003250
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003251 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003252
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003253 if (!hdev_is_powered(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003254 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003255 MGMT_STATUS_NOT_POWERED);
Johan Hedbergbd2d1332011-11-07 23:13:37 +02003256 goto failed;
3257 }
3258
Andre Guedes642be6c2012-03-21 00:03:37 -03003259 if (test_bit(HCI_PERIODIC_INQ, &hdev->dev_flags)) {
3260 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3261 MGMT_STATUS_BUSY);
3262 goto failed;
3263 }
3264
Johan Hedbergff9ef572012-01-04 14:23:45 +02003265 if (hdev->discovery.state != DISCOVERY_STOPPED) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003266 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003267 MGMT_STATUS_BUSY);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003268 goto failed;
3269 }
3270
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003271 cmd = mgmt_pending_add(sk, MGMT_OP_START_DISCOVERY, hdev, NULL, 0);
Johan Hedberg14a53662011-04-27 10:29:56 -04003272 if (!cmd) {
3273 err = -ENOMEM;
3274 goto failed;
3275 }
3276
Andre Guedes4aab14e2012-02-17 20:39:36 -03003277 hdev->discovery.type = cp->type;
3278
Andre Guedes7c307722013-04-30 15:29:28 -03003279 hci_req_init(&req, hdev);
3280
Andre Guedes4aab14e2012-02-17 20:39:36 -03003281 switch (hdev->discovery.type) {
Andre Guedesf39799f2012-02-17 20:39:35 -03003282 case DISCOV_TYPE_BREDR:
Johan Hedberge6fe7982013-10-02 15:45:22 +03003283 status = mgmt_bredr_support(hdev);
3284 if (status) {
Johan Hedberg04106752013-01-10 14:54:09 +02003285 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003286 status);
Johan Hedberg04106752013-01-10 14:54:09 +02003287 mgmt_pending_remove(cmd);
3288 goto failed;
3289 }
3290
Andre Guedes7c307722013-04-30 15:29:28 -03003291 if (test_bit(HCI_INQUIRY, &hdev->flags)) {
3292 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3293 MGMT_STATUS_BUSY);
3294 mgmt_pending_remove(cmd);
3295 goto failed;
3296 }
3297
3298 hci_inquiry_cache_flush(hdev);
3299
3300 memset(&inq_cp, 0, sizeof(inq_cp));
3301 memcpy(&inq_cp.lap, lap, sizeof(inq_cp.lap));
Andre Guedes0d8cc932013-04-30 15:29:31 -03003302 inq_cp.length = DISCOV_BREDR_INQUIRY_LEN;
Andre Guedes7c307722013-04-30 15:29:28 -03003303 hci_req_add(&req, HCI_OP_INQUIRY, sizeof(inq_cp), &inq_cp);
Andre Guedesf39799f2012-02-17 20:39:35 -03003304 break;
3305
3306 case DISCOV_TYPE_LE:
Andre Guedes7c307722013-04-30 15:29:28 -03003307 case DISCOV_TYPE_INTERLEAVED:
Johan Hedberge6fe7982013-10-02 15:45:22 +03003308 status = mgmt_le_support(hdev);
3309 if (status) {
Johan Hedberg04106752013-01-10 14:54:09 +02003310 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003311 status);
Johan Hedberg04106752013-01-10 14:54:09 +02003312 mgmt_pending_remove(cmd);
3313 goto failed;
3314 }
3315
Andre Guedes7c307722013-04-30 15:29:28 -03003316 if (hdev->discovery.type == DISCOV_TYPE_INTERLEAVED &&
Johan Hedberg56f87902013-10-02 13:43:13 +03003317 !test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
Johan Hedberg04106752013-01-10 14:54:09 +02003318 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3319 MGMT_STATUS_NOT_SUPPORTED);
3320 mgmt_pending_remove(cmd);
3321 goto failed;
3322 }
3323
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003324 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
Andre Guedes7c307722013-04-30 15:29:28 -03003325 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3326 MGMT_STATUS_REJECTED);
3327 mgmt_pending_remove(cmd);
3328 goto failed;
3329 }
3330
3331 if (test_bit(HCI_LE_SCAN, &hdev->dev_flags)) {
3332 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3333 MGMT_STATUS_BUSY);
3334 mgmt_pending_remove(cmd);
3335 goto failed;
3336 }
3337
3338 memset(&param_cp, 0, sizeof(param_cp));
3339 param_cp.type = LE_SCAN_ACTIVE;
Andre Guedes0d8cc932013-04-30 15:29:31 -03003340 param_cp.interval = cpu_to_le16(DISCOV_LE_SCAN_INT);
3341 param_cp.window = cpu_to_le16(DISCOV_LE_SCAN_WIN);
Marcel Holtmann79830f62013-10-18 16:38:09 -07003342 param_cp.own_address_type = hdev->own_addr_type;
Andre Guedes7c307722013-04-30 15:29:28 -03003343 hci_req_add(&req, HCI_OP_LE_SET_SCAN_PARAM, sizeof(param_cp),
3344 &param_cp);
3345
3346 memset(&enable_cp, 0, sizeof(enable_cp));
3347 enable_cp.enable = LE_SCAN_ENABLE;
3348 enable_cp.filter_dup = LE_SCAN_FILTER_DUP_ENABLE;
3349 hci_req_add(&req, HCI_OP_LE_SET_SCAN_ENABLE, sizeof(enable_cp),
3350 &enable_cp);
Andre Guedes5e0452c2012-02-17 20:39:38 -03003351 break;
3352
Andre Guedesf39799f2012-02-17 20:39:35 -03003353 default:
Johan Hedberg04106752013-01-10 14:54:09 +02003354 err = cmd_status(sk, hdev->id, MGMT_OP_START_DISCOVERY,
3355 MGMT_STATUS_INVALID_PARAMS);
3356 mgmt_pending_remove(cmd);
3357 goto failed;
Andre Guedesf39799f2012-02-17 20:39:35 -03003358 }
Andre Guedes3fd24152012-02-03 17:48:01 -03003359
Andre Guedes7c307722013-04-30 15:29:28 -03003360 err = hci_req_run(&req, start_discovery_complete);
Johan Hedberg14a53662011-04-27 10:29:56 -04003361 if (err < 0)
3362 mgmt_pending_remove(cmd);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003363 else
3364 hci_discovery_set_state(hdev, DISCOVERY_STARTING);
Johan Hedberg14a53662011-04-27 10:29:56 -04003365
3366failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003367 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003368 return err;
3369}
3370
Andre Guedes1183fdc2013-04-30 15:29:35 -03003371static int mgmt_stop_discovery_failed(struct hci_dev *hdev, u8 status)
3372{
3373 struct pending_cmd *cmd;
3374 int err;
3375
3376 cmd = mgmt_pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
3377 if (!cmd)
3378 return -ENOENT;
3379
3380 err = cmd_complete(cmd->sk, hdev->id, cmd->opcode, mgmt_status(status),
3381 &hdev->discovery.type, sizeof(hdev->discovery.type));
3382 mgmt_pending_remove(cmd);
3383
3384 return err;
3385}
3386
Andre Guedes0e05bba2013-04-30 15:29:33 -03003387static void stop_discovery_complete(struct hci_dev *hdev, u8 status)
3388{
3389 BT_DBG("status %d", status);
3390
3391 hci_dev_lock(hdev);
3392
3393 if (status) {
3394 mgmt_stop_discovery_failed(hdev, status);
3395 goto unlock;
3396 }
3397
3398 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3399
3400unlock:
3401 hci_dev_unlock(hdev);
3402}
3403
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003404static int stop_discovery(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003405 u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003406{
Johan Hedbergd9306502012-02-20 23:25:18 +02003407 struct mgmt_cp_stop_discovery *mgmt_cp = data;
Johan Hedberg14a53662011-04-27 10:29:56 -04003408 struct pending_cmd *cmd;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003409 struct hci_cp_remote_name_req_cancel cp;
3410 struct inquiry_entry *e;
Andre Guedes0e05bba2013-04-30 15:29:33 -03003411 struct hci_request req;
3412 struct hci_cp_le_set_scan_enable enable_cp;
Johan Hedberg14a53662011-04-27 10:29:56 -04003413 int err;
3414
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003415 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003416
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003417 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003418
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003419 if (!hci_discovery_active(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003420 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003421 MGMT_STATUS_REJECTED, &mgmt_cp->type,
3422 sizeof(mgmt_cp->type));
Johan Hedbergd9306502012-02-20 23:25:18 +02003423 goto unlock;
3424 }
3425
3426 if (hdev->discovery.type != mgmt_cp->type) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003427 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003428 MGMT_STATUS_INVALID_PARAMS, &mgmt_cp->type,
3429 sizeof(mgmt_cp->type));
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003430 goto unlock;
Johan Hedbergff9ef572012-01-04 14:23:45 +02003431 }
3432
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003433 cmd = mgmt_pending_add(sk, MGMT_OP_STOP_DISCOVERY, hdev, NULL, 0);
Johan Hedberg14a53662011-04-27 10:29:56 -04003434 if (!cmd) {
3435 err = -ENOMEM;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003436 goto unlock;
Johan Hedberg14a53662011-04-27 10:29:56 -04003437 }
3438
Andre Guedes0e05bba2013-04-30 15:29:33 -03003439 hci_req_init(&req, hdev);
3440
Andre Guedese0d9727e2012-03-20 15:15:36 -03003441 switch (hdev->discovery.state) {
3442 case DISCOVERY_FINDING:
Andre Guedes0e05bba2013-04-30 15:29:33 -03003443 if (test_bit(HCI_INQUIRY, &hdev->flags)) {
3444 hci_req_add(&req, HCI_OP_INQUIRY_CANCEL, 0, NULL);
3445 } else {
3446 cancel_delayed_work(&hdev->le_scan_disable);
3447
3448 memset(&enable_cp, 0, sizeof(enable_cp));
3449 enable_cp.enable = LE_SCAN_DISABLE;
3450 hci_req_add(&req, HCI_OP_LE_SET_SCAN_ENABLE,
3451 sizeof(enable_cp), &enable_cp);
3452 }
Andre Guedesc9ecc482012-03-15 16:52:08 -03003453
Andre Guedese0d9727e2012-03-20 15:15:36 -03003454 break;
3455
3456 case DISCOVERY_RESOLVING:
3457 e = hci_inquiry_cache_lookup_resolve(hdev, BDADDR_ANY,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003458 NAME_PENDING);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003459 if (!e) {
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003460 mgmt_pending_remove(cmd);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003461 err = cmd_complete(sk, hdev->id,
3462 MGMT_OP_STOP_DISCOVERY, 0,
3463 &mgmt_cp->type,
3464 sizeof(mgmt_cp->type));
3465 hci_discovery_set_state(hdev, DISCOVERY_STOPPED);
3466 goto unlock;
3467 }
3468
3469 bacpy(&cp.bdaddr, &e->data.bdaddr);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003470 hci_req_add(&req, HCI_OP_REMOTE_NAME_REQ_CANCEL, sizeof(cp),
3471 &cp);
Andre Guedese0d9727e2012-03-20 15:15:36 -03003472
3473 break;
3474
3475 default:
3476 BT_DBG("unknown discovery state %u", hdev->discovery.state);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003477
3478 mgmt_pending_remove(cmd);
3479 err = cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
3480 MGMT_STATUS_FAILED, &mgmt_cp->type,
3481 sizeof(mgmt_cp->type));
3482 goto unlock;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003483 }
3484
Andre Guedes0e05bba2013-04-30 15:29:33 -03003485 err = hci_req_run(&req, stop_discovery_complete);
Johan Hedberg14a53662011-04-27 10:29:56 -04003486 if (err < 0)
3487 mgmt_pending_remove(cmd);
Johan Hedbergff9ef572012-01-04 14:23:45 +02003488 else
3489 hci_discovery_set_state(hdev, DISCOVERY_STOPPING);
Johan Hedberg14a53662011-04-27 10:29:56 -04003490
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003491unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003492 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003493 return err;
3494}
3495
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003496static int confirm_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003497 u16 len)
Johan Hedberg561aafb2012-01-04 13:31:59 +02003498{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003499 struct mgmt_cp_confirm_name *cp = data;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003500 struct inquiry_entry *e;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003501 int err;
3502
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003503 BT_DBG("%s", hdev->name);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003504
Johan Hedberg561aafb2012-01-04 13:31:59 +02003505 hci_dev_lock(hdev);
3506
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003507 if (!hci_discovery_active(hdev)) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003508 err = cmd_status(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003509 MGMT_STATUS_FAILED);
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003510 goto failed;
3511 }
3512
Johan Hedberga198e7b2012-02-17 14:27:06 +02003513 e = hci_inquiry_cache_lookup_unknown(hdev, &cp->addr.bdaddr);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003514 if (!e) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003515 err = cmd_status(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003516 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003517 goto failed;
3518 }
3519
3520 if (cp->name_known) {
3521 e->name_state = NAME_KNOWN;
3522 list_del(&e->list);
3523 } else {
3524 e->name_state = NAME_NEEDED;
Johan Hedberga3d4e20a2012-01-09 00:53:02 +02003525 hci_inquiry_cache_update_resolve(hdev, e);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003526 }
3527
Johan Hedberge3846622013-01-09 15:29:33 +02003528 err = cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME, 0, &cp->addr,
3529 sizeof(cp->addr));
Johan Hedberg561aafb2012-01-04 13:31:59 +02003530
3531failed:
3532 hci_dev_unlock(hdev);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003533 return err;
3534}
3535
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003536static int block_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003537 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003538{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003539 struct mgmt_cp_block_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003540 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003541 int err;
3542
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003543 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003544
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003545 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg5d0846d2013-01-20 14:27:22 +02003546 return cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE,
3547 MGMT_STATUS_INVALID_PARAMS,
3548 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003549
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003550 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003551
Johan Hedberg88c1fe42012-02-09 15:56:11 +02003552 err = hci_blacklist_add(hdev, &cp->addr.bdaddr, cp->addr.type);
Antti Julku7fbec222011-06-15 12:01:15 +03003553 if (err < 0)
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003554 status = MGMT_STATUS_FAILED;
Antti Julku7fbec222011-06-15 12:01:15 +03003555 else
Szymon Janca6785be2012-12-13 15:11:21 +01003556 status = MGMT_STATUS_SUCCESS;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003557
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003558 err = cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003559 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003560
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003561 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003562
3563 return err;
3564}
3565
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003566static int unblock_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003567 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003568{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003569 struct mgmt_cp_unblock_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003570 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003571 int err;
3572
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003573 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003574
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003575 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg5d0846d2013-01-20 14:27:22 +02003576 return cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE,
3577 MGMT_STATUS_INVALID_PARAMS,
3578 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003579
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003580 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003581
Johan Hedberg88c1fe42012-02-09 15:56:11 +02003582 err = hci_blacklist_del(hdev, &cp->addr.bdaddr, cp->addr.type);
Antti Julku7fbec222011-06-15 12:01:15 +03003583 if (err < 0)
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003584 status = MGMT_STATUS_INVALID_PARAMS;
Antti Julku7fbec222011-06-15 12:01:15 +03003585 else
Szymon Janca6785be2012-12-13 15:11:21 +01003586 status = MGMT_STATUS_SUCCESS;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003587
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003588 err = cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE, status,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003589 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003590
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003591 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003592
3593 return err;
3594}
3595
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003596static int set_device_id(struct sock *sk, struct hci_dev *hdev, void *data,
3597 u16 len)
3598{
3599 struct mgmt_cp_set_device_id *cp = data;
Johan Hedberg890ea892013-03-15 17:06:52 -05003600 struct hci_request req;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003601 int err;
Szymon Jancc72d4b82012-03-16 16:02:57 +01003602 __u16 source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003603
3604 BT_DBG("%s", hdev->name);
3605
Szymon Jancc72d4b82012-03-16 16:02:57 +01003606 source = __le16_to_cpu(cp->source);
3607
3608 if (source > 0x0002)
3609 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEVICE_ID,
3610 MGMT_STATUS_INVALID_PARAMS);
3611
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003612 hci_dev_lock(hdev);
3613
Szymon Jancc72d4b82012-03-16 16:02:57 +01003614 hdev->devid_source = source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003615 hdev->devid_vendor = __le16_to_cpu(cp->vendor);
3616 hdev->devid_product = __le16_to_cpu(cp->product);
3617 hdev->devid_version = __le16_to_cpu(cp->version);
3618
3619 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_DEVICE_ID, 0, NULL, 0);
3620
Johan Hedberg890ea892013-03-15 17:06:52 -05003621 hci_req_init(&req, hdev);
3622 update_eir(&req);
3623 hci_req_run(&req, NULL);
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003624
3625 hci_dev_unlock(hdev);
3626
3627 return err;
3628}
3629
Johan Hedberg4375f102013-09-25 13:26:10 +03003630static void set_advertising_complete(struct hci_dev *hdev, u8 status)
3631{
3632 struct cmd_lookup match = { NULL, hdev };
3633
3634 if (status) {
3635 u8 mgmt_err = mgmt_status(status);
3636
3637 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev,
3638 cmd_status_rsp, &mgmt_err);
3639 return;
3640 }
3641
3642 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev, settings_rsp,
3643 &match);
3644
3645 new_settings(hdev, match.sk);
3646
3647 if (match.sk)
3648 sock_put(match.sk);
3649}
3650
Marcel Holtmann21b51872013-10-10 09:47:53 -07003651static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
3652 u16 len)
Johan Hedberg4375f102013-09-25 13:26:10 +03003653{
3654 struct mgmt_mode *cp = data;
3655 struct pending_cmd *cmd;
3656 struct hci_request req;
Johan Hedberge6fe7982013-10-02 15:45:22 +03003657 u8 val, enabled, status;
Johan Hedberg4375f102013-09-25 13:26:10 +03003658 int err;
3659
3660 BT_DBG("request for %s", hdev->name);
3661
Johan Hedberge6fe7982013-10-02 15:45:22 +03003662 status = mgmt_le_support(hdev);
3663 if (status)
Johan Hedberg4375f102013-09-25 13:26:10 +03003664 return cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
Johan Hedberge6fe7982013-10-02 15:45:22 +03003665 status);
Johan Hedberg4375f102013-09-25 13:26:10 +03003666
3667 if (cp->val != 0x00 && cp->val != 0x01)
3668 return cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3669 MGMT_STATUS_INVALID_PARAMS);
3670
3671 hci_dev_lock(hdev);
3672
3673 val = !!cp->val;
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003674 enabled = test_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedberg4375f102013-09-25 13:26:10 +03003675
Johan Hedbergf74ca9b2013-10-08 15:52:18 +02003676 /* The following conditions are ones which mean that we should
3677 * not do any HCI communication but directly send a mgmt
3678 * response to user space (after toggling the flag if
3679 * necessary).
3680 */
3681 if (!hdev_is_powered(hdev) || val == enabled ||
Marcel Holtmannb145edc2013-10-10 09:47:54 -07003682 hci_conn_num(hdev, LE_LINK) > 0) {
Johan Hedberg4375f102013-09-25 13:26:10 +03003683 bool changed = false;
3684
Johan Hedbergf3d3444a2013-10-05 12:01:04 +02003685 if (val != test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
3686 change_bit(HCI_ADVERTISING, &hdev->dev_flags);
Johan Hedberg4375f102013-09-25 13:26:10 +03003687 changed = true;
3688 }
3689
3690 err = send_settings_rsp(sk, MGMT_OP_SET_ADVERTISING, hdev);
3691 if (err < 0)
3692 goto unlock;
3693
3694 if (changed)
3695 err = new_settings(hdev, sk);
3696
3697 goto unlock;
3698 }
3699
3700 if (mgmt_pending_find(MGMT_OP_SET_ADVERTISING, hdev) ||
3701 mgmt_pending_find(MGMT_OP_SET_LE, hdev)) {
3702 err = cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3703 MGMT_STATUS_BUSY);
3704 goto unlock;
3705 }
3706
3707 cmd = mgmt_pending_add(sk, MGMT_OP_SET_ADVERTISING, hdev, data, len);
3708 if (!cmd) {
3709 err = -ENOMEM;
3710 goto unlock;
3711 }
3712
3713 hci_req_init(&req, hdev);
3714
Marcel Holtmannbba3aa52013-10-06 02:55:21 -07003715 if (val)
3716 enable_advertising(&req);
3717 else
3718 disable_advertising(&req);
Johan Hedberg4375f102013-09-25 13:26:10 +03003719
3720 err = hci_req_run(&req, set_advertising_complete);
3721 if (err < 0)
3722 mgmt_pending_remove(cmd);
3723
3724unlock:
3725 hci_dev_unlock(hdev);
3726 return err;
3727}
3728
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003729static int set_static_address(struct sock *sk, struct hci_dev *hdev,
3730 void *data, u16 len)
3731{
3732 struct mgmt_cp_set_static_address *cp = data;
3733 int err;
3734
3735 BT_DBG("%s", hdev->name);
3736
Marcel Holtmann62af4442013-10-02 22:10:32 -07003737 if (!lmp_le_capable(hdev))
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003738 return cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
Marcel Holtmann62af4442013-10-02 22:10:32 -07003739 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07003740
3741 if (hdev_is_powered(hdev))
3742 return cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
3743 MGMT_STATUS_REJECTED);
3744
3745 if (bacmp(&cp->bdaddr, BDADDR_ANY)) {
3746 if (!bacmp(&cp->bdaddr, BDADDR_NONE))
3747 return cmd_status(sk, hdev->id,
3748 MGMT_OP_SET_STATIC_ADDRESS,
3749 MGMT_STATUS_INVALID_PARAMS);
3750
3751 /* Two most significant bits shall be set */
3752 if ((cp->bdaddr.b[5] & 0xc0) != 0xc0)
3753 return cmd_status(sk, hdev->id,
3754 MGMT_OP_SET_STATIC_ADDRESS,
3755 MGMT_STATUS_INVALID_PARAMS);
3756 }
3757
3758 hci_dev_lock(hdev);
3759
3760 bacpy(&hdev->static_addr, &cp->bdaddr);
3761
3762 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS, 0, NULL, 0);
3763
3764 hci_dev_unlock(hdev);
3765
3766 return err;
3767}
3768
Marcel Holtmann14b49b92013-10-11 08:23:20 -07003769static int set_scan_params(struct sock *sk, struct hci_dev *hdev,
3770 void *data, u16 len)
3771{
3772 struct mgmt_cp_set_scan_params *cp = data;
3773 __u16 interval, window;
3774 int err;
3775
3776 BT_DBG("%s", hdev->name);
3777
3778 if (!lmp_le_capable(hdev))
3779 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3780 MGMT_STATUS_NOT_SUPPORTED);
3781
3782 interval = __le16_to_cpu(cp->interval);
3783
3784 if (interval < 0x0004 || interval > 0x4000)
3785 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3786 MGMT_STATUS_INVALID_PARAMS);
3787
3788 window = __le16_to_cpu(cp->window);
3789
3790 if (window < 0x0004 || window > 0x4000)
3791 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3792 MGMT_STATUS_INVALID_PARAMS);
3793
Marcel Holtmann899e1072013-10-14 09:55:32 -07003794 if (window > interval)
3795 return cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
3796 MGMT_STATUS_INVALID_PARAMS);
3797
Marcel Holtmann14b49b92013-10-11 08:23:20 -07003798 hci_dev_lock(hdev);
3799
3800 hdev->le_scan_interval = interval;
3801 hdev->le_scan_window = window;
3802
3803 err = cmd_complete(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS, 0, NULL, 0);
3804
3805 hci_dev_unlock(hdev);
3806
3807 return err;
3808}
3809
Johan Hedberg33e38b32013-03-15 17:07:05 -05003810static void fast_connectable_complete(struct hci_dev *hdev, u8 status)
3811{
3812 struct pending_cmd *cmd;
3813
3814 BT_DBG("status 0x%02x", status);
3815
3816 hci_dev_lock(hdev);
3817
3818 cmd = mgmt_pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev);
3819 if (!cmd)
3820 goto unlock;
3821
3822 if (status) {
3823 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3824 mgmt_status(status));
3825 } else {
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05003826 struct mgmt_mode *cp = cmd->param;
3827
3828 if (cp->val)
3829 set_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3830 else
3831 clear_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3832
Johan Hedberg33e38b32013-03-15 17:07:05 -05003833 send_settings_rsp(cmd->sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev);
3834 new_settings(hdev, cmd->sk);
3835 }
3836
3837 mgmt_pending_remove(cmd);
3838
3839unlock:
3840 hci_dev_unlock(hdev);
3841}
3842
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003843static int set_fast_connectable(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003844 void *data, u16 len)
Antti Julkuf6422ec2011-06-22 13:11:56 +03003845{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003846 struct mgmt_mode *cp = data;
Johan Hedberg33e38b32013-03-15 17:07:05 -05003847 struct pending_cmd *cmd;
3848 struct hci_request req;
Antti Julkuf6422ec2011-06-22 13:11:56 +03003849 int err;
3850
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003851 BT_DBG("%s", hdev->name);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003852
Johan Hedberg56f87902013-10-02 13:43:13 +03003853 if (!test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags) ||
3854 hdev->hci_ver < BLUETOOTH_VER_1_2)
Johan Hedberg33c525c2012-10-24 21:11:58 +03003855 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3856 MGMT_STATUS_NOT_SUPPORTED);
3857
Johan Hedberga7e80f22013-01-09 16:05:19 +02003858 if (cp->val != 0x00 && cp->val != 0x01)
3859 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3860 MGMT_STATUS_INVALID_PARAMS);
3861
Johan Hedberg5400c042012-02-21 16:40:33 +02003862 if (!hdev_is_powered(hdev))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003863 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003864 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5400c042012-02-21 16:40:33 +02003865
3866 if (!test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003867 return cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003868 MGMT_STATUS_REJECTED);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003869
3870 hci_dev_lock(hdev);
3871
Johan Hedberg05cbf292013-03-15 17:07:07 -05003872 if (mgmt_pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev)) {
3873 err = cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
3874 MGMT_STATUS_BUSY);
3875 goto unlock;
3876 }
3877
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05003878 if (!!cp->val == test_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags)) {
3879 err = send_settings_rsp(sk, MGMT_OP_SET_FAST_CONNECTABLE,
3880 hdev);
3881 goto unlock;
3882 }
3883
Johan Hedberg33e38b32013-03-15 17:07:05 -05003884 cmd = mgmt_pending_add(sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev,
3885 data, len);
3886 if (!cmd) {
3887 err = -ENOMEM;
3888 goto unlock;
3889 }
3890
3891 hci_req_init(&req, hdev);
3892
Johan Hedberg406d7802013-03-15 17:07:09 -05003893 write_fast_connectable(&req, cp->val);
Johan Hedberg33e38b32013-03-15 17:07:05 -05003894
3895 err = hci_req_run(&req, fast_connectable_complete);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003896 if (err < 0) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003897 err = cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003898 MGMT_STATUS_FAILED);
Johan Hedberg33e38b32013-03-15 17:07:05 -05003899 mgmt_pending_remove(cmd);
Antti Julkuf6422ec2011-06-22 13:11:56 +03003900 }
3901
Johan Hedberg33e38b32013-03-15 17:07:05 -05003902unlock:
Antti Julkuf6422ec2011-06-22 13:11:56 +03003903 hci_dev_unlock(hdev);
Johan Hedberg33e38b32013-03-15 17:07:05 -05003904
Antti Julkuf6422ec2011-06-22 13:11:56 +03003905 return err;
3906}
3907
Johan Hedberg67e5a7a2013-10-14 21:15:25 +03003908static void set_bredr_scan(struct hci_request *req)
3909{
3910 struct hci_dev *hdev = req->hdev;
3911 u8 scan = 0;
3912
3913 /* Ensure that fast connectable is disabled. This function will
3914 * not do anything if the page scan parameters are already what
3915 * they should be.
3916 */
3917 write_fast_connectable(req, false);
3918
3919 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
3920 scan |= SCAN_PAGE;
3921 if (test_bit(HCI_DISCOVERABLE, &hdev->dev_flags))
3922 scan |= SCAN_INQUIRY;
3923
3924 if (scan)
3925 hci_req_add(req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
3926}
3927
Johan Hedberg0663ca22013-10-02 13:43:14 +03003928static void set_bredr_complete(struct hci_dev *hdev, u8 status)
3929{
3930 struct pending_cmd *cmd;
3931
3932 BT_DBG("status 0x%02x", status);
3933
3934 hci_dev_lock(hdev);
3935
3936 cmd = mgmt_pending_find(MGMT_OP_SET_BREDR, hdev);
3937 if (!cmd)
3938 goto unlock;
3939
3940 if (status) {
3941 u8 mgmt_err = mgmt_status(status);
3942
3943 /* We need to restore the flag if related HCI commands
3944 * failed.
3945 */
3946 clear_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
3947
3948 cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
3949 } else {
3950 send_settings_rsp(cmd->sk, MGMT_OP_SET_BREDR, hdev);
3951 new_settings(hdev, cmd->sk);
3952 }
3953
3954 mgmt_pending_remove(cmd);
3955
3956unlock:
3957 hci_dev_unlock(hdev);
3958}
3959
3960static int set_bredr(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
3961{
3962 struct mgmt_mode *cp = data;
3963 struct pending_cmd *cmd;
3964 struct hci_request req;
3965 int err;
3966
3967 BT_DBG("request for %s", hdev->name);
3968
3969 if (!lmp_bredr_capable(hdev) || !lmp_le_capable(hdev))
3970 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
3971 MGMT_STATUS_NOT_SUPPORTED);
3972
3973 if (!test_bit(HCI_LE_ENABLED, &hdev->dev_flags))
3974 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
3975 MGMT_STATUS_REJECTED);
3976
3977 if (cp->val != 0x00 && cp->val != 0x01)
3978 return cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
3979 MGMT_STATUS_INVALID_PARAMS);
3980
3981 hci_dev_lock(hdev);
3982
3983 if (cp->val == test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
3984 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
3985 goto unlock;
3986 }
3987
3988 if (!hdev_is_powered(hdev)) {
3989 if (!cp->val) {
Johan Hedberg0663ca22013-10-02 13:43:14 +03003990 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
3991 clear_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
3992 clear_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
3993 clear_bit(HCI_FAST_CONNECTABLE, &hdev->dev_flags);
3994 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
3995 }
3996
3997 change_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
3998
3999 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4000 if (err < 0)
4001 goto unlock;
4002
4003 err = new_settings(hdev, sk);
4004 goto unlock;
4005 }
4006
4007 /* Reject disabling when powered on */
4008 if (!cp->val) {
4009 err = cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4010 MGMT_STATUS_REJECTED);
4011 goto unlock;
4012 }
4013
4014 if (mgmt_pending_find(MGMT_OP_SET_BREDR, hdev)) {
4015 err = cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4016 MGMT_STATUS_BUSY);
4017 goto unlock;
4018 }
4019
4020 cmd = mgmt_pending_add(sk, MGMT_OP_SET_BREDR, hdev, data, len);
4021 if (!cmd) {
4022 err = -ENOMEM;
4023 goto unlock;
4024 }
4025
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004026 /* We need to flip the bit already here so that update_adv_data
Johan Hedberg0663ca22013-10-02 13:43:14 +03004027 * generates the correct flags.
4028 */
4029 set_bit(HCI_BREDR_ENABLED, &hdev->dev_flags);
4030
4031 hci_req_init(&req, hdev);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004032
4033 if (test_bit(HCI_CONNECTABLE, &hdev->dev_flags))
4034 set_bredr_scan(&req);
4035
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004036 /* Since only the advertising data flags will change, there
4037 * is no need to update the scan response data.
4038 */
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004039 update_adv_data(&req);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004040
Johan Hedberg0663ca22013-10-02 13:43:14 +03004041 err = hci_req_run(&req, set_bredr_complete);
4042 if (err < 0)
4043 mgmt_pending_remove(cmd);
4044
4045unlock:
4046 hci_dev_unlock(hdev);
4047 return err;
4048}
4049
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004050static int set_secure_conn(struct sock *sk, struct hci_dev *hdev,
4051 void *data, u16 len)
4052{
4053 struct mgmt_mode *cp = data;
4054 struct pending_cmd *cmd;
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004055 u8 val, status;
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004056 int err;
4057
4058 BT_DBG("request for %s", hdev->name);
4059
4060 status = mgmt_bredr_support(hdev);
4061 if (status)
4062 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4063 status);
4064
Marcel Holtmann5afeac12014-01-10 02:07:27 -08004065 if (!lmp_sc_capable(hdev) &&
4066 !test_bit(HCI_FORCE_SC, &hdev->dev_flags))
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004067 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4068 MGMT_STATUS_NOT_SUPPORTED);
4069
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004070 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004071 return cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4072 MGMT_STATUS_INVALID_PARAMS);
4073
4074 hci_dev_lock(hdev);
4075
4076 if (!hdev_is_powered(hdev)) {
4077 bool changed;
4078
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004079 if (cp->val) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004080 changed = !test_and_set_bit(HCI_SC_ENABLED,
4081 &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004082 if (cp->val == 0x02)
4083 set_bit(HCI_SC_ONLY, &hdev->dev_flags);
4084 else
4085 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4086 } else {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004087 changed = test_and_clear_bit(HCI_SC_ENABLED,
4088 &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004089 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4090 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004091
4092 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4093 if (err < 0)
4094 goto failed;
4095
4096 if (changed)
4097 err = new_settings(hdev, sk);
4098
4099 goto failed;
4100 }
4101
4102 if (mgmt_pending_find(MGMT_OP_SET_SECURE_CONN, hdev)) {
4103 err = cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4104 MGMT_STATUS_BUSY);
4105 goto failed;
4106 }
4107
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004108 val = !!cp->val;
4109
4110 if (val == test_bit(HCI_SC_ENABLED, &hdev->dev_flags) &&
4111 (cp->val == 0x02) == test_bit(HCI_SC_ONLY, &hdev->dev_flags)) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004112 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4113 goto failed;
4114 }
4115
4116 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SECURE_CONN, hdev, data, len);
4117 if (!cmd) {
4118 err = -ENOMEM;
4119 goto failed;
4120 }
4121
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004122 err = hci_send_cmd(hdev, HCI_OP_WRITE_SC_SUPPORT, 1, &val);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004123 if (err < 0) {
4124 mgmt_pending_remove(cmd);
4125 goto failed;
4126 }
4127
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004128 if (cp->val == 0x02)
4129 set_bit(HCI_SC_ONLY, &hdev->dev_flags);
4130 else
4131 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
4132
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004133failed:
4134 hci_dev_unlock(hdev);
4135 return err;
4136}
4137
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004138static int set_debug_keys(struct sock *sk, struct hci_dev *hdev,
4139 void *data, u16 len)
4140{
4141 struct mgmt_mode *cp = data;
4142 bool changed;
4143 int err;
4144
4145 BT_DBG("request for %s", hdev->name);
4146
4147 if (cp->val != 0x00 && cp->val != 0x01)
4148 return cmd_status(sk, hdev->id, MGMT_OP_SET_DEBUG_KEYS,
4149 MGMT_STATUS_INVALID_PARAMS);
4150
4151 hci_dev_lock(hdev);
4152
4153 if (cp->val)
4154 changed = !test_and_set_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
4155 else
4156 changed = test_and_clear_bit(HCI_DEBUG_KEYS, &hdev->dev_flags);
4157
4158 err = send_settings_rsp(sk, MGMT_OP_SET_DEBUG_KEYS, hdev);
4159 if (err < 0)
4160 goto unlock;
4161
4162 if (changed)
4163 err = new_settings(hdev, sk);
4164
4165unlock:
4166 hci_dev_unlock(hdev);
4167 return err;
4168}
4169
Johan Hedberg41edf162014-02-18 10:19:35 +02004170static bool irk_is_valid(struct mgmt_irk_info *irk)
4171{
4172 switch (irk->addr.type) {
4173 case BDADDR_LE_PUBLIC:
4174 return true;
4175
4176 case BDADDR_LE_RANDOM:
4177 /* Two most significant bits shall be set */
4178 if ((irk->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4179 return false;
4180 return true;
4181 }
4182
4183 return false;
4184}
4185
4186static int load_irks(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4187 u16 len)
4188{
4189 struct mgmt_cp_load_irks *cp = cp_data;
4190 u16 irk_count, expected_len;
4191 int i, err;
4192
4193 BT_DBG("request for %s", hdev->name);
4194
4195 if (!lmp_le_capable(hdev))
4196 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4197 MGMT_STATUS_NOT_SUPPORTED);
4198
4199 irk_count = __le16_to_cpu(cp->irk_count);
4200
4201 expected_len = sizeof(*cp) + irk_count * sizeof(struct mgmt_irk_info);
4202 if (expected_len != len) {
4203 BT_ERR("load_irks: expected %u bytes, got %u bytes",
4204 len, expected_len);
4205 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4206 MGMT_STATUS_INVALID_PARAMS);
4207 }
4208
4209 BT_DBG("%s irk_count %u", hdev->name, irk_count);
4210
4211 for (i = 0; i < irk_count; i++) {
4212 struct mgmt_irk_info *key = &cp->irks[i];
4213
4214 if (!irk_is_valid(key))
4215 return cmd_status(sk, hdev->id,
4216 MGMT_OP_LOAD_IRKS,
4217 MGMT_STATUS_INVALID_PARAMS);
4218 }
4219
4220 hci_dev_lock(hdev);
4221
4222 hci_smp_irks_clear(hdev);
4223
4224 for (i = 0; i < irk_count; i++) {
4225 struct mgmt_irk_info *irk = &cp->irks[i];
4226 u8 addr_type;
4227
4228 if (irk->addr.type == BDADDR_LE_PUBLIC)
4229 addr_type = ADDR_LE_DEV_PUBLIC;
4230 else
4231 addr_type = ADDR_LE_DEV_RANDOM;
4232
4233 hci_add_irk(hdev, &irk->addr.bdaddr, addr_type, irk->val,
4234 BDADDR_ANY);
4235 }
4236
4237 set_bit(HCI_RPA_RESOLVING, &hdev->dev_flags);
4238
4239 err = cmd_complete(sk, hdev->id, MGMT_OP_LOAD_IRKS, 0, NULL, 0);
4240
4241 hci_dev_unlock(hdev);
4242
4243 return err;
4244}
4245
Johan Hedberg3f706b72013-01-20 14:27:16 +02004246static bool ltk_is_valid(struct mgmt_ltk_info *key)
4247{
4248 if (key->master != 0x00 && key->master != 0x01)
4249 return false;
Marcel Holtmann490cb0b2014-02-16 12:59:05 -08004250
4251 switch (key->addr.type) {
4252 case BDADDR_LE_PUBLIC:
4253 return true;
4254
4255 case BDADDR_LE_RANDOM:
4256 /* Two most significant bits shall be set */
4257 if ((key->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4258 return false;
4259 return true;
4260 }
4261
4262 return false;
Johan Hedberg3f706b72013-01-20 14:27:16 +02004263}
4264
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004265static int load_long_term_keys(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004266 void *cp_data, u16 len)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004267{
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004268 struct mgmt_cp_load_long_term_keys *cp = cp_data;
4269 u16 key_count, expected_len;
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004270 int i, err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004271
Marcel Holtmanncf99ba12013-10-02 21:16:08 -07004272 BT_DBG("request for %s", hdev->name);
4273
4274 if (!lmp_le_capable(hdev))
4275 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4276 MGMT_STATUS_NOT_SUPPORTED);
4277
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004278 key_count = __le16_to_cpu(cp->key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004279
4280 expected_len = sizeof(*cp) + key_count *
4281 sizeof(struct mgmt_ltk_info);
4282 if (expected_len != len) {
4283 BT_ERR("load_keys: expected %u bytes, got %u bytes",
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004284 len, expected_len);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004285 return cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
Johan Hedberge57e6192013-01-20 14:27:14 +02004286 MGMT_STATUS_INVALID_PARAMS);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004287 }
4288
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004289 BT_DBG("%s key_count %u", hdev->name, key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004290
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004291 for (i = 0; i < key_count; i++) {
4292 struct mgmt_ltk_info *key = &cp->keys[i];
4293
Johan Hedberg3f706b72013-01-20 14:27:16 +02004294 if (!ltk_is_valid(key))
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004295 return cmd_status(sk, hdev->id,
4296 MGMT_OP_LOAD_LONG_TERM_KEYS,
4297 MGMT_STATUS_INVALID_PARAMS);
4298 }
4299
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004300 hci_dev_lock(hdev);
4301
4302 hci_smp_ltks_clear(hdev);
4303
4304 for (i = 0; i < key_count; i++) {
4305 struct mgmt_ltk_info *key = &cp->keys[i];
Marcel Holtmann79d95a12013-10-13 03:57:38 -07004306 u8 type, addr_type;
4307
4308 if (key->addr.type == BDADDR_LE_PUBLIC)
4309 addr_type = ADDR_LE_DEV_PUBLIC;
4310 else
4311 addr_type = ADDR_LE_DEV_RANDOM;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004312
4313 if (key->master)
4314 type = HCI_SMP_LTK;
4315 else
4316 type = HCI_SMP_LTK_SLAVE;
4317
Marcel Holtmann79d95a12013-10-13 03:57:38 -07004318 hci_add_ltk(hdev, &key->addr.bdaddr, addr_type,
Marcel Holtmannd40f3ee2014-01-31 18:42:17 -08004319 type, 0, key->type, key->val,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004320 key->enc_size, key->ediv, key->rand);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004321 }
4322
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004323 err = cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS, 0,
4324 NULL, 0);
4325
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004326 hci_dev_unlock(hdev);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004327
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004328 return err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004329}
4330
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +02004331static const struct mgmt_handler {
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004332 int (*func) (struct sock *sk, struct hci_dev *hdev, void *data,
4333 u16 data_len);
Johan Hedbergbe22b542012-03-01 22:24:41 +02004334 bool var_len;
4335 size_t data_len;
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004336} mgmt_handlers[] = {
4337 { NULL }, /* 0x0000 (no command) */
Johan Hedbergbe22b542012-03-01 22:24:41 +02004338 { read_version, false, MGMT_READ_VERSION_SIZE },
4339 { read_commands, false, MGMT_READ_COMMANDS_SIZE },
4340 { read_index_list, false, MGMT_READ_INDEX_LIST_SIZE },
4341 { read_controller_info, false, MGMT_READ_INFO_SIZE },
4342 { set_powered, false, MGMT_SETTING_SIZE },
4343 { set_discoverable, false, MGMT_SET_DISCOVERABLE_SIZE },
4344 { set_connectable, false, MGMT_SETTING_SIZE },
4345 { set_fast_connectable, false, MGMT_SETTING_SIZE },
4346 { set_pairable, false, MGMT_SETTING_SIZE },
4347 { set_link_security, false, MGMT_SETTING_SIZE },
4348 { set_ssp, false, MGMT_SETTING_SIZE },
4349 { set_hs, false, MGMT_SETTING_SIZE },
4350 { set_le, false, MGMT_SETTING_SIZE },
4351 { set_dev_class, false, MGMT_SET_DEV_CLASS_SIZE },
4352 { set_local_name, false, MGMT_SET_LOCAL_NAME_SIZE },
4353 { add_uuid, false, MGMT_ADD_UUID_SIZE },
4354 { remove_uuid, false, MGMT_REMOVE_UUID_SIZE },
4355 { load_link_keys, true, MGMT_LOAD_LINK_KEYS_SIZE },
4356 { load_long_term_keys, true, MGMT_LOAD_LONG_TERM_KEYS_SIZE },
4357 { disconnect, false, MGMT_DISCONNECT_SIZE },
4358 { get_connections, false, MGMT_GET_CONNECTIONS_SIZE },
4359 { pin_code_reply, false, MGMT_PIN_CODE_REPLY_SIZE },
4360 { pin_code_neg_reply, false, MGMT_PIN_CODE_NEG_REPLY_SIZE },
4361 { set_io_capability, false, MGMT_SET_IO_CAPABILITY_SIZE },
4362 { pair_device, false, MGMT_PAIR_DEVICE_SIZE },
4363 { cancel_pair_device, false, MGMT_CANCEL_PAIR_DEVICE_SIZE },
4364 { unpair_device, false, MGMT_UNPAIR_DEVICE_SIZE },
4365 { user_confirm_reply, false, MGMT_USER_CONFIRM_REPLY_SIZE },
4366 { user_confirm_neg_reply, false, MGMT_USER_CONFIRM_NEG_REPLY_SIZE },
4367 { user_passkey_reply, false, MGMT_USER_PASSKEY_REPLY_SIZE },
4368 { user_passkey_neg_reply, false, MGMT_USER_PASSKEY_NEG_REPLY_SIZE },
4369 { read_local_oob_data, false, MGMT_READ_LOCAL_OOB_DATA_SIZE },
Marcel Holtmannec109112014-01-10 02:07:30 -08004370 { add_remote_oob_data, true, MGMT_ADD_REMOTE_OOB_DATA_SIZE },
Johan Hedbergbe22b542012-03-01 22:24:41 +02004371 { remove_remote_oob_data, false, MGMT_REMOVE_REMOTE_OOB_DATA_SIZE },
4372 { start_discovery, false, MGMT_START_DISCOVERY_SIZE },
4373 { stop_discovery, false, MGMT_STOP_DISCOVERY_SIZE },
4374 { confirm_name, false, MGMT_CONFIRM_NAME_SIZE },
4375 { block_device, false, MGMT_BLOCK_DEVICE_SIZE },
4376 { unblock_device, false, MGMT_UNBLOCK_DEVICE_SIZE },
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07004377 { set_device_id, false, MGMT_SET_DEVICE_ID_SIZE },
Johan Hedberg4375f102013-09-25 13:26:10 +03004378 { set_advertising, false, MGMT_SETTING_SIZE },
Johan Hedberg0663ca22013-10-02 13:43:14 +03004379 { set_bredr, false, MGMT_SETTING_SIZE },
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004380 { set_static_address, false, MGMT_SET_STATIC_ADDRESS_SIZE },
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004381 { set_scan_params, false, MGMT_SET_SCAN_PARAMS_SIZE },
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004382 { set_secure_conn, false, MGMT_SETTING_SIZE },
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004383 { set_debug_keys, false, MGMT_SETTING_SIZE },
Johan Hedberg41edf162014-02-18 10:19:35 +02004384 { },
4385 { load_irks, true, MGMT_LOAD_IRKS_SIZE },
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004386};
4387
4388
Johan Hedberg03811012010-12-08 00:21:06 +02004389int mgmt_control(struct sock *sk, struct msghdr *msg, size_t msglen)
4390{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004391 void *buf;
4392 u8 *cp;
Johan Hedberg03811012010-12-08 00:21:06 +02004393 struct mgmt_hdr *hdr;
Szymon Janc4e51eae2011-02-25 19:05:48 +01004394 u16 opcode, index, len;
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004395 struct hci_dev *hdev = NULL;
Andrei Emeltchenko2e3c35e2012-03-14 18:54:15 +02004396 const struct mgmt_handler *handler;
Johan Hedberg03811012010-12-08 00:21:06 +02004397 int err;
4398
4399 BT_DBG("got %zu bytes", msglen);
4400
4401 if (msglen < sizeof(*hdr))
4402 return -EINVAL;
4403
Gustavo F. Padovane63a15e2011-04-04 18:56:53 -03004404 buf = kmalloc(msglen, GFP_KERNEL);
Johan Hedberg03811012010-12-08 00:21:06 +02004405 if (!buf)
4406 return -ENOMEM;
4407
4408 if (memcpy_fromiovec(buf, msg->msg_iov, msglen)) {
4409 err = -EFAULT;
4410 goto done;
4411 }
4412
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004413 hdr = buf;
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004414 opcode = __le16_to_cpu(hdr->opcode);
4415 index = __le16_to_cpu(hdr->index);
4416 len = __le16_to_cpu(hdr->len);
Johan Hedberg03811012010-12-08 00:21:06 +02004417
4418 if (len != msglen - sizeof(*hdr)) {
4419 err = -EINVAL;
4420 goto done;
4421 }
4422
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004423 if (index != MGMT_INDEX_NONE) {
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004424 hdev = hci_dev_get(index);
4425 if (!hdev) {
4426 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004427 MGMT_STATUS_INVALID_INDEX);
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004428 goto done;
4429 }
Marcel Holtmann0736cfa2013-08-26 21:40:51 -07004430
Johan Hedbergcebf4cf2013-10-10 18:06:04 +02004431 if (test_bit(HCI_SETUP, &hdev->dev_flags) ||
4432 test_bit(HCI_USER_CHANNEL, &hdev->dev_flags)) {
Marcel Holtmann0736cfa2013-08-26 21:40:51 -07004433 err = cmd_status(sk, index, opcode,
4434 MGMT_STATUS_INVALID_INDEX);
4435 goto done;
4436 }
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004437 }
4438
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004439 if (opcode >= ARRAY_SIZE(mgmt_handlers) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004440 mgmt_handlers[opcode].func == NULL) {
Johan Hedberg03811012010-12-08 00:21:06 +02004441 BT_DBG("Unknown op %u", opcode);
Johan Hedbergca69b792011-11-11 18:10:00 +02004442 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004443 MGMT_STATUS_UNKNOWN_COMMAND);
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004444 goto done;
Johan Hedberg03811012010-12-08 00:21:06 +02004445 }
4446
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004447 if ((hdev && opcode < MGMT_OP_READ_INFO) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004448 (!hdev && opcode >= MGMT_OP_READ_INFO)) {
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004449 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004450 MGMT_STATUS_INVALID_INDEX);
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004451 goto done;
4452 }
4453
Johan Hedbergbe22b542012-03-01 22:24:41 +02004454 handler = &mgmt_handlers[opcode];
4455
4456 if ((handler->var_len && len < handler->data_len) ||
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004457 (!handler->var_len && len != handler->data_len)) {
Johan Hedbergbe22b542012-03-01 22:24:41 +02004458 err = cmd_status(sk, index, opcode,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004459 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergbe22b542012-03-01 22:24:41 +02004460 goto done;
4461 }
4462
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02004463 if (hdev)
4464 mgmt_init_hdev(sk, hdev);
4465
4466 cp = buf + sizeof(*hdr);
4467
Johan Hedbergbe22b542012-03-01 22:24:41 +02004468 err = handler->func(sk, hdev, cp, len);
Johan Hedberge41d8b42010-12-13 21:07:03 +02004469 if (err < 0)
4470 goto done;
4471
Johan Hedberg03811012010-12-08 00:21:06 +02004472 err = msglen;
4473
4474done:
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004475 if (hdev)
4476 hci_dev_put(hdev);
4477
Johan Hedberg03811012010-12-08 00:21:06 +02004478 kfree(buf);
4479 return err;
4480}
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004481
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004482void mgmt_index_added(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004483{
Marcel Holtmann1514b892013-10-06 08:25:01 -07004484 if (hdev->dev_type != HCI_BREDR)
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004485 return;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03004486
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004487 mgmt_event(MGMT_EV_INDEX_ADDED, hdev, NULL, 0, NULL);
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004488}
4489
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004490void mgmt_index_removed(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02004491{
Johan Hedberg5f159032012-03-02 03:13:19 +02004492 u8 status = MGMT_STATUS_INVALID_INDEX;
Johan Hedbergb24752f2011-11-03 14:40:33 +02004493
Marcel Holtmann1514b892013-10-06 08:25:01 -07004494 if (hdev->dev_type != HCI_BREDR)
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004495 return;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03004496
Johan Hedberg744cf192011-11-08 20:40:14 +02004497 mgmt_pending_foreach(0, hdev, cmd_status_rsp, &status);
Johan Hedbergb24752f2011-11-03 14:40:33 +02004498
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07004499 mgmt_event(MGMT_EV_INDEX_REMOVED, hdev, NULL, 0, NULL);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02004500}
4501
Johan Hedberg229ab392013-03-15 17:06:53 -05004502static void powered_complete(struct hci_dev *hdev, u8 status)
4503{
4504 struct cmd_lookup match = { NULL, hdev };
4505
4506 BT_DBG("status 0x%02x", status);
4507
4508 hci_dev_lock(hdev);
4509
4510 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
4511
4512 new_settings(hdev, match.sk);
4513
4514 hci_dev_unlock(hdev);
4515
4516 if (match.sk)
4517 sock_put(match.sk);
4518}
4519
Johan Hedberg70da6242013-03-15 17:06:51 -05004520static int powered_update_hci(struct hci_dev *hdev)
4521{
Johan Hedberg890ea892013-03-15 17:06:52 -05004522 struct hci_request req;
Johan Hedberg70da6242013-03-15 17:06:51 -05004523 u8 link_sec;
4524
Johan Hedberg890ea892013-03-15 17:06:52 -05004525 hci_req_init(&req, hdev);
4526
Johan Hedberg70da6242013-03-15 17:06:51 -05004527 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags) &&
4528 !lmp_host_ssp_capable(hdev)) {
4529 u8 ssp = 1;
4530
Johan Hedberg890ea892013-03-15 17:06:52 -05004531 hci_req_add(&req, HCI_OP_WRITE_SSP_MODE, 1, &ssp);
Johan Hedberg70da6242013-03-15 17:06:51 -05004532 }
4533
Johan Hedbergc73eee92013-04-19 18:35:21 +03004534 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags) &&
4535 lmp_bredr_capable(hdev)) {
Johan Hedberg70da6242013-03-15 17:06:51 -05004536 struct hci_cp_write_le_host_supported cp;
4537
4538 cp.le = 1;
4539 cp.simul = lmp_le_br_capable(hdev);
4540
4541 /* Check first if we already have the right
4542 * host state (host features set)
4543 */
4544 if (cp.le != lmp_host_le_capable(hdev) ||
4545 cp.simul != lmp_host_le_br_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05004546 hci_req_add(&req, HCI_OP_WRITE_LE_HOST_SUPPORTED,
4547 sizeof(cp), &cp);
Johan Hedberg70da6242013-03-15 17:06:51 -05004548 }
4549
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004550 if (lmp_le_capable(hdev)) {
4551 /* Set random address to static address if configured */
4552 if (bacmp(&hdev->static_addr, BDADDR_ANY))
4553 hci_req_add(&req, HCI_OP_LE_SET_RANDOM_ADDR, 6,
4554 &hdev->static_addr);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004555
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07004556 /* Make sure the controller has a good default for
4557 * advertising data. This also applies to the case
4558 * where BR/EDR was toggled during the AUTO_OFF phase.
4559 */
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004560 if (test_bit(HCI_LE_ENABLED, &hdev->dev_flags)) {
Marcel Holtmann5947f4b2013-10-16 00:16:50 -07004561 update_adv_data(&req);
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004562 update_scan_rsp_data(&req);
4563 }
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07004564
Marcel Holtmannbba3aa52013-10-06 02:55:21 -07004565 if (test_bit(HCI_ADVERTISING, &hdev->dev_flags))
4566 enable_advertising(&req);
Johan Hedbergeeca6f82013-09-25 13:26:09 +03004567 }
4568
Johan Hedberg70da6242013-03-15 17:06:51 -05004569 link_sec = test_bit(HCI_LINK_SECURITY, &hdev->dev_flags);
4570 if (link_sec != test_bit(HCI_AUTH, &hdev->flags))
Johan Hedberg890ea892013-03-15 17:06:52 -05004571 hci_req_add(&req, HCI_OP_WRITE_AUTH_ENABLE,
4572 sizeof(link_sec), &link_sec);
Johan Hedberg70da6242013-03-15 17:06:51 -05004573
4574 if (lmp_bredr_capable(hdev)) {
Johan Hedberg56f87902013-10-02 13:43:13 +03004575 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags))
4576 set_bredr_scan(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05004577 update_class(&req);
Johan Hedberg13928972013-03-15 17:07:00 -05004578 update_name(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05004579 update_eir(&req);
Johan Hedberg70da6242013-03-15 17:06:51 -05004580 }
4581
Johan Hedberg229ab392013-03-15 17:06:53 -05004582 return hci_req_run(&req, powered_complete);
Johan Hedberg70da6242013-03-15 17:06:51 -05004583}
4584
Johan Hedberg744cf192011-11-08 20:40:14 +02004585int mgmt_powered(struct hci_dev *hdev, u8 powered)
Johan Hedberg5add6af2010-12-16 10:00:37 +02004586{
Johan Hedberg76a7f3a2012-02-17 00:34:40 +02004587 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg229ab392013-03-15 17:06:53 -05004588 u8 status_not_powered = MGMT_STATUS_NOT_POWERED;
4589 u8 zero_cod[] = { 0, 0, 0 };
Johan Hedberg7bb895d2012-02-17 01:20:00 +02004590 int err;
Johan Hedberg5add6af2010-12-16 10:00:37 +02004591
Johan Hedberg5e5282b2012-02-21 16:01:30 +02004592 if (!test_bit(HCI_MGMT, &hdev->dev_flags))
4593 return 0;
4594
Johan Hedberg5e5282b2012-02-21 16:01:30 +02004595 if (powered) {
Johan Hedberg229ab392013-03-15 17:06:53 -05004596 if (powered_update_hci(hdev) == 0)
4597 return 0;
Johan Hedbergfe038882013-01-16 16:15:34 +02004598
Johan Hedberg229ab392013-03-15 17:06:53 -05004599 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp,
4600 &match);
4601 goto new_settings;
Johan Hedbergb24752f2011-11-03 14:40:33 +02004602 }
4603
Johan Hedberg229ab392013-03-15 17:06:53 -05004604 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
4605 mgmt_pending_foreach(0, hdev, cmd_status_rsp, &status_not_powered);
4606
4607 if (memcmp(hdev->dev_class, zero_cod, sizeof(zero_cod)) != 0)
4608 mgmt_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev,
4609 zero_cod, sizeof(zero_cod), NULL);
4610
4611new_settings:
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02004612 err = new_settings(hdev, match.sk);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02004613
4614 if (match.sk)
4615 sock_put(match.sk);
4616
Johan Hedberg7bb895d2012-02-17 01:20:00 +02004617 return err;
Johan Hedberg5add6af2010-12-16 10:00:37 +02004618}
Johan Hedberg73f22f62010-12-29 16:00:25 +02004619
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004620void mgmt_set_powered_failed(struct hci_dev *hdev, int err)
Johan Hedberg96570ff2013-05-29 09:51:29 +03004621{
4622 struct pending_cmd *cmd;
4623 u8 status;
4624
4625 cmd = mgmt_pending_find(MGMT_OP_SET_POWERED, hdev);
4626 if (!cmd)
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004627 return;
Johan Hedberg96570ff2013-05-29 09:51:29 +03004628
4629 if (err == -ERFKILL)
4630 status = MGMT_STATUS_RFKILLED;
4631 else
4632 status = MGMT_STATUS_FAILED;
4633
Marcel Holtmann3eec7052013-10-06 23:55:46 -07004634 cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_POWERED, status);
Johan Hedberg96570ff2013-05-29 09:51:29 +03004635
4636 mgmt_pending_remove(cmd);
Johan Hedberg96570ff2013-05-29 09:51:29 +03004637}
4638
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004639void mgmt_discoverable_timeout(struct hci_dev *hdev)
4640{
4641 struct hci_request req;
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004642
4643 hci_dev_lock(hdev);
4644
4645 /* When discoverable timeout triggers, then just make sure
4646 * the limited discoverable flag is cleared. Even in the case
4647 * of a timeout triggered from general discoverable, it is
4648 * safe to unconditionally clear the flag.
4649 */
4650 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004651 clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004652
4653 hci_req_init(&req, hdev);
Johan Hedberg4b580612013-10-19 23:38:21 +03004654 if (test_bit(HCI_BREDR_ENABLED, &hdev->dev_flags)) {
4655 u8 scan = SCAN_PAGE;
4656 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE,
4657 sizeof(scan), &scan);
4658 }
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004659 update_class(&req);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004660 update_adv_data(&req);
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004661 hci_req_run(&req, NULL);
4662
4663 hdev->discov_timeout = 0;
4664
Johan Hedberg9a43e252013-10-20 19:00:07 +03004665 new_settings(hdev, NULL);
4666
Marcel Holtmannd1967ff2013-10-15 10:57:40 -07004667 hci_dev_unlock(hdev);
4668}
4669
Marcel Holtmann86a75642013-10-15 06:33:54 -07004670void mgmt_discoverable(struct hci_dev *hdev, u8 discoverable)
Johan Hedberg73f22f62010-12-29 16:00:25 +02004671{
Marcel Holtmann86a75642013-10-15 06:33:54 -07004672 bool changed;
Johan Hedberg73f22f62010-12-29 16:00:25 +02004673
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03004674 /* Nothing needed here if there's a pending command since that
4675 * commands request completion callback takes care of everything
4676 * necessary.
4677 */
4678 if (mgmt_pending_find(MGMT_OP_SET_DISCOVERABLE, hdev))
Marcel Holtmann86a75642013-10-15 06:33:54 -07004679 return;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03004680
Johan Hedberg9a43e252013-10-20 19:00:07 +03004681 if (discoverable) {
Marcel Holtmann86a75642013-10-15 06:33:54 -07004682 changed = !test_and_set_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004683 } else {
4684 clear_bit(HCI_LIMITED_DISCOVERABLE, &hdev->dev_flags);
Marcel Holtmann86a75642013-10-15 06:33:54 -07004685 changed = test_and_clear_bit(HCI_DISCOVERABLE, &hdev->dev_flags);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004686 }
Johan Hedberg73f22f62010-12-29 16:00:25 +02004687
Johan Hedberg9a43e252013-10-20 19:00:07 +03004688 if (changed) {
4689 struct hci_request req;
4690
4691 /* In case this change in discoverable was triggered by
4692 * a disabling of connectable there could be a need to
4693 * update the advertising flags.
4694 */
4695 hci_req_init(&req, hdev);
4696 update_adv_data(&req);
4697 hci_req_run(&req, NULL);
4698
Marcel Holtmann86a75642013-10-15 06:33:54 -07004699 new_settings(hdev, NULL);
Johan Hedberg9a43e252013-10-20 19:00:07 +03004700 }
Johan Hedberg73f22f62010-12-29 16:00:25 +02004701}
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004702
Marcel Holtmanna3309162013-10-15 06:33:55 -07004703void mgmt_connectable(struct hci_dev *hdev, u8 connectable)
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004704{
Marcel Holtmanna3309162013-10-15 06:33:55 -07004705 bool changed;
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004706
Johan Hedbergd7b856f2013-10-14 16:20:04 +03004707 /* Nothing needed here if there's a pending command since that
4708 * commands request completion callback takes care of everything
4709 * necessary.
4710 */
4711 if (mgmt_pending_find(MGMT_OP_SET_CONNECTABLE, hdev))
Marcel Holtmanna3309162013-10-15 06:33:55 -07004712 return;
Johan Hedbergd7b856f2013-10-14 16:20:04 +03004713
Marcel Holtmanna3309162013-10-15 06:33:55 -07004714 if (connectable)
4715 changed = !test_and_set_bit(HCI_CONNECTABLE, &hdev->dev_flags);
4716 else
4717 changed = test_and_clear_bit(HCI_CONNECTABLE, &hdev->dev_flags);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004718
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02004719 if (changed)
Marcel Holtmanna3309162013-10-15 06:33:55 -07004720 new_settings(hdev, NULL);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02004721}
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02004722
Marcel Holtmann4796e8a2013-10-15 06:33:56 -07004723void mgmt_write_scan_failed(struct hci_dev *hdev, u8 scan, u8 status)
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004724{
Johan Hedbergca69b792011-11-11 18:10:00 +02004725 u8 mgmt_err = mgmt_status(status);
4726
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004727 if (scan & SCAN_PAGE)
Johan Hedberg744cf192011-11-08 20:40:14 +02004728 mgmt_pending_foreach(MGMT_OP_SET_CONNECTABLE, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004729 cmd_status_rsp, &mgmt_err);
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004730
4731 if (scan & SCAN_INQUIRY)
Johan Hedberg744cf192011-11-08 20:40:14 +02004732 mgmt_pending_foreach(MGMT_OP_SET_DISCOVERABLE, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004733 cmd_status_rsp, &mgmt_err);
Johan Hedberg2d7cee52011-11-07 22:16:03 +02004734}
4735
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07004736void mgmt_new_link_key(struct hci_dev *hdev, struct link_key *key,
4737 bool persistent)
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02004738{
Johan Hedberg86742e12011-11-07 23:13:38 +02004739 struct mgmt_ev_new_link_key ev;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02004740
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004741 memset(&ev, 0, sizeof(ev));
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02004742
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004743 ev.store_hint = persistent;
Johan Hedbergd753fdc2012-02-17 14:06:34 +02004744 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004745 ev.key.addr.type = BDADDR_BREDR;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004746 ev.key.type = key->type;
Andrei Emeltchenko9b3b4462012-05-23 11:31:20 +03004747 memcpy(ev.key.val, key->val, HCI_LINK_KEY_SIZE);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03004748 ev.key.pin_len = key->pin_len;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02004749
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07004750 mgmt_event(MGMT_EV_NEW_LINK_KEY, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02004751}
Johan Hedbergf7520542011-01-20 12:34:39 +02004752
Marcel Holtmann083368f2013-10-15 14:26:29 -07004753void mgmt_new_ltk(struct hci_dev *hdev, struct smp_ltk *key, u8 persistent)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004754{
4755 struct mgmt_ev_new_long_term_key ev;
4756
4757 memset(&ev, 0, sizeof(ev));
4758
4759 ev.store_hint = persistent;
4760 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004761 ev.key.addr.type = link_to_bdaddr(LE_LINK, key->bdaddr_type);
Marcel Holtmannd40f3ee2014-01-31 18:42:17 -08004762 ev.key.type = key->authenticated;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004763 ev.key.enc_size = key->enc_size;
4764 ev.key.ediv = key->ediv;
4765
4766 if (key->type == HCI_SMP_LTK)
4767 ev.key.master = 1;
4768
4769 memcpy(ev.key.rand, key->rand, sizeof(key->rand));
4770 memcpy(ev.key.val, key->val, sizeof(key->val));
4771
Marcel Holtmann083368f2013-10-15 14:26:29 -07004772 mgmt_event(MGMT_EV_NEW_LONG_TERM_KEY, hdev, &ev, sizeof(ev), NULL);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004773}
4774
Marcel Holtmann94933992013-10-15 10:26:39 -07004775static inline u16 eir_append_data(u8 *eir, u16 eir_len, u8 type, u8 *data,
4776 u8 data_len)
4777{
4778 eir[eir_len++] = sizeof(type) + data_len;
4779 eir[eir_len++] = type;
4780 memcpy(&eir[eir_len], data, data_len);
4781 eir_len += data_len;
4782
4783 return eir_len;
4784}
4785
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07004786void mgmt_device_connected(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
4787 u8 addr_type, u32 flags, u8 *name, u8 name_len,
4788 u8 *dev_class)
Johan Hedbergf7520542011-01-20 12:34:39 +02004789{
Johan Hedbergb644ba32012-01-17 21:48:47 +02004790 char buf[512];
4791 struct mgmt_ev_device_connected *ev = (void *) buf;
4792 u16 eir_len = 0;
Johan Hedbergf7520542011-01-20 12:34:39 +02004793
Johan Hedbergb644ba32012-01-17 21:48:47 +02004794 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004795 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergf7520542011-01-20 12:34:39 +02004796
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02004797 ev->flags = __cpu_to_le32(flags);
Johan Hedberg08c79b62012-02-23 22:31:51 +02004798
Johan Hedbergb644ba32012-01-17 21:48:47 +02004799 if (name_len > 0)
4800 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004801 name, name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02004802
4803 if (dev_class && memcmp(dev_class, "\0\0\0", 3) != 0)
Brian Gix53156382012-03-09 14:07:03 -08004804 eir_len = eir_append_data(ev->eir, eir_len,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004805 EIR_CLASS_OF_DEV, dev_class, 3);
Johan Hedbergb644ba32012-01-17 21:48:47 +02004806
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02004807 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02004808
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07004809 mgmt_event(MGMT_EV_DEVICE_CONNECTED, hdev, buf,
4810 sizeof(*ev) + eir_len, NULL);
Johan Hedbergf7520542011-01-20 12:34:39 +02004811}
4812
Johan Hedberg8962ee72011-01-20 12:40:27 +02004813static void disconnect_rsp(struct pending_cmd *cmd, void *data)
4814{
Szymon Jancc68fb7f2011-03-22 13:12:19 +01004815 struct mgmt_cp_disconnect *cp = cmd->param;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004816 struct sock **sk = data;
Johan Hedberga38528f2011-01-22 06:46:43 +02004817 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004818
Johan Hedberg88c3df12012-02-09 14:27:38 +02004819 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
4820 rp.addr.type = cp->addr.type;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004821
Johan Hedbergaee9b212012-02-18 15:07:59 +02004822 cmd_complete(cmd->sk, cmd->index, MGMT_OP_DISCONNECT, 0, &rp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004823 sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02004824
4825 *sk = cmd->sk;
4826 sock_hold(*sk);
4827
Johan Hedberga664b5b2011-02-19 12:06:02 -03004828 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004829}
4830
Johan Hedberg124f6e32012-02-09 13:50:12 +02004831static void unpair_device_rsp(struct pending_cmd *cmd, void *data)
Johan Hedberga8a1d192011-11-10 15:54:38 +02004832{
Johan Hedbergb1078ad2012-02-09 17:21:16 +02004833 struct hci_dev *hdev = data;
Johan Hedberg124f6e32012-02-09 13:50:12 +02004834 struct mgmt_cp_unpair_device *cp = cmd->param;
4835 struct mgmt_rp_unpair_device rp;
Johan Hedberga8a1d192011-11-10 15:54:38 +02004836
4837 memset(&rp, 0, sizeof(rp));
Johan Hedberg124f6e32012-02-09 13:50:12 +02004838 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
4839 rp.addr.type = cp->addr.type;
Johan Hedberga8a1d192011-11-10 15:54:38 +02004840
Johan Hedbergb1078ad2012-02-09 17:21:16 +02004841 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, cmd->sk);
4842
Johan Hedbergaee9b212012-02-18 15:07:59 +02004843 cmd_complete(cmd->sk, cmd->index, cmd->opcode, 0, &rp, sizeof(rp));
Johan Hedberga8a1d192011-11-10 15:54:38 +02004844
4845 mgmt_pending_remove(cmd);
4846}
4847
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07004848void mgmt_device_disconnected(struct hci_dev *hdev, bdaddr_t *bdaddr,
4849 u8 link_type, u8 addr_type, u8 reason)
Johan Hedbergf7520542011-01-20 12:34:39 +02004850{
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02004851 struct mgmt_ev_device_disconnected ev;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004852 struct sock *sk = NULL;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004853
Andre Guedes57eb7762013-10-30 19:01:41 -03004854 if (link_type != ACL_LINK && link_type != LE_LINK)
4855 return;
4856
Johan Hedberg744cf192011-11-08 20:40:14 +02004857 mgmt_pending_foreach(MGMT_OP_DISCONNECT, hdev, disconnect_rsp, &sk);
Johan Hedbergf7520542011-01-20 12:34:39 +02004858
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02004859 bacpy(&ev.addr.bdaddr, bdaddr);
4860 ev.addr.type = link_to_bdaddr(link_type, addr_type);
4861 ev.reason = reason;
Johan Hedbergf7520542011-01-20 12:34:39 +02004862
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07004863 mgmt_event(MGMT_EV_DEVICE_DISCONNECTED, hdev, &ev, sizeof(ev), sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004864
4865 if (sk)
Szymon Jancd97dcb62012-03-16 16:02:56 +01004866 sock_put(sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004867
Johan Hedberg124f6e32012-02-09 13:50:12 +02004868 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004869 hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004870}
4871
Marcel Holtmann78929242013-10-06 23:55:47 -07004872void mgmt_disconnect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr,
4873 u8 link_type, u8 addr_type, u8 status)
Johan Hedberg8962ee72011-01-20 12:40:27 +02004874{
Andre Guedes3655bba2013-10-30 19:01:40 -03004875 u8 bdaddr_type = link_to_bdaddr(link_type, addr_type);
4876 struct mgmt_cp_disconnect *cp;
Johan Hedberg88c3df12012-02-09 14:27:38 +02004877 struct mgmt_rp_disconnect rp;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004878 struct pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004879
Jefferson Delfes36a75f12012-09-18 13:36:54 -04004880 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
4881 hdev);
4882
Johan Hedberg2e58ef32011-11-08 20:40:15 +02004883 cmd = mgmt_pending_find(MGMT_OP_DISCONNECT, hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02004884 if (!cmd)
Marcel Holtmann78929242013-10-06 23:55:47 -07004885 return;
Johan Hedberg8962ee72011-01-20 12:40:27 +02004886
Andre Guedes3655bba2013-10-30 19:01:40 -03004887 cp = cmd->param;
4888
4889 if (bacmp(bdaddr, &cp->addr.bdaddr))
4890 return;
4891
4892 if (cp->addr.type != bdaddr_type)
4893 return;
4894
Johan Hedberg88c3df12012-02-09 14:27:38 +02004895 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes3655bba2013-10-30 19:01:40 -03004896 rp.addr.type = bdaddr_type;
Johan Hedberg37d9ef72011-11-10 15:54:39 +02004897
Marcel Holtmann78929242013-10-06 23:55:47 -07004898 cmd_complete(cmd->sk, cmd->index, MGMT_OP_DISCONNECT,
4899 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02004900
Johan Hedberga664b5b2011-02-19 12:06:02 -03004901 mgmt_pending_remove(cmd);
Johan Hedbergf7520542011-01-20 12:34:39 +02004902}
Johan Hedberg17d5c042011-01-22 06:09:08 +02004903
Marcel Holtmann445608d2013-10-06 23:55:48 -07004904void mgmt_connect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
4905 u8 addr_type, u8 status)
Johan Hedberg17d5c042011-01-22 06:09:08 +02004906{
4907 struct mgmt_ev_connect_failed ev;
4908
Johan Hedberg4c659c32011-11-07 23:13:39 +02004909 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004910 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02004911 ev.status = mgmt_status(status);
Johan Hedberg17d5c042011-01-22 06:09:08 +02004912
Marcel Holtmann445608d2013-10-06 23:55:48 -07004913 mgmt_event(MGMT_EV_CONNECT_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg17d5c042011-01-22 06:09:08 +02004914}
Johan Hedberg980e1a52011-01-22 06:10:07 +02004915
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07004916void mgmt_pin_code_request(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 secure)
Johan Hedberg980e1a52011-01-22 06:10:07 +02004917{
4918 struct mgmt_ev_pin_code_request ev;
4919
Johan Hedbergd8457692012-02-17 14:24:57 +02004920 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004921 ev.addr.type = BDADDR_BREDR;
Waldemar Rymarkiewicza770bb52011-04-28 12:07:59 +02004922 ev.secure = secure;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004923
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07004924 mgmt_event(MGMT_EV_PIN_CODE_REQUEST, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004925}
4926
Marcel Holtmanne669cf82013-10-15 14:26:21 -07004927void mgmt_pin_code_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
4928 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02004929{
4930 struct pending_cmd *cmd;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004931 struct mgmt_rp_pin_code_reply rp;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004932
Johan Hedberg2e58ef32011-11-08 20:40:15 +02004933 cmd = mgmt_pending_find(MGMT_OP_PIN_CODE_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004934 if (!cmd)
Marcel Holtmanne669cf82013-10-15 14:26:21 -07004935 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004936
Johan Hedbergd8457692012-02-17 14:24:57 +02004937 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004938 rp.addr.type = BDADDR_BREDR;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004939
Marcel Holtmanne669cf82013-10-15 14:26:21 -07004940 cmd_complete(cmd->sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
4941 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg980e1a52011-01-22 06:10:07 +02004942
Johan Hedberga664b5b2011-02-19 12:06:02 -03004943 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004944}
4945
Marcel Holtmann3eb38522013-10-15 14:26:22 -07004946void mgmt_pin_code_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
4947 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02004948{
4949 struct pending_cmd *cmd;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004950 struct mgmt_rp_pin_code_reply rp;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004951
Johan Hedberg2e58ef32011-11-08 20:40:15 +02004952 cmd = mgmt_pending_find(MGMT_OP_PIN_CODE_NEG_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004953 if (!cmd)
Marcel Holtmann3eb38522013-10-15 14:26:22 -07004954 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02004955
Johan Hedbergd8457692012-02-17 14:24:57 +02004956 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03004957 rp.addr.type = BDADDR_BREDR;
Johan Hedbergac56fb12011-02-19 12:05:59 -03004958
Marcel Holtmann3eb38522013-10-15 14:26:22 -07004959 cmd_complete(cmd->sk, hdev->id, MGMT_OP_PIN_CODE_NEG_REPLY,
4960 mgmt_status(status), &rp, sizeof(rp));
Johan Hedberg980e1a52011-01-22 06:10:07 +02004961
Johan Hedberga664b5b2011-02-19 12:06:02 -03004962 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02004963}
Johan Hedberga5c29682011-02-19 12:05:57 -03004964
Johan Hedberg744cf192011-11-08 20:40:14 +02004965int mgmt_user_confirm_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004966 u8 link_type, u8 addr_type, __le32 value,
4967 u8 confirm_hint)
Johan Hedberga5c29682011-02-19 12:05:57 -03004968{
4969 struct mgmt_ev_user_confirm_request ev;
4970
Johan Hedberg744cf192011-11-08 20:40:14 +02004971 BT_DBG("%s", hdev->name);
Johan Hedberga5c29682011-02-19 12:05:57 -03004972
Johan Hedberg272d90d2012-02-09 15:26:12 +02004973 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004974 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberg55bc1a32011-04-28 11:28:56 -07004975 ev.confirm_hint = confirm_hint;
Andrei Emeltchenko78e80982012-03-09 13:00:50 +02004976 ev.value = value;
Johan Hedberga5c29682011-02-19 12:05:57 -03004977
Johan Hedberg744cf192011-11-08 20:40:14 +02004978 return mgmt_event(MGMT_EV_USER_CONFIRM_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004979 NULL);
Johan Hedberga5c29682011-02-19 12:05:57 -03004980}
4981
Johan Hedberg272d90d2012-02-09 15:26:12 +02004982int mgmt_user_passkey_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004983 u8 link_type, u8 addr_type)
Brian Gix604086b2011-11-23 08:28:33 -08004984{
4985 struct mgmt_ev_user_passkey_request ev;
4986
4987 BT_DBG("%s", hdev->name);
4988
Johan Hedberg272d90d2012-02-09 15:26:12 +02004989 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03004990 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Brian Gix604086b2011-11-23 08:28:33 -08004991
4992 return mgmt_event(MGMT_EV_USER_PASSKEY_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004993 NULL);
Brian Gix604086b2011-11-23 08:28:33 -08004994}
4995
Brian Gix0df4c182011-11-16 13:53:13 -08004996static int user_pairing_resp_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03004997 u8 link_type, u8 addr_type, u8 status,
4998 u8 opcode)
Johan Hedberga5c29682011-02-19 12:05:57 -03004999{
5000 struct pending_cmd *cmd;
5001 struct mgmt_rp_user_confirm_reply rp;
5002 int err;
5003
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005004 cmd = mgmt_pending_find(opcode, hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03005005 if (!cmd)
5006 return -ENOENT;
5007
Johan Hedberg272d90d2012-02-09 15:26:12 +02005008 bacpy(&rp.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005009 rp.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergaee9b212012-02-18 15:07:59 +02005010 err = cmd_complete(cmd->sk, hdev->id, opcode, mgmt_status(status),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005011 &rp, sizeof(rp));
Johan Hedberga5c29682011-02-19 12:05:57 -03005012
Johan Hedberga664b5b2011-02-19 12:06:02 -03005013 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03005014
5015 return err;
5016}
5017
Johan Hedberg744cf192011-11-08 20:40:14 +02005018int mgmt_user_confirm_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005019 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03005020{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005021 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005022 status, MGMT_OP_USER_CONFIRM_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03005023}
5024
Johan Hedberg272d90d2012-02-09 15:26:12 +02005025int mgmt_user_confirm_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005026 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03005027{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005028 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03005029 status,
5030 MGMT_OP_USER_CONFIRM_NEG_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03005031}
Johan Hedberg2a611692011-02-19 12:06:00 -03005032
Brian Gix604086b2011-11-23 08:28:33 -08005033int mgmt_user_passkey_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005034 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08005035{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005036 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005037 status, MGMT_OP_USER_PASSKEY_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08005038}
5039
Johan Hedberg272d90d2012-02-09 15:26:12 +02005040int mgmt_user_passkey_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005041 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08005042{
Johan Hedberg272d90d2012-02-09 15:26:12 +02005043 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03005044 status,
5045 MGMT_OP_USER_PASSKEY_NEG_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08005046}
5047
Johan Hedberg92a25252012-09-06 18:39:26 +03005048int mgmt_user_passkey_notify(struct hci_dev *hdev, bdaddr_t *bdaddr,
5049 u8 link_type, u8 addr_type, u32 passkey,
5050 u8 entered)
5051{
5052 struct mgmt_ev_passkey_notify ev;
5053
5054 BT_DBG("%s", hdev->name);
5055
5056 bacpy(&ev.addr.bdaddr, bdaddr);
5057 ev.addr.type = link_to_bdaddr(link_type, addr_type);
5058 ev.passkey = __cpu_to_le32(passkey);
5059 ev.entered = entered;
5060
5061 return mgmt_event(MGMT_EV_PASSKEY_NOTIFY, hdev, &ev, sizeof(ev), NULL);
5062}
5063
Marcel Holtmanne5460992013-10-15 14:26:23 -07005064void mgmt_auth_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5065 u8 addr_type, u8 status)
Johan Hedberg2a611692011-02-19 12:06:00 -03005066{
5067 struct mgmt_ev_auth_failed ev;
5068
Johan Hedbergbab73cb2012-02-09 16:07:29 +02005069 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005070 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02005071 ev.status = mgmt_status(status);
Johan Hedberg2a611692011-02-19 12:06:00 -03005072
Marcel Holtmanne5460992013-10-15 14:26:23 -07005073 mgmt_event(MGMT_EV_AUTH_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg2a611692011-02-19 12:06:00 -03005074}
Johan Hedbergb312b1612011-03-16 14:29:37 +02005075
Marcel Holtmann464996a2013-10-15 14:26:24 -07005076void mgmt_auth_enable_complete(struct hci_dev *hdev, u8 status)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005077{
5078 struct cmd_lookup match = { NULL, hdev };
Marcel Holtmann464996a2013-10-15 14:26:24 -07005079 bool changed;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005080
5081 if (status) {
5082 u8 mgmt_err = mgmt_status(status);
5083 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005084 cmd_status_rsp, &mgmt_err);
Marcel Holtmann464996a2013-10-15 14:26:24 -07005085 return;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005086 }
5087
Marcel Holtmann464996a2013-10-15 14:26:24 -07005088 if (test_bit(HCI_AUTH, &hdev->flags))
5089 changed = !test_and_set_bit(HCI_LINK_SECURITY,
5090 &hdev->dev_flags);
5091 else
5092 changed = test_and_clear_bit(HCI_LINK_SECURITY,
5093 &hdev->dev_flags);
Johan Hedberg47990ea2012-02-22 11:58:37 +02005094
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005095 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev, settings_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005096 &match);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005097
Johan Hedberg47990ea2012-02-22 11:58:37 +02005098 if (changed)
Marcel Holtmann464996a2013-10-15 14:26:24 -07005099 new_settings(hdev, match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005100
5101 if (match.sk)
5102 sock_put(match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02005103}
5104
Johan Hedberg890ea892013-03-15 17:06:52 -05005105static void clear_eir(struct hci_request *req)
Johan Hedbergcacaf522012-02-21 00:52:42 +02005106{
Johan Hedberg890ea892013-03-15 17:06:52 -05005107 struct hci_dev *hdev = req->hdev;
Johan Hedbergcacaf522012-02-21 00:52:42 +02005108 struct hci_cp_write_eir cp;
5109
Johan Hedberg976eb202012-10-24 21:12:01 +03005110 if (!lmp_ext_inq_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05005111 return;
Johan Hedbergcacaf522012-02-21 00:52:42 +02005112
Johan Hedbergc80da272012-02-22 15:38:48 +02005113 memset(hdev->eir, 0, sizeof(hdev->eir));
5114
Johan Hedbergcacaf522012-02-21 00:52:42 +02005115 memset(&cp, 0, sizeof(cp));
5116
Johan Hedberg890ea892013-03-15 17:06:52 -05005117 hci_req_add(req, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
Johan Hedbergcacaf522012-02-21 00:52:42 +02005118}
5119
Marcel Holtmann3e248562013-10-15 14:26:25 -07005120void mgmt_ssp_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005121{
5122 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg890ea892013-03-15 17:06:52 -05005123 struct hci_request req;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005124 bool changed = false;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005125
5126 if (status) {
5127 u8 mgmt_err = mgmt_status(status);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005128
5129 if (enable && test_and_clear_bit(HCI_SSP_ENABLED,
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005130 &hdev->dev_flags)) {
5131 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Marcel Holtmann3e248562013-10-15 14:26:25 -07005132 new_settings(hdev, NULL);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005133 }
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005134
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005135 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, cmd_status_rsp,
5136 &mgmt_err);
Marcel Holtmann3e248562013-10-15 14:26:25 -07005137 return;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005138 }
5139
5140 if (enable) {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005141 changed = !test_and_set_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005142 } else {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07005143 changed = test_and_clear_bit(HCI_SSP_ENABLED, &hdev->dev_flags);
5144 if (!changed)
5145 changed = test_and_clear_bit(HCI_HS_ENABLED,
5146 &hdev->dev_flags);
5147 else
5148 clear_bit(HCI_HS_ENABLED, &hdev->dev_flags);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005149 }
5150
5151 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, settings_rsp, &match);
5152
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02005153 if (changed)
Marcel Holtmann3e248562013-10-15 14:26:25 -07005154 new_settings(hdev, match.sk);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005155
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005156 if (match.sk)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005157 sock_put(match.sk);
5158
Johan Hedberg890ea892013-03-15 17:06:52 -05005159 hci_req_init(&req, hdev);
5160
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005161 if (test_bit(HCI_SSP_ENABLED, &hdev->dev_flags))
Johan Hedberg890ea892013-03-15 17:06:52 -05005162 update_eir(&req);
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02005163 else
Johan Hedberg890ea892013-03-15 17:06:52 -05005164 clear_eir(&req);
5165
5166 hci_req_run(&req, NULL);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02005167}
5168
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005169void mgmt_sc_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
5170{
5171 struct cmd_lookup match = { NULL, hdev };
5172 bool changed = false;
5173
5174 if (status) {
5175 u8 mgmt_err = mgmt_status(status);
5176
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005177 if (enable) {
5178 if (test_and_clear_bit(HCI_SC_ENABLED,
5179 &hdev->dev_flags))
5180 new_settings(hdev, NULL);
5181 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
5182 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005183
5184 mgmt_pending_foreach(MGMT_OP_SET_SECURE_CONN, hdev,
5185 cmd_status_rsp, &mgmt_err);
5186 return;
5187 }
5188
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005189 if (enable) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005190 changed = !test_and_set_bit(HCI_SC_ENABLED, &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005191 } else {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005192 changed = test_and_clear_bit(HCI_SC_ENABLED, &hdev->dev_flags);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08005193 clear_bit(HCI_SC_ONLY, &hdev->dev_flags);
5194 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08005195
5196 mgmt_pending_foreach(MGMT_OP_SET_SECURE_CONN, hdev,
5197 settings_rsp, &match);
5198
5199 if (changed)
5200 new_settings(hdev, match.sk);
5201
5202 if (match.sk)
5203 sock_put(match.sk);
5204}
5205
Johan Hedberg92da6092013-03-15 17:06:55 -05005206static void sk_lookup(struct pending_cmd *cmd, void *data)
Johan Hedberg90e70452012-02-23 23:09:40 +02005207{
5208 struct cmd_lookup *match = data;
5209
Johan Hedberg90e70452012-02-23 23:09:40 +02005210 if (match->sk == NULL) {
5211 match->sk = cmd->sk;
5212 sock_hold(match->sk);
5213 }
Johan Hedberg90e70452012-02-23 23:09:40 +02005214}
5215
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07005216void mgmt_set_class_of_dev_complete(struct hci_dev *hdev, u8 *dev_class,
5217 u8 status)
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005218{
Johan Hedberg90e70452012-02-23 23:09:40 +02005219 struct cmd_lookup match = { NULL, hdev, mgmt_status(status) };
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005220
Johan Hedberg92da6092013-03-15 17:06:55 -05005221 mgmt_pending_foreach(MGMT_OP_SET_DEV_CLASS, hdev, sk_lookup, &match);
5222 mgmt_pending_foreach(MGMT_OP_ADD_UUID, hdev, sk_lookup, &match);
5223 mgmt_pending_foreach(MGMT_OP_REMOVE_UUID, hdev, sk_lookup, &match);
Johan Hedberg90e70452012-02-23 23:09:40 +02005224
5225 if (!status)
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07005226 mgmt_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev, dev_class, 3,
5227 NULL);
Johan Hedberg90e70452012-02-23 23:09:40 +02005228
5229 if (match.sk)
5230 sock_put(match.sk);
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01005231}
5232
Marcel Holtmann7667da32013-10-15 14:26:27 -07005233void mgmt_set_local_name_complete(struct hci_dev *hdev, u8 *name, u8 status)
Johan Hedbergb312b1612011-03-16 14:29:37 +02005234{
Johan Hedbergb312b1612011-03-16 14:29:37 +02005235 struct mgmt_cp_set_local_name ev;
Johan Hedberg13928972013-03-15 17:07:00 -05005236 struct pending_cmd *cmd;
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02005237
Johan Hedberg13928972013-03-15 17:07:00 -05005238 if (status)
Marcel Holtmann7667da32013-10-15 14:26:27 -07005239 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02005240
5241 memset(&ev, 0, sizeof(ev));
5242 memcpy(ev.name, name, HCI_MAX_NAME_LENGTH);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02005243 memcpy(ev.short_name, hdev->short_name, HCI_MAX_SHORT_NAME_LENGTH);
Johan Hedbergb312b1612011-03-16 14:29:37 +02005244
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005245 cmd = mgmt_pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
Johan Hedberg13928972013-03-15 17:07:00 -05005246 if (!cmd) {
5247 memcpy(hdev->dev_name, name, sizeof(hdev->dev_name));
Johan Hedbergb312b1612011-03-16 14:29:37 +02005248
Johan Hedberg13928972013-03-15 17:07:00 -05005249 /* If this is a HCI command related to powering on the
5250 * HCI dev don't send any mgmt signals.
5251 */
5252 if (mgmt_pending_find(MGMT_OP_SET_POWERED, hdev))
Marcel Holtmann7667da32013-10-15 14:26:27 -07005253 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02005254 }
5255
Marcel Holtmann7667da32013-10-15 14:26:27 -07005256 mgmt_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, &ev, sizeof(ev),
5257 cmd ? cmd->sk : NULL);
Johan Hedbergb312b1612011-03-16 14:29:37 +02005258}
Szymon Jancc35938b2011-03-22 13:12:21 +01005259
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005260void mgmt_read_local_oob_data_complete(struct hci_dev *hdev, u8 *hash192,
5261 u8 *randomizer192, u8 *hash256,
5262 u8 *randomizer256, u8 status)
Szymon Jancc35938b2011-03-22 13:12:21 +01005263{
5264 struct pending_cmd *cmd;
Szymon Jancc35938b2011-03-22 13:12:21 +01005265
Johan Hedberg744cf192011-11-08 20:40:14 +02005266 BT_DBG("%s status %u", hdev->name, status);
Szymon Jancc35938b2011-03-22 13:12:21 +01005267
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005268 cmd = mgmt_pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01005269 if (!cmd)
Marcel Holtmann3edaf092013-10-15 14:26:28 -07005270 return;
Szymon Jancc35938b2011-03-22 13:12:21 +01005271
5272 if (status) {
Marcel Holtmann3edaf092013-10-15 14:26:28 -07005273 cmd_status(cmd->sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
5274 mgmt_status(status));
Szymon Jancc35938b2011-03-22 13:12:21 +01005275 } else {
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005276 if (test_bit(HCI_SC_ENABLED, &hdev->dev_flags) &&
5277 hash256 && randomizer256) {
5278 struct mgmt_rp_read_local_oob_ext_data rp;
Szymon Jancc35938b2011-03-22 13:12:21 +01005279
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005280 memcpy(rp.hash192, hash192, sizeof(rp.hash192));
5281 memcpy(rp.randomizer192, randomizer192,
5282 sizeof(rp.randomizer192));
Szymon Jancc35938b2011-03-22 13:12:21 +01005283
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08005284 memcpy(rp.hash256, hash256, sizeof(rp.hash256));
5285 memcpy(rp.randomizer256, randomizer256,
5286 sizeof(rp.randomizer256));
5287
5288 cmd_complete(cmd->sk, hdev->id,
5289 MGMT_OP_READ_LOCAL_OOB_DATA, 0,
5290 &rp, sizeof(rp));
5291 } else {
5292 struct mgmt_rp_read_local_oob_data rp;
5293
5294 memcpy(rp.hash, hash192, sizeof(rp.hash));
5295 memcpy(rp.randomizer, randomizer192,
5296 sizeof(rp.randomizer));
5297
5298 cmd_complete(cmd->sk, hdev->id,
5299 MGMT_OP_READ_LOCAL_OOB_DATA, 0,
5300 &rp, sizeof(rp));
5301 }
Szymon Jancc35938b2011-03-22 13:12:21 +01005302 }
5303
5304 mgmt_pending_remove(cmd);
Szymon Jancc35938b2011-03-22 13:12:21 +01005305}
Johan Hedberge17acd42011-03-30 23:57:16 +03005306
Marcel Holtmann901801b2013-10-06 23:55:51 -07005307void mgmt_device_found(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5308 u8 addr_type, u8 *dev_class, s8 rssi, u8 cfm_name, u8
5309 ssp, u8 *eir, u16 eir_len)
Johan Hedberge17acd42011-03-30 23:57:16 +03005310{
Johan Hedberge319d2e2012-01-15 19:51:59 +02005311 char buf[512];
5312 struct mgmt_ev_device_found *ev = (void *) buf;
Johan Hedberg1dc06092012-01-15 21:01:23 +02005313 size_t ev_size;
Johan Hedberge17acd42011-03-30 23:57:16 +03005314
Andre Guedes12602d02013-04-30 15:29:40 -03005315 if (!hci_discovery_active(hdev))
Marcel Holtmann901801b2013-10-06 23:55:51 -07005316 return;
Andre Guedes12602d02013-04-30 15:29:40 -03005317
Johan Hedberg1dc06092012-01-15 21:01:23 +02005318 /* Leave 5 bytes for a potential CoD field */
5319 if (sizeof(*ev) + eir_len + 5 > sizeof(buf))
Marcel Holtmann901801b2013-10-06 23:55:51 -07005320 return;
Andre Guedes7d262f82012-01-10 18:20:49 -03005321
Johan Hedberg1dc06092012-01-15 21:01:23 +02005322 memset(buf, 0, sizeof(buf));
5323
Johan Hedberge319d2e2012-01-15 19:51:59 +02005324 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005325 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberge319d2e2012-01-15 19:51:59 +02005326 ev->rssi = rssi;
Johan Hedberg9a395a82012-02-23 00:00:32 +02005327 if (cfm_name)
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05305328 ev->flags |= __constant_cpu_to_le32(MGMT_DEV_FOUND_CONFIRM_NAME);
Johan Hedberg388fc8f2012-02-23 00:38:59 +02005329 if (!ssp)
Syam Sidhardhan612dfce2012-10-29 22:37:36 +05305330 ev->flags |= __constant_cpu_to_le32(MGMT_DEV_FOUND_LEGACY_PAIRING);
Johan Hedberge17acd42011-03-30 23:57:16 +03005331
Johan Hedberg1dc06092012-01-15 21:01:23 +02005332 if (eir_len > 0)
Johan Hedberge319d2e2012-01-15 19:51:59 +02005333 memcpy(ev->eir, eir, eir_len);
Johan Hedberge17acd42011-03-30 23:57:16 +03005334
Johan Hedberg1dc06092012-01-15 21:01:23 +02005335 if (dev_class && !eir_has_data_type(ev->eir, eir_len, EIR_CLASS_OF_DEV))
5336 eir_len = eir_append_data(ev->eir, eir_len, EIR_CLASS_OF_DEV,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005337 dev_class, 3);
Johan Hedberg1dc06092012-01-15 21:01:23 +02005338
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02005339 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedberg1dc06092012-01-15 21:01:23 +02005340 ev_size = sizeof(*ev) + eir_len;
Andre Guedesf8523592011-09-09 18:56:26 -03005341
Marcel Holtmann901801b2013-10-06 23:55:51 -07005342 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, ev_size, NULL);
Johan Hedberge17acd42011-03-30 23:57:16 +03005343}
Johan Hedberga88a9652011-03-30 13:18:12 +03005344
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07005345void mgmt_remote_name(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
5346 u8 addr_type, s8 rssi, u8 *name, u8 name_len)
Johan Hedberga88a9652011-03-30 13:18:12 +03005347{
Johan Hedbergb644ba32012-01-17 21:48:47 +02005348 struct mgmt_ev_device_found *ev;
5349 char buf[sizeof(*ev) + HCI_MAX_NAME_LENGTH + 2];
5350 u16 eir_len;
Johan Hedberga88a9652011-03-30 13:18:12 +03005351
Johan Hedbergb644ba32012-01-17 21:48:47 +02005352 ev = (struct mgmt_ev_device_found *) buf;
Johan Hedberga88a9652011-03-30 13:18:12 +03005353
Johan Hedbergb644ba32012-01-17 21:48:47 +02005354 memset(buf, 0, sizeof(buf));
Johan Hedberga88a9652011-03-30 13:18:12 +03005355
Johan Hedbergb644ba32012-01-17 21:48:47 +02005356 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03005357 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005358 ev->rssi = rssi;
5359
5360 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE, name,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005361 name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005362
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02005363 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02005364
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07005365 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, sizeof(*ev) + eir_len, NULL);
Johan Hedberga88a9652011-03-30 13:18:12 +03005366}
Johan Hedberg314b2382011-04-27 10:29:57 -04005367
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07005368void mgmt_discovering(struct hci_dev *hdev, u8 discovering)
Johan Hedberg314b2382011-04-27 10:29:57 -04005369{
Johan Hedbergf963e8e2012-02-20 23:30:44 +02005370 struct mgmt_ev_discovering ev;
Johan Hedberg164a6e72011-11-01 17:06:44 +02005371 struct pending_cmd *cmd;
5372
Andre Guedes343fb142011-11-22 17:14:19 -03005373 BT_DBG("%s discovering %u", hdev->name, discovering);
5374
Johan Hedberg164a6e72011-11-01 17:06:44 +02005375 if (discovering)
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005376 cmd = mgmt_pending_find(MGMT_OP_START_DISCOVERY, hdev);
Johan Hedberg164a6e72011-11-01 17:06:44 +02005377 else
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005378 cmd = mgmt_pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
Johan Hedberg164a6e72011-11-01 17:06:44 +02005379
5380 if (cmd != NULL) {
Johan Hedbergf808e162012-02-19 12:52:07 +02005381 u8 type = hdev->discovery.type;
5382
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005383 cmd_complete(cmd->sk, hdev->id, cmd->opcode, 0, &type,
5384 sizeof(type));
Johan Hedberg164a6e72011-11-01 17:06:44 +02005385 mgmt_pending_remove(cmd);
5386 }
5387
Johan Hedbergf963e8e2012-02-20 23:30:44 +02005388 memset(&ev, 0, sizeof(ev));
5389 ev.type = hdev->discovery.type;
5390 ev.discovering = discovering;
5391
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07005392 mgmt_event(MGMT_EV_DISCOVERING, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg314b2382011-04-27 10:29:57 -04005393}
Antti Julku5e762442011-08-25 16:48:02 +03005394
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005395int mgmt_device_blocked(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 type)
Antti Julku5e762442011-08-25 16:48:02 +03005396{
5397 struct pending_cmd *cmd;
5398 struct mgmt_ev_device_blocked ev;
5399
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005400 cmd = mgmt_pending_find(MGMT_OP_BLOCK_DEVICE, hdev);
Antti Julku5e762442011-08-25 16:48:02 +03005401
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005402 bacpy(&ev.addr.bdaddr, bdaddr);
5403 ev.addr.type = type;
Antti Julku5e762442011-08-25 16:48:02 +03005404
Johan Hedberg744cf192011-11-08 20:40:14 +02005405 return mgmt_event(MGMT_EV_DEVICE_BLOCKED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005406 cmd ? cmd->sk : NULL);
Antti Julku5e762442011-08-25 16:48:02 +03005407}
5408
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005409int mgmt_device_unblocked(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 type)
Antti Julku5e762442011-08-25 16:48:02 +03005410{
5411 struct pending_cmd *cmd;
5412 struct mgmt_ev_device_unblocked ev;
5413
Johan Hedberg2e58ef32011-11-08 20:40:15 +02005414 cmd = mgmt_pending_find(MGMT_OP_UNBLOCK_DEVICE, hdev);
Antti Julku5e762442011-08-25 16:48:02 +03005415
Johan Hedberg88c1fe42012-02-09 15:56:11 +02005416 bacpy(&ev.addr.bdaddr, bdaddr);
5417 ev.addr.type = type;
Antti Julku5e762442011-08-25 16:48:02 +03005418
Johan Hedberg744cf192011-11-08 20:40:14 +02005419 return mgmt_event(MGMT_EV_DEVICE_UNBLOCKED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03005420 cmd ? cmd->sk : NULL);
Antti Julku5e762442011-08-25 16:48:02 +03005421}
Marcel Holtmann5976e602013-10-06 04:08:14 -07005422
5423static void adv_enable_complete(struct hci_dev *hdev, u8 status)
5424{
5425 BT_DBG("%s status %u", hdev->name, status);
5426
5427 /* Clear the advertising mgmt setting if we failed to re-enable it */
5428 if (status) {
5429 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Marcel Holtmanna6d811e2013-10-06 04:11:12 -07005430 new_settings(hdev, NULL);
Marcel Holtmann5976e602013-10-06 04:08:14 -07005431 }
5432}
5433
5434void mgmt_reenable_advertising(struct hci_dev *hdev)
5435{
5436 struct hci_request req;
5437
Marcel Holtmannb145edc2013-10-10 09:47:54 -07005438 if (hci_conn_num(hdev, LE_LINK) > 0)
Marcel Holtmann5976e602013-10-06 04:08:14 -07005439 return;
5440
5441 if (!test_bit(HCI_ADVERTISING, &hdev->dev_flags))
5442 return;
5443
5444 hci_req_init(&req, hdev);
5445 enable_advertising(&req);
5446
5447 /* If this fails we have no option but to let user space know
5448 * that we've disabled advertising.
5449 */
5450 if (hci_req_run(&req, adv_enable_complete) < 0) {
5451 clear_bit(HCI_ADVERTISING, &hdev->dev_flags);
Marcel Holtmanna6d811e2013-10-06 04:11:12 -07005452 new_settings(hdev, NULL);
Marcel Holtmann5976e602013-10-06 04:08:14 -07005453 }
5454}