blob: cd9f345894e0c4257826f2e58a9f9c19d601a6bf [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>
Johan Hedberg71290692015-02-20 13:26:23 +020032#include <net/bluetooth/hci_sock.h>
Johan Hedberg4bc58f52014-05-20 09:45:47 +030033#include <net/bluetooth/l2cap.h>
Johan Hedberg03811012010-12-08 00:21:06 +020034#include <net/bluetooth/mgmt.h>
Marcel Holtmannac4b7232013-10-10 14:54:16 -070035
Johan Hedberg0857dd32014-12-19 13:40:20 +020036#include "hci_request.h"
Marcel Holtmannac4b7232013-10-10 14:54:16 -070037#include "smp.h"
Johan Hedberga380b6c2015-03-17 13:48:48 +020038#include "mgmt_util.h"
Johan Hedberg03811012010-12-08 00:21:06 +020039
Johan Hedberg2da9c552012-02-17 14:39:28 +020040#define MGMT_VERSION 1
Szymon Janc33102302016-09-18 12:50:07 +020041#define MGMT_REVISION 14
Johan Hedberg02d98122010-12-13 21:07:04 +020042
Johan Hedberge70bb2e2012-02-13 16:59:33 +020043static const u16 mgmt_commands[] = {
44 MGMT_OP_READ_INDEX_LIST,
45 MGMT_OP_READ_INFO,
46 MGMT_OP_SET_POWERED,
47 MGMT_OP_SET_DISCOVERABLE,
48 MGMT_OP_SET_CONNECTABLE,
49 MGMT_OP_SET_FAST_CONNECTABLE,
Johan Hedbergb2939472014-07-30 09:22:23 +030050 MGMT_OP_SET_BONDABLE,
Johan Hedberge70bb2e2012-02-13 16:59:33 +020051 MGMT_OP_SET_LINK_SECURITY,
52 MGMT_OP_SET_SSP,
53 MGMT_OP_SET_HS,
54 MGMT_OP_SET_LE,
55 MGMT_OP_SET_DEV_CLASS,
56 MGMT_OP_SET_LOCAL_NAME,
57 MGMT_OP_ADD_UUID,
58 MGMT_OP_REMOVE_UUID,
59 MGMT_OP_LOAD_LINK_KEYS,
60 MGMT_OP_LOAD_LONG_TERM_KEYS,
61 MGMT_OP_DISCONNECT,
62 MGMT_OP_GET_CONNECTIONS,
63 MGMT_OP_PIN_CODE_REPLY,
64 MGMT_OP_PIN_CODE_NEG_REPLY,
65 MGMT_OP_SET_IO_CAPABILITY,
66 MGMT_OP_PAIR_DEVICE,
67 MGMT_OP_CANCEL_PAIR_DEVICE,
68 MGMT_OP_UNPAIR_DEVICE,
69 MGMT_OP_USER_CONFIRM_REPLY,
70 MGMT_OP_USER_CONFIRM_NEG_REPLY,
71 MGMT_OP_USER_PASSKEY_REPLY,
72 MGMT_OP_USER_PASSKEY_NEG_REPLY,
73 MGMT_OP_READ_LOCAL_OOB_DATA,
74 MGMT_OP_ADD_REMOTE_OOB_DATA,
75 MGMT_OP_REMOVE_REMOTE_OOB_DATA,
76 MGMT_OP_START_DISCOVERY,
77 MGMT_OP_STOP_DISCOVERY,
78 MGMT_OP_CONFIRM_NAME,
79 MGMT_OP_BLOCK_DEVICE,
80 MGMT_OP_UNBLOCK_DEVICE,
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -070081 MGMT_OP_SET_DEVICE_ID,
Johan Hedberg4375f102013-09-25 13:26:10 +030082 MGMT_OP_SET_ADVERTISING,
Johan Hedberg0663ca22013-10-02 13:43:14 +030083 MGMT_OP_SET_BREDR,
Marcel Holtmannd13eafc2013-10-02 04:41:30 -070084 MGMT_OP_SET_STATIC_ADDRESS,
Marcel Holtmann7f72134e2013-10-11 14:44:58 -070085 MGMT_OP_SET_SCAN_PARAMS,
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -080086 MGMT_OP_SET_SECURE_CONN,
Marcel Holtmann4e39ac82014-01-31 11:55:22 -080087 MGMT_OP_SET_DEBUG_KEYS,
Johan Hedberg62b04cd2014-02-23 19:42:27 +020088 MGMT_OP_SET_PRIVACY,
Johan Hedberg41edf162014-02-18 10:19:35 +020089 MGMT_OP_LOAD_IRKS,
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +020090 MGMT_OP_GET_CONN_INFO,
Johan Hedberg95868422014-06-28 17:54:07 +030091 MGMT_OP_GET_CLOCK_INFO,
Marcel Holtmann2faade52014-06-29 19:44:03 +020092 MGMT_OP_ADD_DEVICE,
93 MGMT_OP_REMOVE_DEVICE,
Johan Hedberga26f3dc2014-07-02 17:37:29 +030094 MGMT_OP_LOAD_CONN_PARAM,
Marcel Holtmann73d1df22014-07-02 22:10:52 +020095 MGMT_OP_READ_UNCONF_INDEX_LIST,
Marcel Holtmann9fc3bfb2014-07-04 00:46:56 +020096 MGMT_OP_READ_CONFIG_INFO,
Marcel Holtmanndbece372014-07-04 18:11:55 +020097 MGMT_OP_SET_EXTERNAL_CONFIG,
Marcel Holtmann9713c172014-07-06 12:11:15 +020098 MGMT_OP_SET_PUBLIC_ADDRESS,
Jakub Pawlowski66ea9422014-12-05 10:55:59 +010099 MGMT_OP_START_SERVICE_DISCOVERY,
Marcel Holtmann4f0f1552015-03-14 22:43:19 -0700100 MGMT_OP_READ_LOCAL_OOB_EXT_DATA,
Marcel Holtmann96f14742015-03-14 19:27:57 -0700101 MGMT_OP_READ_EXT_INDEX_LIST,
Marcel Holtmannd3d53052015-03-14 20:53:25 -0700102 MGMT_OP_READ_ADV_FEATURES,
Arman Uguray24b4f382015-03-23 15:57:12 -0700103 MGMT_OP_ADD_ADVERTISING,
Arman Ugurayda9293352015-03-23 15:57:13 -0700104 MGMT_OP_REMOVE_ADVERTISING,
Marcel Holtmann40b25fe2015-11-19 16:16:43 +0100105 MGMT_OP_GET_ADV_SIZE_INFO,
Johan Hedberg78b781c2016-01-05 13:19:32 +0200106 MGMT_OP_START_LIMITED_DISCOVERY,
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200107 MGMT_OP_READ_EXT_INFO,
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +0200108 MGMT_OP_SET_APPEARANCE,
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200109};
110
111static const u16 mgmt_events[] = {
112 MGMT_EV_CONTROLLER_ERROR,
113 MGMT_EV_INDEX_ADDED,
114 MGMT_EV_INDEX_REMOVED,
115 MGMT_EV_NEW_SETTINGS,
116 MGMT_EV_CLASS_OF_DEV_CHANGED,
117 MGMT_EV_LOCAL_NAME_CHANGED,
118 MGMT_EV_NEW_LINK_KEY,
119 MGMT_EV_NEW_LONG_TERM_KEY,
120 MGMT_EV_DEVICE_CONNECTED,
121 MGMT_EV_DEVICE_DISCONNECTED,
122 MGMT_EV_CONNECT_FAILED,
123 MGMT_EV_PIN_CODE_REQUEST,
124 MGMT_EV_USER_CONFIRM_REQUEST,
125 MGMT_EV_USER_PASSKEY_REQUEST,
126 MGMT_EV_AUTH_FAILED,
127 MGMT_EV_DEVICE_FOUND,
128 MGMT_EV_DISCOVERING,
129 MGMT_EV_DEVICE_BLOCKED,
130 MGMT_EV_DEVICE_UNBLOCKED,
131 MGMT_EV_DEVICE_UNPAIRED,
Johan Hedberg92a25252012-09-06 18:39:26 +0300132 MGMT_EV_PASSKEY_NOTIFY,
Marcel Holtmann1b60ef22014-02-21 21:35:30 -0800133 MGMT_EV_NEW_IRK,
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -0700134 MGMT_EV_NEW_CSRK,
Marcel Holtmann8afef092014-06-29 22:28:34 +0200135 MGMT_EV_DEVICE_ADDED,
136 MGMT_EV_DEVICE_REMOVED,
Andre Guedesffb5a8272014-07-01 18:10:11 -0300137 MGMT_EV_NEW_CONN_PARAM,
Marcel Holtmann0602a8a2014-07-02 21:30:54 +0200138 MGMT_EV_UNCONF_INDEX_ADDED,
Marcel Holtmannedd38962014-07-02 21:30:55 +0200139 MGMT_EV_UNCONF_INDEX_REMOVED,
Marcel Holtmannf4537c02014-07-04 19:06:23 +0200140 MGMT_EV_NEW_CONFIG_OPTIONS,
Marcel Holtmannced85542015-03-14 19:27:56 -0700141 MGMT_EV_EXT_INDEX_ADDED,
142 MGMT_EV_EXT_INDEX_REMOVED,
Marcel Holtmann72000df2015-03-16 16:11:21 -0700143 MGMT_EV_LOCAL_OOB_DATA_UPDATED,
Arman Uguray24b4f382015-03-23 15:57:12 -0700144 MGMT_EV_ADVERTISING_ADDED,
145 MGMT_EV_ADVERTISING_REMOVED,
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200146 MGMT_EV_EXT_INFO_CHANGED,
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200147};
148
Marcel Holtmann99c679a2015-03-24 17:31:03 -0700149static const u16 mgmt_untrusted_commands[] = {
150 MGMT_OP_READ_INDEX_LIST,
151 MGMT_OP_READ_INFO,
152 MGMT_OP_READ_UNCONF_INDEX_LIST,
153 MGMT_OP_READ_CONFIG_INFO,
154 MGMT_OP_READ_EXT_INDEX_LIST,
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200155 MGMT_OP_READ_EXT_INFO,
Marcel Holtmann99c679a2015-03-24 17:31:03 -0700156};
157
158static const u16 mgmt_untrusted_events[] = {
159 MGMT_EV_INDEX_ADDED,
160 MGMT_EV_INDEX_REMOVED,
161 MGMT_EV_NEW_SETTINGS,
162 MGMT_EV_CLASS_OF_DEV_CHANGED,
163 MGMT_EV_LOCAL_NAME_CHANGED,
164 MGMT_EV_UNCONF_INDEX_ADDED,
165 MGMT_EV_UNCONF_INDEX_REMOVED,
166 MGMT_EV_NEW_CONFIG_OPTIONS,
167 MGMT_EV_EXT_INDEX_ADDED,
168 MGMT_EV_EXT_INDEX_REMOVED,
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200169 MGMT_EV_EXT_INFO_CHANGED,
Marcel Holtmann99c679a2015-03-24 17:31:03 -0700170};
171
Marcel Holtmann17b02e62012-03-01 14:32:37 -0800172#define CACHE_TIMEOUT msecs_to_jiffies(2 * 1000)
Johan Hedberg7d785252011-12-15 00:47:39 +0200173
Johan Hedbergd25b78e2015-01-27 12:55:52 +0200174#define ZERO_KEY "\x00\x00\x00\x00\x00\x00\x00\x00" \
175 "\x00\x00\x00\x00\x00\x00\x00\x00"
176
Johan Hedbergca69b792011-11-11 18:10:00 +0200177/* HCI to MGMT error code conversion table */
178static u8 mgmt_status_table[] = {
179 MGMT_STATUS_SUCCESS,
180 MGMT_STATUS_UNKNOWN_COMMAND, /* Unknown Command */
181 MGMT_STATUS_NOT_CONNECTED, /* No Connection */
182 MGMT_STATUS_FAILED, /* Hardware Failure */
183 MGMT_STATUS_CONNECT_FAILED, /* Page Timeout */
184 MGMT_STATUS_AUTH_FAILED, /* Authentication Failed */
Johan Hedbergeadd6632014-01-13 17:15:53 +0200185 MGMT_STATUS_AUTH_FAILED, /* PIN or Key Missing */
Johan Hedbergca69b792011-11-11 18:10:00 +0200186 MGMT_STATUS_NO_RESOURCES, /* Memory Full */
187 MGMT_STATUS_TIMEOUT, /* Connection Timeout */
188 MGMT_STATUS_NO_RESOURCES, /* Max Number of Connections */
189 MGMT_STATUS_NO_RESOURCES, /* Max Number of SCO Connections */
190 MGMT_STATUS_ALREADY_CONNECTED, /* ACL Connection Exists */
191 MGMT_STATUS_BUSY, /* Command Disallowed */
192 MGMT_STATUS_NO_RESOURCES, /* Rejected Limited Resources */
193 MGMT_STATUS_REJECTED, /* Rejected Security */
194 MGMT_STATUS_REJECTED, /* Rejected Personal */
195 MGMT_STATUS_TIMEOUT, /* Host Timeout */
196 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported Feature */
197 MGMT_STATUS_INVALID_PARAMS, /* Invalid Parameters */
198 MGMT_STATUS_DISCONNECTED, /* OE User Ended Connection */
199 MGMT_STATUS_NO_RESOURCES, /* OE Low Resources */
200 MGMT_STATUS_DISCONNECTED, /* OE Power Off */
201 MGMT_STATUS_DISCONNECTED, /* Connection Terminated */
202 MGMT_STATUS_BUSY, /* Repeated Attempts */
203 MGMT_STATUS_REJECTED, /* Pairing Not Allowed */
204 MGMT_STATUS_FAILED, /* Unknown LMP PDU */
205 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported Remote Feature */
206 MGMT_STATUS_REJECTED, /* SCO Offset Rejected */
207 MGMT_STATUS_REJECTED, /* SCO Interval Rejected */
208 MGMT_STATUS_REJECTED, /* Air Mode Rejected */
209 MGMT_STATUS_INVALID_PARAMS, /* Invalid LMP Parameters */
210 MGMT_STATUS_FAILED, /* Unspecified Error */
211 MGMT_STATUS_NOT_SUPPORTED, /* Unsupported LMP Parameter Value */
212 MGMT_STATUS_FAILED, /* Role Change Not Allowed */
213 MGMT_STATUS_TIMEOUT, /* LMP Response Timeout */
214 MGMT_STATUS_FAILED, /* LMP Error Transaction Collision */
215 MGMT_STATUS_FAILED, /* LMP PDU Not Allowed */
216 MGMT_STATUS_REJECTED, /* Encryption Mode Not Accepted */
217 MGMT_STATUS_FAILED, /* Unit Link Key Used */
218 MGMT_STATUS_NOT_SUPPORTED, /* QoS Not Supported */
219 MGMT_STATUS_TIMEOUT, /* Instant Passed */
220 MGMT_STATUS_NOT_SUPPORTED, /* Pairing Not Supported */
221 MGMT_STATUS_FAILED, /* Transaction Collision */
222 MGMT_STATUS_INVALID_PARAMS, /* Unacceptable Parameter */
223 MGMT_STATUS_REJECTED, /* QoS Rejected */
224 MGMT_STATUS_NOT_SUPPORTED, /* Classification Not Supported */
225 MGMT_STATUS_REJECTED, /* Insufficient Security */
226 MGMT_STATUS_INVALID_PARAMS, /* Parameter Out Of Range */
227 MGMT_STATUS_BUSY, /* Role Switch Pending */
228 MGMT_STATUS_FAILED, /* Slot Violation */
229 MGMT_STATUS_FAILED, /* Role Switch Failed */
230 MGMT_STATUS_INVALID_PARAMS, /* EIR Too Large */
231 MGMT_STATUS_NOT_SUPPORTED, /* Simple Pairing Not Supported */
232 MGMT_STATUS_BUSY, /* Host Busy Pairing */
233 MGMT_STATUS_REJECTED, /* Rejected, No Suitable Channel */
234 MGMT_STATUS_BUSY, /* Controller Busy */
235 MGMT_STATUS_INVALID_PARAMS, /* Unsuitable Connection Interval */
236 MGMT_STATUS_TIMEOUT, /* Directed Advertising Timeout */
237 MGMT_STATUS_AUTH_FAILED, /* Terminated Due to MIC Failure */
238 MGMT_STATUS_CONNECT_FAILED, /* Connection Establishment Failed */
239 MGMT_STATUS_CONNECT_FAILED, /* MAC Connection Failed */
240};
241
242static u8 mgmt_status(u8 hci_status)
243{
244 if (hci_status < ARRAY_SIZE(mgmt_status_table))
245 return mgmt_status_table[hci_status];
246
247 return MGMT_STATUS_FAILED;
248}
249
Marcel Holtmannc08b1a12015-03-14 19:27:59 -0700250static int mgmt_index_event(u16 event, struct hci_dev *hdev, void *data,
251 u16 len, int flag)
Marcel Holtmannf9207332015-03-14 19:27:55 -0700252{
Marcel Holtmannc08b1a12015-03-14 19:27:59 -0700253 return mgmt_send_event(event, hdev, HCI_CHANNEL_CONTROL, data, len,
254 flag, NULL);
Marcel Holtmannf9207332015-03-14 19:27:55 -0700255}
256
Marcel Holtmann72000df2015-03-16 16:11:21 -0700257static int mgmt_limited_event(u16 event, struct hci_dev *hdev, void *data,
258 u16 len, int flag, struct sock *skip_sk)
259{
260 return mgmt_send_event(event, hdev, HCI_CHANNEL_CONTROL, data, len,
261 flag, skip_sk);
262}
263
Johan Hedberg7a00ff42015-03-06 21:08:56 +0200264static int mgmt_event(u16 event, struct hci_dev *hdev, void *data, u16 len,
265 struct sock *skip_sk)
266{
267 return mgmt_send_event(event, hdev, HCI_CHANNEL_CONTROL, data, len,
Marcel Holtmannc08b1a12015-03-14 19:27:59 -0700268 HCI_SOCK_TRUSTED, skip_sk);
Johan Hedberg7a00ff42015-03-06 21:08:56 +0200269}
270
Johan Hedberg85813a72015-10-21 18:02:59 +0300271static u8 le_addr_type(u8 mgmt_addr_type)
272{
273 if (mgmt_addr_type == BDADDR_LE_PUBLIC)
274 return ADDR_LE_DEV_PUBLIC;
275 else
276 return ADDR_LE_DEV_RANDOM;
277}
278
Marcel Holtmann03c979c2016-08-27 20:23:39 +0200279void mgmt_fill_version_info(void *ver)
280{
281 struct mgmt_rp_read_version *rp = ver;
282
283 rp->version = MGMT_VERSION;
284 rp->revision = cpu_to_le16(MGMT_REVISION);
285}
286
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300287static int read_version(struct sock *sk, struct hci_dev *hdev, void *data,
288 u16 data_len)
Johan Hedberga38528f2011-01-22 06:46:43 +0200289{
290 struct mgmt_rp_read_version rp;
291
292 BT_DBG("sock %p", sk);
293
Marcel Holtmann03c979c2016-08-27 20:23:39 +0200294 mgmt_fill_version_info(&rp);
Johan Hedberga38528f2011-01-22 06:46:43 +0200295
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200296 return mgmt_cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_VERSION, 0,
297 &rp, sizeof(rp));
Johan Hedberga38528f2011-01-22 06:46:43 +0200298}
299
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300300static int read_commands(struct sock *sk, struct hci_dev *hdev, void *data,
301 u16 data_len)
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200302{
303 struct mgmt_rp_read_commands *rp;
Marcel Holtmann99c679a2015-03-24 17:31:03 -0700304 u16 num_commands, num_events;
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200305 size_t rp_size;
306 int i, err;
307
308 BT_DBG("sock %p", sk);
309
Marcel Holtmann99c679a2015-03-24 17:31:03 -0700310 if (hci_sock_test_flag(sk, HCI_SOCK_TRUSTED)) {
311 num_commands = ARRAY_SIZE(mgmt_commands);
312 num_events = ARRAY_SIZE(mgmt_events);
313 } else {
314 num_commands = ARRAY_SIZE(mgmt_untrusted_commands);
315 num_events = ARRAY_SIZE(mgmt_untrusted_events);
316 }
317
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200318 rp_size = sizeof(*rp) + ((num_commands + num_events) * sizeof(u16));
319
320 rp = kmalloc(rp_size, GFP_KERNEL);
321 if (!rp)
322 return -ENOMEM;
323
Joe Perchesdcf4adb2014-03-12 10:52:35 -0700324 rp->num_commands = cpu_to_le16(num_commands);
325 rp->num_events = cpu_to_le16(num_events);
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200326
Marcel Holtmann99c679a2015-03-24 17:31:03 -0700327 if (hci_sock_test_flag(sk, HCI_SOCK_TRUSTED)) {
328 __le16 *opcode = rp->opcodes;
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200329
Marcel Holtmann99c679a2015-03-24 17:31:03 -0700330 for (i = 0; i < num_commands; i++, opcode++)
331 put_unaligned_le16(mgmt_commands[i], opcode);
332
333 for (i = 0; i < num_events; i++, opcode++)
334 put_unaligned_le16(mgmt_events[i], opcode);
335 } else {
336 __le16 *opcode = rp->opcodes;
337
338 for (i = 0; i < num_commands; i++, opcode++)
339 put_unaligned_le16(mgmt_untrusted_commands[i], opcode);
340
341 for (i = 0; i < num_events; i++, opcode++)
342 put_unaligned_le16(mgmt_untrusted_events[i], opcode);
343 }
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200344
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200345 err = mgmt_cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_COMMANDS, 0,
346 rp, rp_size);
Johan Hedberge70bb2e2012-02-13 16:59:33 +0200347 kfree(rp);
348
349 return err;
350}
351
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300352static int read_index_list(struct sock *sk, struct hci_dev *hdev, void *data,
353 u16 data_len)
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200354{
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200355 struct mgmt_rp_read_index_list *rp;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +0200356 struct hci_dev *d;
Johan Hedberga38528f2011-01-22 06:46:43 +0200357 size_t rp_len;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200358 u16 count;
Johan Hedberg476e44c2012-10-19 20:10:46 +0300359 int err;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200360
361 BT_DBG("sock %p", sk);
362
363 read_lock(&hci_dev_list_lock);
364
365 count = 0;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +0300366 list_for_each_entry(d, &hci_dev_list, list) {
Marcel Holtmannca8bee52016-07-05 14:30:14 +0200367 if (d->dev_type == HCI_PRIMARY &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700368 !hci_dev_test_flag(d, HCI_UNCONFIGURED))
Marcel Holtmann1514b892013-10-06 08:25:01 -0700369 count++;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200370 }
371
Johan Hedberga38528f2011-01-22 06:46:43 +0200372 rp_len = sizeof(*rp) + (2 * count);
373 rp = kmalloc(rp_len, GFP_ATOMIC);
374 if (!rp) {
Jesper Juhlb2c60d42011-01-14 00:18:49 +0100375 read_unlock(&hci_dev_list_lock);
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200376 return -ENOMEM;
Jesper Juhlb2c60d42011-01-14 00:18:49 +0100377 }
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200378
Johan Hedberg476e44c2012-10-19 20:10:46 +0300379 count = 0;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +0200380 list_for_each_entry(d, &hci_dev_list, list) {
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700381 if (hci_dev_test_flag(d, HCI_SETUP) ||
382 hci_dev_test_flag(d, HCI_CONFIG) ||
383 hci_dev_test_flag(d, HCI_USER_CHANNEL))
Johan Hedbergab81cbf2010-12-15 13:53:18 +0200384 continue;
385
Marcel Holtmann73d1df22014-07-02 22:10:52 +0200386 /* Devices marked as raw-only are neither configured
387 * nor unconfigured controllers.
388 */
389 if (test_bit(HCI_QUIRK_RAW_DEVICE, &d->quirks))
Marcel Holtmann0736cfa2013-08-26 21:40:51 -0700390 continue;
391
Marcel Holtmannca8bee52016-07-05 14:30:14 +0200392 if (d->dev_type == HCI_PRIMARY &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700393 !hci_dev_test_flag(d, HCI_UNCONFIGURED)) {
Marcel Holtmann1514b892013-10-06 08:25:01 -0700394 rp->index[count++] = cpu_to_le16(d->id);
395 BT_DBG("Added hci%u", d->id);
396 }
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200397 }
398
Johan Hedberg476e44c2012-10-19 20:10:46 +0300399 rp->num_controllers = cpu_to_le16(count);
400 rp_len = sizeof(*rp) + (2 * count);
401
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200402 read_unlock(&hci_dev_list_lock);
403
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200404 err = mgmt_cmd_complete(sk, MGMT_INDEX_NONE, MGMT_OP_READ_INDEX_LIST,
405 0, rp, rp_len);
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200406
Johan Hedberga38528f2011-01-22 06:46:43 +0200407 kfree(rp);
408
409 return err;
Johan Hedbergfaba42e2010-12-13 21:07:05 +0200410}
411
Marcel Holtmann73d1df22014-07-02 22:10:52 +0200412static int read_unconf_index_list(struct sock *sk, struct hci_dev *hdev,
413 void *data, u16 data_len)
414{
415 struct mgmt_rp_read_unconf_index_list *rp;
416 struct hci_dev *d;
417 size_t rp_len;
418 u16 count;
419 int err;
420
421 BT_DBG("sock %p", sk);
422
423 read_lock(&hci_dev_list_lock);
424
425 count = 0;
426 list_for_each_entry(d, &hci_dev_list, list) {
Marcel Holtmannca8bee52016-07-05 14:30:14 +0200427 if (d->dev_type == HCI_PRIMARY &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700428 hci_dev_test_flag(d, HCI_UNCONFIGURED))
Marcel Holtmann73d1df22014-07-02 22:10:52 +0200429 count++;
430 }
431
432 rp_len = sizeof(*rp) + (2 * count);
433 rp = kmalloc(rp_len, GFP_ATOMIC);
434 if (!rp) {
435 read_unlock(&hci_dev_list_lock);
436 return -ENOMEM;
437 }
438
439 count = 0;
440 list_for_each_entry(d, &hci_dev_list, list) {
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700441 if (hci_dev_test_flag(d, HCI_SETUP) ||
442 hci_dev_test_flag(d, HCI_CONFIG) ||
443 hci_dev_test_flag(d, HCI_USER_CHANNEL))
Marcel Holtmann73d1df22014-07-02 22:10:52 +0200444 continue;
445
446 /* Devices marked as raw-only are neither configured
447 * nor unconfigured controllers.
448 */
449 if (test_bit(HCI_QUIRK_RAW_DEVICE, &d->quirks))
450 continue;
451
Marcel Holtmannca8bee52016-07-05 14:30:14 +0200452 if (d->dev_type == HCI_PRIMARY &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700453 hci_dev_test_flag(d, HCI_UNCONFIGURED)) {
Marcel Holtmann73d1df22014-07-02 22:10:52 +0200454 rp->index[count++] = cpu_to_le16(d->id);
455 BT_DBG("Added hci%u", d->id);
456 }
457 }
458
459 rp->num_controllers = cpu_to_le16(count);
460 rp_len = sizeof(*rp) + (2 * count);
461
462 read_unlock(&hci_dev_list_lock);
463
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200464 err = mgmt_cmd_complete(sk, MGMT_INDEX_NONE,
465 MGMT_OP_READ_UNCONF_INDEX_LIST, 0, rp, rp_len);
Marcel Holtmann73d1df22014-07-02 22:10:52 +0200466
467 kfree(rp);
468
469 return err;
470}
471
Marcel Holtmann96f14742015-03-14 19:27:57 -0700472static int read_ext_index_list(struct sock *sk, struct hci_dev *hdev,
473 void *data, u16 data_len)
474{
475 struct mgmt_rp_read_ext_index_list *rp;
476 struct hci_dev *d;
477 size_t rp_len;
478 u16 count;
479 int err;
480
481 BT_DBG("sock %p", sk);
482
483 read_lock(&hci_dev_list_lock);
484
485 count = 0;
486 list_for_each_entry(d, &hci_dev_list, list) {
Marcel Holtmannca8bee52016-07-05 14:30:14 +0200487 if (d->dev_type == HCI_PRIMARY || d->dev_type == HCI_AMP)
Marcel Holtmann96f14742015-03-14 19:27:57 -0700488 count++;
489 }
490
491 rp_len = sizeof(*rp) + (sizeof(rp->entry[0]) * count);
492 rp = kmalloc(rp_len, GFP_ATOMIC);
493 if (!rp) {
494 read_unlock(&hci_dev_list_lock);
495 return -ENOMEM;
496 }
497
498 count = 0;
499 list_for_each_entry(d, &hci_dev_list, list) {
500 if (hci_dev_test_flag(d, HCI_SETUP) ||
501 hci_dev_test_flag(d, HCI_CONFIG) ||
502 hci_dev_test_flag(d, HCI_USER_CHANNEL))
503 continue;
504
505 /* Devices marked as raw-only are neither configured
506 * nor unconfigured controllers.
507 */
508 if (test_bit(HCI_QUIRK_RAW_DEVICE, &d->quirks))
509 continue;
510
Marcel Holtmannca8bee52016-07-05 14:30:14 +0200511 if (d->dev_type == HCI_PRIMARY) {
Marcel Holtmann96f14742015-03-14 19:27:57 -0700512 if (hci_dev_test_flag(d, HCI_UNCONFIGURED))
513 rp->entry[count].type = 0x01;
514 else
515 rp->entry[count].type = 0x00;
516 } else if (d->dev_type == HCI_AMP) {
517 rp->entry[count].type = 0x02;
518 } else {
519 continue;
520 }
521
522 rp->entry[count].bus = d->bus;
523 rp->entry[count++].index = cpu_to_le16(d->id);
524 BT_DBG("Added hci%u", d->id);
525 }
526
527 rp->num_controllers = cpu_to_le16(count);
528 rp_len = sizeof(*rp) + (sizeof(rp->entry[0]) * count);
529
530 read_unlock(&hci_dev_list_lock);
531
532 /* If this command is called at least once, then all the
533 * default index and unconfigured index events are disabled
534 * and from now on only extended index events are used.
535 */
536 hci_sock_set_flag(sk, HCI_MGMT_EXT_INDEX_EVENTS);
537 hci_sock_clear_flag(sk, HCI_MGMT_INDEX_EVENTS);
538 hci_sock_clear_flag(sk, HCI_MGMT_UNCONF_INDEX_EVENTS);
539
540 err = mgmt_cmd_complete(sk, MGMT_INDEX_NONE,
541 MGMT_OP_READ_EXT_INDEX_LIST, 0, rp, rp_len);
542
543 kfree(rp);
544
545 return err;
546}
547
Marcel Holtmanndbece372014-07-04 18:11:55 +0200548static bool is_configured(struct hci_dev *hdev)
549{
550 if (test_bit(HCI_QUIRK_EXTERNAL_CONFIG, &hdev->quirks) &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700551 !hci_dev_test_flag(hdev, HCI_EXT_CONFIGURED))
Marcel Holtmanndbece372014-07-04 18:11:55 +0200552 return false;
553
554 if (test_bit(HCI_QUIRK_INVALID_BDADDR, &hdev->quirks) &&
555 !bacmp(&hdev->public_addr, BDADDR_ANY))
556 return false;
557
558 return true;
559}
560
Marcel Holtmann89bc22d2014-07-04 16:54:37 +0200561static __le32 get_missing_options(struct hci_dev *hdev)
562{
563 u32 options = 0;
564
Marcel Holtmanndbece372014-07-04 18:11:55 +0200565 if (test_bit(HCI_QUIRK_EXTERNAL_CONFIG, &hdev->quirks) &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700566 !hci_dev_test_flag(hdev, HCI_EXT_CONFIGURED))
Marcel Holtmanneb1904f2014-07-04 17:23:33 +0200567 options |= MGMT_OPTION_EXTERNAL_CONFIG;
568
Marcel Holtmann89bc22d2014-07-04 16:54:37 +0200569 if (test_bit(HCI_QUIRK_INVALID_BDADDR, &hdev->quirks) &&
570 !bacmp(&hdev->public_addr, BDADDR_ANY))
571 options |= MGMT_OPTION_PUBLIC_ADDRESS;
572
573 return cpu_to_le32(options);
574}
575
Marcel Holtmannf4537c02014-07-04 19:06:23 +0200576static int new_options(struct hci_dev *hdev, struct sock *skip)
577{
578 __le32 options = get_missing_options(hdev);
579
Marcel Holtmann5504c3a2016-08-29 06:19:46 +0200580 return mgmt_limited_event(MGMT_EV_NEW_CONFIG_OPTIONS, hdev, &options,
581 sizeof(options), HCI_MGMT_OPTION_EVENTS, skip);
Marcel Holtmannf4537c02014-07-04 19:06:23 +0200582}
583
Marcel Holtmanndbece372014-07-04 18:11:55 +0200584static int send_options_rsp(struct sock *sk, u16 opcode, struct hci_dev *hdev)
585{
586 __le32 options = get_missing_options(hdev);
587
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200588 return mgmt_cmd_complete(sk, hdev->id, opcode, 0, &options,
589 sizeof(options));
Marcel Holtmanndbece372014-07-04 18:11:55 +0200590}
591
Marcel Holtmann9fc3bfb2014-07-04 00:46:56 +0200592static int read_config_info(struct sock *sk, struct hci_dev *hdev,
593 void *data, u16 data_len)
594{
595 struct mgmt_rp_read_config_info rp;
Marcel Holtmann89bc22d2014-07-04 16:54:37 +0200596 u32 options = 0;
Marcel Holtmann9fc3bfb2014-07-04 00:46:56 +0200597
598 BT_DBG("sock %p %s", sk, hdev->name);
599
600 hci_dev_lock(hdev);
601
602 memset(&rp, 0, sizeof(rp));
603 rp.manufacturer = cpu_to_le16(hdev->manufacturer);
Marcel Holtmann89bc22d2014-07-04 16:54:37 +0200604
Marcel Holtmanneb1904f2014-07-04 17:23:33 +0200605 if (test_bit(HCI_QUIRK_EXTERNAL_CONFIG, &hdev->quirks))
606 options |= MGMT_OPTION_EXTERNAL_CONFIG;
607
Marcel Holtmann9fc3bfb2014-07-04 00:46:56 +0200608 if (hdev->set_bdaddr)
Marcel Holtmann89bc22d2014-07-04 16:54:37 +0200609 options |= MGMT_OPTION_PUBLIC_ADDRESS;
610
611 rp.supported_options = cpu_to_le32(options);
612 rp.missing_options = get_missing_options(hdev);
Marcel Holtmann9fc3bfb2014-07-04 00:46:56 +0200613
614 hci_dev_unlock(hdev);
615
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200616 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_READ_CONFIG_INFO, 0,
617 &rp, sizeof(rp));
Marcel Holtmann9fc3bfb2014-07-04 00:46:56 +0200618}
619
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200620static u32 get_supported_settings(struct hci_dev *hdev)
Johan Hedberg03811012010-12-08 00:21:06 +0200621{
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200622 u32 settings = 0;
Johan Hedberg03811012010-12-08 00:21:06 +0200623
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200624 settings |= MGMT_SETTING_POWERED;
Johan Hedbergb2939472014-07-30 09:22:23 +0300625 settings |= MGMT_SETTING_BONDABLE;
Marcel Holtmannb1de97d2014-01-31 11:55:21 -0800626 settings |= MGMT_SETTING_DEBUG_KEYS;
Johan Hedberg3742abf2014-07-08 16:07:34 +0300627 settings |= MGMT_SETTING_CONNECTABLE;
628 settings |= MGMT_SETTING_DISCOVERABLE;
Johan Hedberg03811012010-12-08 00:21:06 +0200629
Andre Guedesed3fa312012-07-24 15:03:46 -0300630 if (lmp_bredr_capable(hdev)) {
Johan Hedberg1a47aee2013-03-15 17:07:06 -0500631 if (hdev->hci_ver >= BLUETOOTH_VER_1_2)
632 settings |= MGMT_SETTING_FAST_CONNECTABLE;
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200633 settings |= MGMT_SETTING_BREDR;
634 settings |= MGMT_SETTING_LINK_SECURITY;
Marcel Holtmanna82974c2013-10-11 09:48:47 -0700635
636 if (lmp_ssp_capable(hdev)) {
637 settings |= MGMT_SETTING_SSP;
638 settings |= MGMT_SETTING_HS;
639 }
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800640
Marcel Holtmann05b3c3e2014-12-31 14:43:18 -0800641 if (lmp_sc_capable(hdev))
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800642 settings |= MGMT_SETTING_SECURE_CONN;
Marcel Holtmann848566b2013-10-01 22:59:22 -0700643 }
Marcel Holtmannd7b7e792012-02-20 21:47:49 +0100644
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300645 if (lmp_le_capable(hdev)) {
Marcel Holtmann9d428202012-05-03 07:12:31 +0200646 settings |= MGMT_SETTING_LE;
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300647 settings |= MGMT_SETTING_ADVERTISING;
Johan Hedberga3209692014-05-26 11:23:35 +0300648 settings |= MGMT_SETTING_SECURE_CONN;
Johan Hedberg0f4bd942014-02-22 19:06:35 +0200649 settings |= MGMT_SETTING_PRIVACY;
Marcel Holtmann93690c22015-03-06 10:11:21 -0800650 settings |= MGMT_SETTING_STATIC_ADDRESS;
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300651 }
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200652
Marcel Holtmanneb1904f2014-07-04 17:23:33 +0200653 if (test_bit(HCI_QUIRK_EXTERNAL_CONFIG, &hdev->quirks) ||
654 hdev->set_bdaddr)
Marcel Holtmann9fc3bfb2014-07-04 00:46:56 +0200655 settings |= MGMT_SETTING_CONFIGURATION;
656
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200657 return settings;
658}
Johan Hedbergebc99fe2011-01-04 11:54:26 +0200659
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200660static u32 get_current_settings(struct hci_dev *hdev)
661{
662 u32 settings = 0;
Johan Hedbergdc4fe302011-03-16 14:29:36 +0200663
Johan Hedbergf1f0eb02012-02-21 17:15:41 +0200664 if (hdev_is_powered(hdev))
Marcel Holtmannf0d4b782012-02-21 12:14:25 +0100665 settings |= MGMT_SETTING_POWERED;
666
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700667 if (hci_dev_test_flag(hdev, HCI_CONNECTABLE))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200668 settings |= MGMT_SETTING_CONNECTABLE;
669
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700670 if (hci_dev_test_flag(hdev, HCI_FAST_CONNECTABLE))
Johan Hedberg1a4d3c42013-03-15 17:07:08 -0500671 settings |= MGMT_SETTING_FAST_CONNECTABLE;
672
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700673 if (hci_dev_test_flag(hdev, HCI_DISCOVERABLE))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200674 settings |= MGMT_SETTING_DISCOVERABLE;
675
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700676 if (hci_dev_test_flag(hdev, HCI_BONDABLE))
Johan Hedbergb2939472014-07-30 09:22:23 +0300677 settings |= MGMT_SETTING_BONDABLE;
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200678
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700679 if (hci_dev_test_flag(hdev, HCI_BREDR_ENABLED))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200680 settings |= MGMT_SETTING_BREDR;
681
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700682 if (hci_dev_test_flag(hdev, HCI_LE_ENABLED))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200683 settings |= MGMT_SETTING_LE;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200684
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700685 if (hci_dev_test_flag(hdev, HCI_LINK_SECURITY))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200686 settings |= MGMT_SETTING_LINK_SECURITY;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200687
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700688 if (hci_dev_test_flag(hdev, HCI_SSP_ENABLED))
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200689 settings |= MGMT_SETTING_SSP;
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200690
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700691 if (hci_dev_test_flag(hdev, HCI_HS_ENABLED))
Johan Hedberg6d80dfd2012-02-20 23:50:38 +0200692 settings |= MGMT_SETTING_HS;
693
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700694 if (hci_dev_test_flag(hdev, HCI_ADVERTISING))
Johan Hedbergeeca6f82013-09-25 13:26:09 +0300695 settings |= MGMT_SETTING_ADVERTISING;
696
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700697 if (hci_dev_test_flag(hdev, HCI_SC_ENABLED))
Marcel Holtmanne98d2ce2014-01-10 02:07:22 -0800698 settings |= MGMT_SETTING_SECURE_CONN;
699
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700700 if (hci_dev_test_flag(hdev, HCI_KEEP_DEBUG_KEYS))
Marcel Holtmannb1de97d2014-01-31 11:55:21 -0800701 settings |= MGMT_SETTING_DEBUG_KEYS;
702
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700703 if (hci_dev_test_flag(hdev, HCI_PRIVACY))
Johan Hedberg0f4bd942014-02-22 19:06:35 +0200704 settings |= MGMT_SETTING_PRIVACY;
705
Marcel Holtmann93690c22015-03-06 10:11:21 -0800706 /* The current setting for static address has two purposes. The
707 * first is to indicate if the static address will be used and
708 * the second is to indicate if it is actually set.
709 *
710 * This means if the static address is not configured, this flag
Marcel Holtmann08dc0e92015-03-25 18:32:13 -0700711 * will never be set. If the address is configured, then if the
Marcel Holtmann93690c22015-03-06 10:11:21 -0800712 * address is actually used decides if the flag is set or not.
713 *
714 * For single mode LE only controllers and dual-mode controllers
715 * with BR/EDR disabled, the existence of the static address will
716 * be evaluated.
717 */
Marcel Holtmannb7cb93e2015-03-13 10:20:35 -0700718 if (hci_dev_test_flag(hdev, HCI_FORCE_STATIC_ADDR) ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700719 !hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) ||
Marcel Holtmann93690c22015-03-06 10:11:21 -0800720 !bacmp(&hdev->bdaddr, BDADDR_ANY)) {
721 if (bacmp(&hdev->static_addr, BDADDR_ANY))
722 settings |= MGMT_SETTING_STATIC_ADDRESS;
723 }
724
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200725 return settings;
Johan Hedbergc542a062011-01-26 13:11:03 +0200726}
727
Johan Hedberg333ae952015-03-17 13:48:47 +0200728static struct mgmt_pending_cmd *pending_find(u16 opcode, struct hci_dev *hdev)
729{
730 return mgmt_pending_find(HCI_CHANNEL_CONTROL, opcode, hdev);
731}
732
Johan Hedberg333ae952015-03-17 13:48:47 +0200733static struct mgmt_pending_cmd *pending_find_data(u16 opcode,
734 struct hci_dev *hdev,
735 const void *data)
736{
737 return mgmt_pending_find_data(HCI_CHANNEL_CONTROL, opcode, hdev, data);
738}
739
Johan Hedbergf2252572015-11-18 12:49:20 +0200740u8 mgmt_get_adv_discov_flags(struct hci_dev *hdev)
Johan Hedberg9a43e252013-10-20 19:00:07 +0300741{
Johan Hedberg3b0602c2015-03-06 21:08:55 +0200742 struct mgmt_pending_cmd *cmd;
Johan Hedberg9a43e252013-10-20 19:00:07 +0300743
744 /* If there's a pending mgmt command the flags will not yet have
745 * their final values, so check for this first.
746 */
Johan Hedberg333ae952015-03-17 13:48:47 +0200747 cmd = pending_find(MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedberg9a43e252013-10-20 19:00:07 +0300748 if (cmd) {
749 struct mgmt_mode *cp = cmd->param;
750 if (cp->val == 0x01)
751 return LE_AD_GENERAL;
752 else if (cp->val == 0x02)
753 return LE_AD_LIMITED;
754 } else {
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700755 if (hci_dev_test_flag(hdev, HCI_LIMITED_DISCOVERABLE))
Johan Hedberg9a43e252013-10-20 19:00:07 +0300756 return LE_AD_LIMITED;
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700757 else if (hci_dev_test_flag(hdev, HCI_DISCOVERABLE))
Johan Hedberg9a43e252013-10-20 19:00:07 +0300758 return LE_AD_GENERAL;
759 }
760
761 return 0;
762}
763
Johan Hedbergf2252572015-11-18 12:49:20 +0200764bool mgmt_get_connectable(struct hci_dev *hdev)
Arman Uguraye7a685d2015-03-25 18:53:40 -0700765{
766 struct mgmt_pending_cmd *cmd;
767
768 /* If there's a pending mgmt command the flag will not yet have
769 * it's final value, so check for this first.
770 */
771 cmd = pending_find(MGMT_OP_SET_CONNECTABLE, hdev);
772 if (cmd) {
773 struct mgmt_mode *cp = cmd->param;
774
775 return cp->val;
776 }
777
778 return hci_dev_test_flag(hdev, HCI_CONNECTABLE);
779}
780
Johan Hedberg7d785252011-12-15 00:47:39 +0200781static void service_cache_off(struct work_struct *work)
782{
783 struct hci_dev *hdev = container_of(work, struct hci_dev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300784 service_cache.work);
Johan Hedberg890ea892013-03-15 17:06:52 -0500785 struct hci_request req;
Johan Hedberg7d785252011-12-15 00:47:39 +0200786
Marcel Holtmanna69d8922015-03-13 02:11:05 -0700787 if (!hci_dev_test_and_clear_flag(hdev, HCI_SERVICE_CACHE))
Johan Hedberg7d785252011-12-15 00:47:39 +0200788 return;
789
Johan Hedberg890ea892013-03-15 17:06:52 -0500790 hci_req_init(&req, hdev);
791
Johan Hedberg7d785252011-12-15 00:47:39 +0200792 hci_dev_lock(hdev);
793
Johan Hedbergb1a89172015-11-25 16:15:42 +0200794 __hci_req_update_eir(&req);
Johan Hedberg14bf5ea2015-11-22 19:00:22 +0200795 __hci_req_update_class(&req);
Johan Hedberg7d785252011-12-15 00:47:39 +0200796
797 hci_dev_unlock(hdev);
Johan Hedberg890ea892013-03-15 17:06:52 -0500798
799 hci_req_run(&req, NULL);
Johan Hedberg7d785252011-12-15 00:47:39 +0200800}
801
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200802static void rpa_expired(struct work_struct *work)
803{
804 struct hci_dev *hdev = container_of(work, struct hci_dev,
805 rpa_expired.work);
806 struct hci_request req;
807
808 BT_DBG("");
809
Marcel Holtmanna1536da2015-03-13 02:11:01 -0700810 hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200811
Marcel Holtmannd7a5a112015-03-13 02:11:00 -0700812 if (!hci_dev_test_flag(hdev, HCI_ADVERTISING))
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200813 return;
814
815 /* The generation of a new RPA and programming it into the
Johan Hedbergf2252572015-11-18 12:49:20 +0200816 * controller happens in the hci_req_enable_advertising()
817 * function.
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200818 */
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200819 hci_req_init(&req, hdev);
Johan Hedbergf2252572015-11-18 12:49:20 +0200820 __hci_req_enable_advertising(&req);
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200821 hci_req_run(&req, NULL);
822}
823
Johan Hedberg6a919082012-02-28 06:17:26 +0200824static void mgmt_init_hdev(struct sock *sk, struct hci_dev *hdev)
Johan Hedberg7d785252011-12-15 00:47:39 +0200825{
Marcel Holtmann238be782015-03-13 02:11:06 -0700826 if (hci_dev_test_and_set_flag(hdev, HCI_MGMT))
Johan Hedberg6a919082012-02-28 06:17:26 +0200827 return;
828
Johan Hedberg4f87da82012-03-02 19:55:56 +0200829 INIT_DELAYED_WORK(&hdev->service_cache, service_cache_off);
Johan Hedbergd6bfd592014-02-23 19:42:20 +0200830 INIT_DELAYED_WORK(&hdev->rpa_expired, rpa_expired);
Johan Hedberg7d785252011-12-15 00:47:39 +0200831
Johan Hedberg4f87da82012-03-02 19:55:56 +0200832 /* Non-mgmt controlled devices get this bit set
833 * implicitly so that pairing works for them, however
834 * for mgmt we require user-space to explicitly enable
835 * it
836 */
Marcel Holtmanna358dc12015-03-13 02:11:02 -0700837 hci_dev_clear_flag(hdev, HCI_BONDABLE);
Johan Hedberg7d785252011-12-15 00:47:39 +0200838}
839
Johan Hedberg0f4e68c2012-02-28 17:18:30 +0200840static int read_controller_info(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300841 void *data, u16 data_len)
Johan Hedberg03811012010-12-08 00:21:06 +0200842{
843 struct mgmt_rp_read_info rp;
Johan Hedberg03811012010-12-08 00:21:06 +0200844
Johan Hedbergbdb6d972012-02-28 06:13:32 +0200845 BT_DBG("sock %p %s", sk, hdev->name);
Johan Hedberg03811012010-12-08 00:21:06 +0200846
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300847 hci_dev_lock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +0200848
Johan Hedberg03811012010-12-08 00:21:06 +0200849 memset(&rp, 0, sizeof(rp));
850
Johan Hedberg03811012010-12-08 00:21:06 +0200851 bacpy(&rp.bdaddr, &hdev->bdaddr);
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200852
853 rp.version = hdev->hci_ver;
Marcel Holtmanneb55ef02012-03-14 18:08:46 +0200854 rp.manufacturer = cpu_to_le16(hdev->manufacturer);
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200855
856 rp.supported_settings = cpu_to_le32(get_supported_settings(hdev));
857 rp.current_settings = cpu_to_le32(get_current_settings(hdev));
858
859 memcpy(rp.dev_class, hdev->dev_class, 3);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200860
861 memcpy(rp.name, hdev->dev_name, sizeof(hdev->dev_name));
Johan Hedberg27fcc362012-02-22 21:46:22 +0200862 memcpy(rp.short_name, hdev->short_name, sizeof(hdev->short_name));
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200863
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300864 hci_dev_unlock(hdev);
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200865
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200866 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_READ_INFO, 0, &rp,
867 sizeof(rp));
Johan Hedbergf7b64e62010-12-13 21:07:06 +0200868}
869
MichaƂ Narajowski8a0c9f42016-09-01 16:46:24 +0200870static inline u16 eir_append_data(u8 *eir, u16 eir_len, u8 type, u8 *data,
871 u8 data_len)
872{
873 eir[eir_len++] = sizeof(type) + data_len;
874 eir[eir_len++] = type;
875 memcpy(&eir[eir_len], data, data_len);
876 eir_len += data_len;
877
878 return eir_len;
879}
880
Szymon Janc6a9e90b2016-09-19 20:25:54 +0200881static inline u16 eir_append_le16(u8 *eir, u16 eir_len, u8 type, u16 data)
882{
883 eir[eir_len++] = sizeof(type) + sizeof(data);
884 eir[eir_len++] = type;
885 put_unaligned_le16(data, &eir[eir_len]);
886 eir_len += sizeof(data);
887
888 return eir_len;
889}
890
MichaƂ Narajowskicde7a862016-09-19 20:25:53 +0200891static u16 append_eir_data_to_buf(struct hci_dev *hdev, u8 *eir)
892{
893 u16 eir_len = 0;
894 size_t name_len;
895
896 if (hci_dev_test_flag(hdev, HCI_BREDR_ENABLED))
897 eir_len = eir_append_data(eir, eir_len, EIR_CLASS_OF_DEV,
898 hdev->dev_class, 3);
899
Szymon Janc6a9e90b2016-09-19 20:25:54 +0200900 if (hci_dev_test_flag(hdev, HCI_LE_ENABLED))
901 eir_len = eir_append_le16(eir, eir_len, EIR_APPEARANCE,
902 hdev->appearance);
903
MichaƂ Narajowskicde7a862016-09-19 20:25:53 +0200904 name_len = strlen(hdev->dev_name);
905 eir_len = eir_append_data(eir, eir_len, EIR_NAME_COMPLETE,
906 hdev->dev_name, name_len);
907
908 name_len = strlen(hdev->short_name);
909 eir_len = eir_append_data(eir, eir_len, EIR_NAME_SHORT,
910 hdev->short_name, name_len);
911
912 return eir_len;
913}
914
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200915static int read_ext_controller_info(struct sock *sk, struct hci_dev *hdev,
916 void *data, u16 data_len)
917{
Szymon Janc7d5c11d2016-09-19 20:25:52 +0200918 char buf[512];
919 struct mgmt_rp_read_ext_info *rp = (void *)buf;
MichaƂ Narajowskicde7a862016-09-19 20:25:53 +0200920 u16 eir_len;
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200921
922 BT_DBG("sock %p %s", sk, hdev->name);
923
Szymon Janc7d5c11d2016-09-19 20:25:52 +0200924 memset(&buf, 0, sizeof(buf));
925
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200926 hci_dev_lock(hdev);
927
MichaƂ Narajowski8a0c9f42016-09-01 16:46:24 +0200928 bacpy(&rp->bdaddr, &hdev->bdaddr);
929
930 rp->version = hdev->hci_ver;
931 rp->manufacturer = cpu_to_le16(hdev->manufacturer);
932
933 rp->supported_settings = cpu_to_le32(get_supported_settings(hdev));
934 rp->current_settings = cpu_to_le32(get_current_settings(hdev));
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200935
Szymon Janc7d5c11d2016-09-19 20:25:52 +0200936
MichaƂ Narajowskicde7a862016-09-19 20:25:53 +0200937 eir_len = append_eir_data_to_buf(hdev, rp->eir);
Szymon Janc7d5c11d2016-09-19 20:25:52 +0200938 rp->eir_len = cpu_to_le16(eir_len);
939
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200940 hci_dev_unlock(hdev);
941
942 /* If this command is called at least once, then the events
943 * for class of device and local name changes are disabled
944 * and only the new extended controller information event
945 * is used.
946 */
947 hci_sock_set_flag(sk, HCI_MGMT_EXT_INFO_EVENTS);
948 hci_sock_clear_flag(sk, HCI_MGMT_DEV_CLASS_EVENTS);
949 hci_sock_clear_flag(sk, HCI_MGMT_LOCAL_NAME_EVENTS);
950
MichaƂ Narajowski8a0c9f42016-09-01 16:46:24 +0200951 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_READ_EXT_INFO, 0, rp,
952 sizeof(*rp) + eir_len);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200953}
954
955static int ext_info_changed(struct hci_dev *hdev, struct sock *skip)
956{
MichaƂ Narajowski5e9fae42016-09-19 20:25:55 +0200957 char buf[512];
958 struct mgmt_ev_ext_info_changed *ev = (void *)buf;
959 u16 eir_len;
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200960
MichaƂ Narajowski5e9fae42016-09-19 20:25:55 +0200961 memset(buf, 0, sizeof(buf));
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200962
MichaƂ Narajowski5e9fae42016-09-19 20:25:55 +0200963 eir_len = append_eir_data_to_buf(hdev, ev->eir);
964 ev->eir_len = cpu_to_le16(eir_len);
965
966 return mgmt_limited_event(MGMT_EV_EXT_INFO_CHANGED, hdev, ev,
967 sizeof(*ev) + eir_len,
968 HCI_MGMT_EXT_INFO_EVENTS, skip);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +0200969}
970
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200971static int send_settings_rsp(struct sock *sk, u16 opcode, struct hci_dev *hdev)
Johan Hedberg86805702011-11-11 16:18:52 +0200972{
Johan Hedberg69ab39e2011-12-15 00:47:35 +0200973 __le32 settings = cpu_to_le32(get_current_settings(hdev));
Johan Hedberg86805702011-11-11 16:18:52 +0200974
Johan Hedberg2a1afb52015-03-06 21:08:54 +0200975 return mgmt_cmd_complete(sk, hdev->id, opcode, 0, &settings,
976 sizeof(settings));
Johan Hedberg86805702011-11-11 16:18:52 +0200977}
978
Marcel Holtmann1904a852015-01-11 13:50:44 -0800979static void clean_up_hci_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg8b064a32014-02-24 14:52:22 +0200980{
981 BT_DBG("%s status 0x%02x", hdev->name, status);
982
Johan Hedberga3172b72014-02-28 09:33:44 +0200983 if (hci_conn_count(hdev) == 0) {
984 cancel_delayed_work(&hdev->power_off);
Johan Hedberg8b064a32014-02-24 14:52:22 +0200985 queue_work(hdev->req_workqueue, &hdev->power_off.work);
Johan Hedberga3172b72014-02-28 09:33:44 +0200986 }
Johan Hedberg8b064a32014-02-24 14:52:22 +0200987}
988
Johan Hedbergf2252572015-11-18 12:49:20 +0200989void mgmt_advertising_added(struct sock *sk, struct hci_dev *hdev, u8 instance)
Arman Uguray912098a2015-03-23 15:57:15 -0700990{
991 struct mgmt_ev_advertising_added ev;
992
993 ev.instance = instance;
994
995 mgmt_event(MGMT_EV_ADVERTISING_ADDED, hdev, &ev, sizeof(ev), sk);
996}
997
Johan Hedbergf2252572015-11-18 12:49:20 +0200998void mgmt_advertising_removed(struct sock *sk, struct hci_dev *hdev,
999 u8 instance)
Arman Uguray912098a2015-03-23 15:57:15 -07001000{
1001 struct mgmt_ev_advertising_removed ev;
1002
1003 ev.instance = instance;
1004
1005 mgmt_event(MGMT_EV_ADVERTISING_REMOVED, hdev, &ev, sizeof(ev), sk);
1006}
1007
Florian Grandel7816b822015-06-18 03:16:45 +02001008static void cancel_adv_timeout(struct hci_dev *hdev)
1009{
1010 if (hdev->adv_instance_timeout) {
1011 hdev->adv_instance_timeout = 0;
1012 cancel_delayed_work(&hdev->adv_instance_expire);
1013 }
1014}
1015
Johan Hedberg8b064a32014-02-24 14:52:22 +02001016static int clean_up_hci_state(struct hci_dev *hdev)
1017{
1018 struct hci_request req;
1019 struct hci_conn *conn;
Johan Hedberg23a48092014-07-08 16:05:06 +03001020 bool discov_stopped;
1021 int err;
Johan Hedberg8b064a32014-02-24 14:52:22 +02001022
1023 hci_req_init(&req, hdev);
1024
1025 if (test_bit(HCI_ISCAN, &hdev->flags) ||
1026 test_bit(HCI_PSCAN, &hdev->flags)) {
1027 u8 scan = 0x00;
1028 hci_req_add(&req, HCI_OP_WRITE_SCAN_ENABLE, 1, &scan);
1029 }
1030
Johan Hedberg37d3a1f2016-08-28 20:53:34 +03001031 hci_req_clear_adv_instance(hdev, NULL, NULL, 0x00, false);
Arman Uguray912098a2015-03-23 15:57:15 -07001032
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001033 if (hci_dev_test_flag(hdev, HCI_LE_ADV))
Johan Hedbergf2252572015-11-18 12:49:20 +02001034 __hci_req_disable_advertising(&req);
Johan Hedberg8b064a32014-02-24 14:52:22 +02001035
Johan Hedberg2154d3f2015-11-11 08:30:45 +02001036 discov_stopped = hci_req_stop_discovery(&req);
Johan Hedberg8b064a32014-02-24 14:52:22 +02001037
1038 list_for_each_entry(conn, &hdev->conn_hash.list, list) {
Johan Hedberg89e0ccc2015-10-22 10:49:38 +03001039 /* 0x15 == Terminated due to Power Off */
1040 __hci_abort_conn(&req, conn, 0x15);
Johan Hedberg8b064a32014-02-24 14:52:22 +02001041 }
1042
Johan Hedberg23a48092014-07-08 16:05:06 +03001043 err = hci_req_run(&req, clean_up_hci_complete);
1044 if (!err && discov_stopped)
1045 hci_discovery_set_state(hdev, DISCOVERY_STOPPING);
1046
1047 return err;
Johan Hedberg8b064a32014-02-24 14:52:22 +02001048}
1049
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001050static int set_powered(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001051 u16 len)
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001052{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001053 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001054 struct mgmt_pending_cmd *cmd;
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001055 int err;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001056
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001057 BT_DBG("request for %s", hdev->name);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001058
Johan Hedberga7e80f22013-01-09 16:05:19 +02001059 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02001060 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_POWERED,
1061 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001062
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001063 hci_dev_lock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001064
Johan Hedberg333ae952015-03-17 13:48:47 +02001065 if (pending_find(MGMT_OP_SET_POWERED, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001066 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_POWERED,
1067 MGMT_STATUS_BUSY);
Johan Hedberg87b95ba2013-09-25 13:26:06 +03001068 goto failed;
1069 }
1070
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001071 if (!!cp->val == hdev_is_powered(hdev)) {
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001072 err = send_settings_rsp(sk, MGMT_OP_SET_POWERED, hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001073 goto failed;
1074 }
1075
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001076 cmd = mgmt_pending_add(sk, MGMT_OP_SET_POWERED, hdev, data, len);
1077 if (!cmd) {
1078 err = -ENOMEM;
1079 goto failed;
1080 }
1081
Johan Hedberg8b064a32014-02-24 14:52:22 +02001082 if (cp->val) {
Johan Hedberg19202572013-01-14 22:33:51 +02001083 queue_work(hdev->req_workqueue, &hdev->power_on);
Johan Hedberg8b064a32014-02-24 14:52:22 +02001084 err = 0;
1085 } else {
1086 /* Disconnect connections, stop scans, etc */
1087 err = clean_up_hci_state(hdev);
Johan Hedberga3172b72014-02-28 09:33:44 +02001088 if (!err)
1089 queue_delayed_work(hdev->req_workqueue, &hdev->power_off,
1090 HCI_POWER_OFF_TIMEOUT);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001091
Johan Hedberg8b064a32014-02-24 14:52:22 +02001092 /* ENODATA means there were no HCI commands queued */
1093 if (err == -ENODATA) {
Johan Hedberga3172b72014-02-28 09:33:44 +02001094 cancel_delayed_work(&hdev->power_off);
Johan Hedberg8b064a32014-02-24 14:52:22 +02001095 queue_work(hdev->req_workqueue, &hdev->power_off.work);
1096 err = 0;
1097 }
1098 }
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001099
1100failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001101 hci_dev_unlock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001102 return err;
1103}
1104
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001105static int new_settings(struct hci_dev *hdev, struct sock *skip)
1106{
Marcel Holtmannf6b77122015-03-14 19:28:05 -07001107 __le32 ev = cpu_to_le32(get_current_settings(hdev));
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001108
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02001109 return mgmt_limited_event(MGMT_EV_NEW_SETTINGS, hdev, &ev,
1110 sizeof(ev), HCI_MGMT_SETTING_EVENTS, skip);
Johan Hedbergbeadb2b2012-02-21 16:55:31 +02001111}
1112
Johan Hedberg91a668b2014-07-09 13:28:26 +03001113int mgmt_new_settings(struct hci_dev *hdev)
1114{
1115 return new_settings(hdev, NULL);
1116}
1117
Johan Hedbergbd99abd2013-09-25 13:26:07 +03001118struct cmd_lookup {
1119 struct sock *sk;
1120 struct hci_dev *hdev;
1121 u8 mgmt_status;
1122};
1123
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001124static void settings_rsp(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedbergbd99abd2013-09-25 13:26:07 +03001125{
1126 struct cmd_lookup *match = data;
1127
1128 send_settings_rsp(cmd->sk, cmd->opcode, match->hdev);
1129
1130 list_del(&cmd->list);
1131
1132 if (match->sk == NULL) {
1133 match->sk = cmd->sk;
1134 sock_hold(match->sk);
1135 }
1136
1137 mgmt_pending_free(cmd);
1138}
1139
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001140static void cmd_status_rsp(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedbergbd99abd2013-09-25 13:26:07 +03001141{
1142 u8 *status = data;
1143
Johan Hedberga69e8372015-03-06 21:08:53 +02001144 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode, *status);
Johan Hedbergbd99abd2013-09-25 13:26:07 +03001145 mgmt_pending_remove(cmd);
1146}
1147
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001148static void cmd_complete_rsp(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedberg1b9b5ee2014-12-05 13:36:00 +02001149{
1150 if (cmd->cmd_complete) {
1151 u8 *status = data;
1152
1153 cmd->cmd_complete(cmd, *status);
1154 mgmt_pending_remove(cmd);
1155
1156 return;
1157 }
1158
1159 cmd_status_rsp(cmd, data);
1160}
1161
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001162static int generic_cmd_complete(struct mgmt_pending_cmd *cmd, u8 status)
Johan Hedbergf5818c22014-12-05 13:36:02 +02001163{
Johan Hedberg2a1afb52015-03-06 21:08:54 +02001164 return mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, status,
1165 cmd->param, cmd->param_len);
Johan Hedbergf5818c22014-12-05 13:36:02 +02001166}
1167
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001168static int addr_cmd_complete(struct mgmt_pending_cmd *cmd, u8 status)
Johan Hedberg7776d1d2014-12-05 13:36:03 +02001169{
Johan Hedberg2a1afb52015-03-06 21:08:54 +02001170 return mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, status,
1171 cmd->param, sizeof(struct mgmt_addr_info));
Johan Hedberg7776d1d2014-12-05 13:36:03 +02001172}
1173
Johan Hedberge6fe7982013-10-02 15:45:22 +03001174static u8 mgmt_bredr_support(struct hci_dev *hdev)
1175{
1176 if (!lmp_bredr_capable(hdev))
1177 return MGMT_STATUS_NOT_SUPPORTED;
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001178 else if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED))
Johan Hedberge6fe7982013-10-02 15:45:22 +03001179 return MGMT_STATUS_REJECTED;
1180 else
1181 return MGMT_STATUS_SUCCESS;
1182}
1183
1184static u8 mgmt_le_support(struct hci_dev *hdev)
1185{
1186 if (!lmp_le_capable(hdev))
1187 return MGMT_STATUS_NOT_SUPPORTED;
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001188 else if (!hci_dev_test_flag(hdev, HCI_LE_ENABLED))
Johan Hedberge6fe7982013-10-02 15:45:22 +03001189 return MGMT_STATUS_REJECTED;
1190 else
1191 return MGMT_STATUS_SUCCESS;
1192}
1193
Johan Hedbergaed1a882015-11-22 17:24:44 +03001194void mgmt_set_discoverable_complete(struct hci_dev *hdev, u8 status)
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001195{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001196 struct mgmt_pending_cmd *cmd;
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001197
1198 BT_DBG("status 0x%02x", status);
1199
1200 hci_dev_lock(hdev);
1201
Johan Hedberg333ae952015-03-17 13:48:47 +02001202 cmd = pending_find(MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001203 if (!cmd)
1204 goto unlock;
1205
1206 if (status) {
1207 u8 mgmt_err = mgmt_status(status);
Johan Hedberga69e8372015-03-06 21:08:53 +02001208 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
Marcel Holtmanna358dc12015-03-13 02:11:02 -07001209 hci_dev_clear_flag(hdev, HCI_LIMITED_DISCOVERABLE);
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001210 goto remove_cmd;
1211 }
1212
Johan Hedbergaed1a882015-11-22 17:24:44 +03001213 if (hci_dev_test_flag(hdev, HCI_DISCOVERABLE) &&
1214 hdev->discov_timeout > 0) {
1215 int to = msecs_to_jiffies(hdev->discov_timeout * 1000);
1216 queue_delayed_work(hdev->req_workqueue, &hdev->discov_off, to);
Marcel Holtmannd4462a02013-10-15 08:11:02 -07001217 }
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001218
1219 send_settings_rsp(cmd->sk, MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedbergaed1a882015-11-22 17:24:44 +03001220 new_settings(hdev, cmd->sk);
Marcel Holtmann970ba522013-10-15 06:33:57 -07001221
Johan Hedbergbfaf8c92013-10-14 21:15:27 +03001222remove_cmd:
1223 mgmt_pending_remove(cmd);
1224
1225unlock:
1226 hci_dev_unlock(hdev);
1227}
1228
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001229static int set_discoverable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001230 u16 len)
Johan Hedberg73f22f62010-12-29 16:00:25 +02001231{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001232 struct mgmt_cp_set_discoverable *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001233 struct mgmt_pending_cmd *cmd;
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001234 u16 timeout;
Johan Hedberg73f22f62010-12-29 16:00:25 +02001235 int err;
1236
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001237 BT_DBG("request for %s", hdev->name);
Johan Hedberg73f22f62010-12-29 16:00:25 +02001238
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001239 if (!hci_dev_test_flag(hdev, HCI_LE_ENABLED) &&
1240 !hci_dev_test_flag(hdev, HCI_BREDR_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02001241 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1242 MGMT_STATUS_REJECTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001243
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001244 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02001245 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1246 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001247
Marcel Holtmann1f350c82012-03-12 20:31:08 -07001248 timeout = __le16_to_cpu(cp->timeout);
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001249
1250 /* Disabling discoverable requires that no timeout is set,
1251 * and enabling limited discoverable requires a timeout.
1252 */
1253 if ((cp->val == 0x00 && timeout > 0) ||
1254 (cp->val == 0x02 && timeout == 0))
Johan Hedberga69e8372015-03-06 21:08:53 +02001255 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1256 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001257
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001258 hci_dev_lock(hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001259
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001260 if (!hdev_is_powered(hdev) && timeout > 0) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001261 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1262 MGMT_STATUS_NOT_POWERED);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001263 goto failed;
1264 }
1265
Johan Hedberg333ae952015-03-17 13:48:47 +02001266 if (pending_find(MGMT_OP_SET_DISCOVERABLE, hdev) ||
1267 pending_find(MGMT_OP_SET_CONNECTABLE, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001268 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1269 MGMT_STATUS_BUSY);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001270 goto failed;
1271 }
1272
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001273 if (!hci_dev_test_flag(hdev, HCI_CONNECTABLE)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001274 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DISCOVERABLE,
1275 MGMT_STATUS_REJECTED);
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001276 goto failed;
1277 }
1278
1279 if (!hdev_is_powered(hdev)) {
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001280 bool changed = false;
1281
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001282 /* Setting limited discoverable when powered off is
1283 * not a valid operation since it requires a timeout
1284 * and so no need to check HCI_LIMITED_DISCOVERABLE.
1285 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001286 if (!!cp->val != hci_dev_test_flag(hdev, HCI_DISCOVERABLE)) {
Marcel Holtmannce05d602015-03-13 02:11:03 -07001287 hci_dev_change_flag(hdev, HCI_DISCOVERABLE);
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001288 changed = true;
1289 }
1290
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001291 err = send_settings_rsp(sk, MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedberg0224d2f2012-02-21 19:40:05 +02001292 if (err < 0)
1293 goto failed;
1294
1295 if (changed)
1296 err = new_settings(hdev, sk);
1297
Johan Hedberg5e5282b2012-02-21 16:01:30 +02001298 goto failed;
1299 }
1300
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001301 /* If the current mode is the same, then just update the timeout
1302 * value with the new value. And if only the timeout gets updated,
1303 * then no need for any HCI transactions.
1304 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001305 if (!!cp->val == hci_dev_test_flag(hdev, HCI_DISCOVERABLE) &&
1306 (cp->val == 0x02) == hci_dev_test_flag(hdev,
1307 HCI_LIMITED_DISCOVERABLE)) {
Marcel Holtmann36261542013-10-15 08:28:51 -07001308 cancel_delayed_work(&hdev->discov_off);
1309 hdev->discov_timeout = timeout;
Marcel Holtmann955638e2012-02-22 18:21:00 +01001310
Marcel Holtmann36261542013-10-15 08:28:51 -07001311 if (cp->val && hdev->discov_timeout > 0) {
1312 int to = msecs_to_jiffies(hdev->discov_timeout * 1000);
Johan Hedbergc366f552015-11-23 15:43:06 +02001313 queue_delayed_work(hdev->req_workqueue,
1314 &hdev->discov_off, to);
Marcel Holtmann955638e2012-02-22 18:21:00 +01001315 }
1316
Johan Hedberg69ab39e2011-12-15 00:47:35 +02001317 err = send_settings_rsp(sk, MGMT_OP_SET_DISCOVERABLE, hdev);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001318 goto failed;
1319 }
1320
1321 cmd = mgmt_pending_add(sk, MGMT_OP_SET_DISCOVERABLE, hdev, data, len);
1322 if (!cmd) {
1323 err = -ENOMEM;
1324 goto failed;
1325 }
1326
Marcel Holtmann310a3d42013-10-15 09:13:39 -07001327 /* Cancel any potential discoverable timeout that might be
1328 * still active and store new timeout value. The arming of
1329 * the timeout happens in the complete handler.
1330 */
1331 cancel_delayed_work(&hdev->discov_off);
1332 hdev->discov_timeout = timeout;
1333
Johan Hedbergaed1a882015-11-22 17:24:44 +03001334 if (cp->val)
1335 hci_dev_set_flag(hdev, HCI_DISCOVERABLE);
1336 else
1337 hci_dev_clear_flag(hdev, HCI_DISCOVERABLE);
1338
Johan Hedbergb456f872013-10-19 23:38:22 +03001339 /* Limited discoverable mode */
1340 if (cp->val == 0x02)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07001341 hci_dev_set_flag(hdev, HCI_LIMITED_DISCOVERABLE);
Johan Hedbergb456f872013-10-19 23:38:22 +03001342 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07001343 hci_dev_clear_flag(hdev, HCI_LIMITED_DISCOVERABLE);
Johan Hedbergb456f872013-10-19 23:38:22 +03001344
Johan Hedbergaed1a882015-11-22 17:24:44 +03001345 queue_work(hdev->req_workqueue, &hdev->discoverable_update);
1346 err = 0;
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001347
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02001348failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001349 hci_dev_unlock(hdev);
Johan Hedberg03811012010-12-08 00:21:06 +02001350 return err;
1351}
Johan Hedberg73f22f62010-12-29 16:00:25 +02001352
Johan Hedberg53c0ba72015-11-22 16:43:43 +03001353void mgmt_set_connectable_complete(struct hci_dev *hdev, u8 status)
Johan Hedberg2b76f452013-03-15 17:07:04 -05001354{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001355 struct mgmt_pending_cmd *cmd;
Johan Hedberg2b76f452013-03-15 17:07:04 -05001356
1357 BT_DBG("status 0x%02x", status);
1358
1359 hci_dev_lock(hdev);
1360
Johan Hedberg333ae952015-03-17 13:48:47 +02001361 cmd = pending_find(MGMT_OP_SET_CONNECTABLE, hdev);
Johan Hedberg2b76f452013-03-15 17:07:04 -05001362 if (!cmd)
1363 goto unlock;
1364
Johan Hedberg37438c12013-10-14 16:20:05 +03001365 if (status) {
1366 u8 mgmt_err = mgmt_status(status);
Johan Hedberga69e8372015-03-06 21:08:53 +02001367 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
Johan Hedberg37438c12013-10-14 16:20:05 +03001368 goto remove_cmd;
1369 }
1370
Johan Hedberg2b76f452013-03-15 17:07:04 -05001371 send_settings_rsp(cmd->sk, MGMT_OP_SET_CONNECTABLE, hdev);
Johan Hedberg53c0ba72015-11-22 16:43:43 +03001372 new_settings(hdev, cmd->sk);
Johan Hedbergd7b856f2013-10-14 16:20:04 +03001373
Johan Hedberg37438c12013-10-14 16:20:05 +03001374remove_cmd:
Johan Hedberg2b76f452013-03-15 17:07:04 -05001375 mgmt_pending_remove(cmd);
1376
1377unlock:
1378 hci_dev_unlock(hdev);
1379}
1380
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001381static int set_connectable_update_settings(struct hci_dev *hdev,
1382 struct sock *sk, u8 val)
1383{
1384 bool changed = false;
1385 int err;
1386
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001387 if (!!val != hci_dev_test_flag(hdev, HCI_CONNECTABLE))
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001388 changed = true;
1389
1390 if (val) {
Marcel Holtmanna1536da2015-03-13 02:11:01 -07001391 hci_dev_set_flag(hdev, HCI_CONNECTABLE);
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001392 } else {
Marcel Holtmanna358dc12015-03-13 02:11:02 -07001393 hci_dev_clear_flag(hdev, HCI_CONNECTABLE);
1394 hci_dev_clear_flag(hdev, HCI_DISCOVERABLE);
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001395 }
1396
1397 err = send_settings_rsp(sk, MGMT_OP_SET_CONNECTABLE, hdev);
1398 if (err < 0)
1399 return err;
1400
Johan Hedberg562064e2014-07-08 16:35:34 +03001401 if (changed) {
Johan Hedberg01b1cb82015-11-16 12:52:21 +02001402 hci_req_update_scan(hdev);
Johan Hedberg562064e2014-07-08 16:35:34 +03001403 hci_update_background_scan(hdev);
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001404 return new_settings(hdev, sk);
Johan Hedberg562064e2014-07-08 16:35:34 +03001405 }
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001406
1407 return 0;
1408}
1409
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001410static int set_connectable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001411 u16 len)
Johan Hedberg73f22f62010-12-29 16:00:25 +02001412{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001413 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001414 struct mgmt_pending_cmd *cmd;
Johan Hedberg73f22f62010-12-29 16:00:25 +02001415 int err;
1416
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001417 BT_DBG("request for %s", hdev->name);
Johan Hedberg73f22f62010-12-29 16:00:25 +02001418
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001419 if (!hci_dev_test_flag(hdev, HCI_LE_ENABLED) &&
1420 !hci_dev_test_flag(hdev, HCI_BREDR_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02001421 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
1422 MGMT_STATUS_REJECTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001423
Johan Hedberga7e80f22013-01-09 16:05:19 +02001424 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02001425 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
1426 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001427
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001428 hci_dev_lock(hdev);
Johan Hedberg73f22f62010-12-29 16:00:25 +02001429
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001430 if (!hdev_is_powered(hdev)) {
Johan Hedberge8ba3a12013-10-19 23:38:18 +03001431 err = set_connectable_update_settings(hdev, sk, cp->val);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02001432 goto failed;
Johan Hedberg73f22f62010-12-29 16:00:25 +02001433 }
1434
Johan Hedberg333ae952015-03-17 13:48:47 +02001435 if (pending_find(MGMT_OP_SET_DISCOVERABLE, hdev) ||
1436 pending_find(MGMT_OP_SET_CONNECTABLE, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001437 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_CONNECTABLE,
1438 MGMT_STATUS_BUSY);
Johan Hedberg73f22f62010-12-29 16:00:25 +02001439 goto failed;
1440 }
1441
Johan Hedberg73f22f62010-12-29 16:00:25 +02001442 cmd = mgmt_pending_add(sk, MGMT_OP_SET_CONNECTABLE, hdev, data, len);
1443 if (!cmd) {
1444 err = -ENOMEM;
Johan Hedberg72a734e2010-12-30 00:38:22 +02001445 goto failed;
Johan Hedberg73f22f62010-12-29 16:00:25 +02001446 }
1447
Johan Hedberg53c0ba72015-11-22 16:43:43 +03001448 if (cp->val) {
1449 hci_dev_set_flag(hdev, HCI_CONNECTABLE);
1450 } else {
1451 if (hdev->discov_timeout > 0)
1452 cancel_delayed_work(&hdev->discov_off);
Johan Hedberg2b76f452013-03-15 17:07:04 -05001453
Johan Hedberg53c0ba72015-11-22 16:43:43 +03001454 hci_dev_clear_flag(hdev, HCI_LIMITED_DISCOVERABLE);
1455 hci_dev_clear_flag(hdev, HCI_DISCOVERABLE);
1456 hci_dev_clear_flag(hdev, HCI_CONNECTABLE);
Johan Hedberg9b742462013-10-14 16:20:03 +03001457 }
Johan Hedberg2b76f452013-03-15 17:07:04 -05001458
Johan Hedberg53c0ba72015-11-22 16:43:43 +03001459 queue_work(hdev->req_workqueue, &hdev->connectable_update);
1460 err = 0;
Johan Hedberg73f22f62010-12-29 16:00:25 +02001461
1462failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001463 hci_dev_unlock(hdev);
Johan Hedberg72a734e2010-12-30 00:38:22 +02001464 return err;
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02001465}
1466
Johan Hedbergb2939472014-07-30 09:22:23 +03001467static int set_bondable(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001468 u16 len)
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02001469{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001470 struct mgmt_mode *cp = data;
Marcel Holtmann55594352013-10-06 16:11:57 -07001471 bool changed;
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02001472 int err;
1473
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001474 BT_DBG("request for %s", hdev->name);
Johan Hedberg9fbcbb42010-12-30 00:18:33 +02001475
Johan Hedberga7e80f22013-01-09 16:05:19 +02001476 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02001477 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BONDABLE,
1478 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001479
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001480 hci_dev_lock(hdev);
Johan Hedberg053f0212011-01-26 13:07:10 +02001481
1482 if (cp->val)
Marcel Holtmann238be782015-03-13 02:11:06 -07001483 changed = !hci_dev_test_and_set_flag(hdev, HCI_BONDABLE);
Johan Hedberg053f0212011-01-26 13:07:10 +02001484 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07001485 changed = hci_dev_test_and_clear_flag(hdev, HCI_BONDABLE);
Johan Hedberg053f0212011-01-26 13:07:10 +02001486
Johan Hedbergb2939472014-07-30 09:22:23 +03001487 err = send_settings_rsp(sk, MGMT_OP_SET_BONDABLE, hdev);
Johan Hedberg053f0212011-01-26 13:07:10 +02001488 if (err < 0)
Marcel Holtmann55594352013-10-06 16:11:57 -07001489 goto unlock;
Johan Hedberg053f0212011-01-26 13:07:10 +02001490
Johan Hedberg82a37ad2016-03-09 17:30:34 +02001491 if (changed) {
1492 /* In limited privacy mode the change of bondable mode
1493 * may affect the local advertising address.
1494 */
1495 if (hdev_is_powered(hdev) &&
1496 hci_dev_test_flag(hdev, HCI_ADVERTISING) &&
1497 hci_dev_test_flag(hdev, HCI_DISCOVERABLE) &&
1498 hci_dev_test_flag(hdev, HCI_LIMITED_PRIVACY))
1499 queue_work(hdev->req_workqueue,
1500 &hdev->discoverable_update);
1501
Marcel Holtmann55594352013-10-06 16:11:57 -07001502 err = new_settings(hdev, sk);
Johan Hedberg82a37ad2016-03-09 17:30:34 +02001503 }
Johan Hedberg053f0212011-01-26 13:07:10 +02001504
Marcel Holtmann55594352013-10-06 16:11:57 -07001505unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001506 hci_dev_unlock(hdev);
Johan Hedberg053f0212011-01-26 13:07:10 +02001507 return err;
1508}
1509
Gustavo F. Padovan04124682012-03-08 01:25:00 -03001510static int set_link_security(struct sock *sk, struct hci_dev *hdev, void *data,
1511 u16 len)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001512{
1513 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001514 struct mgmt_pending_cmd *cmd;
Johan Hedberge6fe7982013-10-02 15:45:22 +03001515 u8 val, status;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001516 int err;
1517
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001518 BT_DBG("request for %s", hdev->name);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001519
Johan Hedberge6fe7982013-10-02 15:45:22 +03001520 status = mgmt_bredr_support(hdev);
1521 if (status)
Johan Hedberga69e8372015-03-06 21:08:53 +02001522 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
1523 status);
Johan Hedberg33c525c2012-10-24 21:11:58 +03001524
Johan Hedberga7e80f22013-01-09 16:05:19 +02001525 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02001526 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
1527 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001528
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001529 hci_dev_lock(hdev);
1530
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001531 if (!hdev_is_powered(hdev)) {
Johan Hedberg47990ea2012-02-22 11:58:37 +02001532 bool changed = false;
1533
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001534 if (!!cp->val != hci_dev_test_flag(hdev, HCI_LINK_SECURITY)) {
Marcel Holtmannce05d602015-03-13 02:11:03 -07001535 hci_dev_change_flag(hdev, HCI_LINK_SECURITY);
Johan Hedberg47990ea2012-02-22 11:58:37 +02001536 changed = true;
1537 }
1538
1539 err = send_settings_rsp(sk, MGMT_OP_SET_LINK_SECURITY, hdev);
1540 if (err < 0)
1541 goto failed;
1542
1543 if (changed)
1544 err = new_settings(hdev, sk);
1545
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001546 goto failed;
1547 }
1548
Johan Hedberg333ae952015-03-17 13:48:47 +02001549 if (pending_find(MGMT_OP_SET_LINK_SECURITY, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001550 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_LINK_SECURITY,
1551 MGMT_STATUS_BUSY);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001552 goto failed;
1553 }
1554
1555 val = !!cp->val;
1556
1557 if (test_bit(HCI_AUTH, &hdev->flags) == val) {
1558 err = send_settings_rsp(sk, MGMT_OP_SET_LINK_SECURITY, hdev);
1559 goto failed;
1560 }
1561
1562 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LINK_SECURITY, hdev, data, len);
1563 if (!cmd) {
1564 err = -ENOMEM;
1565 goto failed;
1566 }
1567
1568 err = hci_send_cmd(hdev, HCI_OP_WRITE_AUTH_ENABLE, sizeof(val), &val);
1569 if (err < 0) {
1570 mgmt_pending_remove(cmd);
1571 goto failed;
1572 }
1573
1574failed:
1575 hci_dev_unlock(hdev);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02001576 return err;
1577}
1578
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001579static int set_ssp(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001580{
1581 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001582 struct mgmt_pending_cmd *cmd;
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001583 u8 status;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001584 int err;
1585
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001586 BT_DBG("request for %s", hdev->name);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001587
Marcel Holtmanncdba5282013-10-02 21:31:52 -07001588 status = mgmt_bredr_support(hdev);
1589 if (status)
Johan Hedberga69e8372015-03-06 21:08:53 +02001590 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SSP, status);
Marcel Holtmanncdba5282013-10-02 21:31:52 -07001591
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001592 if (!lmp_ssp_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02001593 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1594 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001595
Johan Hedberga7e80f22013-01-09 16:05:19 +02001596 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02001597 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1598 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001599
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001600 hci_dev_lock(hdev);
Johan Hedberg6c8f12c2012-02-22 16:35:26 +02001601
Johan Hedberg4b34ee782012-02-21 14:13:02 +02001602 if (!hdev_is_powered(hdev)) {
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001603 bool changed;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02001604
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001605 if (cp->val) {
Marcel Holtmann238be782015-03-13 02:11:06 -07001606 changed = !hci_dev_test_and_set_flag(hdev,
1607 HCI_SSP_ENABLED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001608 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07001609 changed = hci_dev_test_and_clear_flag(hdev,
1610 HCI_SSP_ENABLED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001611 if (!changed)
Marcel Holtmanna69d8922015-03-13 02:11:05 -07001612 changed = hci_dev_test_and_clear_flag(hdev,
1613 HCI_HS_ENABLED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001614 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07001615 hci_dev_clear_flag(hdev, HCI_HS_ENABLED);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02001616 }
1617
1618 err = send_settings_rsp(sk, MGMT_OP_SET_SSP, hdev);
1619 if (err < 0)
1620 goto failed;
1621
1622 if (changed)
1623 err = new_settings(hdev, sk);
1624
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001625 goto failed;
1626 }
1627
Johan Hedberg333ae952015-03-17 13:48:47 +02001628 if (pending_find(MGMT_OP_SET_SSP, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001629 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SSP,
1630 MGMT_STATUS_BUSY);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001631 goto failed;
1632 }
1633
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001634 if (!!cp->val == hci_dev_test_flag(hdev, HCI_SSP_ENABLED)) {
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001635 err = send_settings_rsp(sk, MGMT_OP_SET_SSP, hdev);
1636 goto failed;
1637 }
1638
1639 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SSP, hdev, data, len);
1640 if (!cmd) {
1641 err = -ENOMEM;
1642 goto failed;
1643 }
1644
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001645 if (!cp->val && hci_dev_test_flag(hdev, HCI_USE_DEBUG_KEYS))
Johan Hedberg37699722014-06-24 14:00:27 +03001646 hci_send_cmd(hdev, HCI_OP_WRITE_SSP_DEBUG_MODE,
1647 sizeof(cp->val), &cp->val);
1648
Marcel Holtmann72ef0c12013-10-10 03:08:10 -07001649 err = hci_send_cmd(hdev, HCI_OP_WRITE_SSP_MODE, 1, &cp->val);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001650 if (err < 0) {
1651 mgmt_pending_remove(cmd);
1652 goto failed;
1653 }
1654
1655failed:
1656 hci_dev_unlock(hdev);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02001657 return err;
1658}
1659
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001660static int set_hs(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001661{
1662 struct mgmt_mode *cp = data;
Marcel Holtmannee392692013-10-01 22:59:23 -07001663 bool changed;
Johan Hedberge6fe7982013-10-02 15:45:22 +03001664 u8 status;
Marcel Holtmannee392692013-10-01 22:59:23 -07001665 int err;
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001666
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001667 BT_DBG("request for %s", hdev->name);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001668
Johan Hedberge6fe7982013-10-02 15:45:22 +03001669 status = mgmt_bredr_support(hdev);
1670 if (status)
Johan Hedberga69e8372015-03-06 21:08:53 +02001671 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_HS, status);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001672
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001673 if (!lmp_ssp_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02001674 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1675 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001676
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001677 if (!hci_dev_test_flag(hdev, HCI_SSP_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02001678 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1679 MGMT_STATUS_REJECTED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07001680
Johan Hedberga7e80f22013-01-09 16:05:19 +02001681 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02001682 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1683 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001684
Marcel Holtmannee392692013-10-01 22:59:23 -07001685 hci_dev_lock(hdev);
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001686
Johan Hedberg333ae952015-03-17 13:48:47 +02001687 if (pending_find(MGMT_OP_SET_SSP, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001688 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1689 MGMT_STATUS_BUSY);
Johan Hedberga2cb01d2015-02-19 17:38:07 +02001690 goto unlock;
1691 }
1692
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001693 if (cp->val) {
Marcel Holtmann238be782015-03-13 02:11:06 -07001694 changed = !hci_dev_test_and_set_flag(hdev, HCI_HS_ENABLED);
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001695 } else {
1696 if (hdev_is_powered(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001697 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_HS,
1698 MGMT_STATUS_REJECTED);
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001699 goto unlock;
1700 }
1701
Marcel Holtmanna69d8922015-03-13 02:11:05 -07001702 changed = hci_dev_test_and_clear_flag(hdev, HCI_HS_ENABLED);
Marcel Holtmanna0cdf9602013-10-02 00:27:02 -07001703 }
Marcel Holtmannee392692013-10-01 22:59:23 -07001704
1705 err = send_settings_rsp(sk, MGMT_OP_SET_HS, hdev);
1706 if (err < 0)
1707 goto unlock;
1708
1709 if (changed)
1710 err = new_settings(hdev, sk);
1711
1712unlock:
1713 hci_dev_unlock(hdev);
1714 return err;
Johan Hedberg6d80dfd2012-02-20 23:50:38 +02001715}
1716
Marcel Holtmann1904a852015-01-11 13:50:44 -08001717static void le_enable_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001718{
1719 struct cmd_lookup match = { NULL, hdev };
1720
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05301721 hci_dev_lock(hdev);
1722
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001723 if (status) {
1724 u8 mgmt_err = mgmt_status(status);
1725
1726 mgmt_pending_foreach(MGMT_OP_SET_LE, hdev, cmd_status_rsp,
1727 &mgmt_err);
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05301728 goto unlock;
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001729 }
1730
1731 mgmt_pending_foreach(MGMT_OP_SET_LE, hdev, settings_rsp, &match);
1732
1733 new_settings(hdev, match.sk);
1734
1735 if (match.sk)
1736 sock_put(match.sk);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001737
1738 /* Make sure the controller has a good default for
1739 * advertising data. Restrict the update to when LE
1740 * has actually been enabled. During power on, the
1741 * update in powered_update_hci will take care of it.
1742 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001743 if (hci_dev_test_flag(hdev, HCI_LE_ENABLED)) {
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001744 struct hci_request req;
1745
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001746 hci_req_init(&req, hdev);
Johan Hedbergcab054a2015-11-30 11:21:45 +02001747 __hci_req_update_adv_data(&req, 0x00);
1748 __hci_req_update_scan_rsp_data(&req, 0x00);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001749 hci_req_run(&req, NULL);
Johan Hedberg2e93e532015-11-11 08:11:17 +02001750 hci_update_background_scan(hdev);
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001751 }
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05301752
1753unlock:
1754 hci_dev_unlock(hdev);
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001755}
1756
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001757static int set_le(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg06199cf2012-02-22 16:37:11 +02001758{
1759 struct mgmt_mode *cp = data;
1760 struct hci_cp_write_le_host_supported hci_cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001761 struct mgmt_pending_cmd *cmd;
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001762 struct hci_request req;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001763 int err;
Johan Hedberg0b60eba2012-02-28 00:57:24 +02001764 u8 val, enabled;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001765
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001766 BT_DBG("request for %s", hdev->name);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001767
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001768 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02001769 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1770 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg1de028c2012-02-29 19:55:35 -08001771
Johan Hedberga7e80f22013-01-09 16:05:19 +02001772 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02001773 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1774 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02001775
Marcel Holtmanne7844ee2015-03-18 16:15:07 -07001776 /* Bluetooth single mode LE only controllers or dual-mode
1777 * controllers configured as LE only devices, do not allow
1778 * switching LE off. These have either LE enabled explicitly
1779 * or BR/EDR has been previously switched off.
1780 *
1781 * When trying to enable an already enabled LE, then gracefully
1782 * send a positive response. Trying to disable it however will
1783 * result into rejection.
1784 */
1785 if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED)) {
1786 if (cp->val == 0x01)
1787 return send_settings_rsp(sk, MGMT_OP_SET_LE, hdev);
1788
Johan Hedberga69e8372015-03-06 21:08:53 +02001789 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1790 MGMT_STATUS_REJECTED);
Marcel Holtmanne7844ee2015-03-18 16:15:07 -07001791 }
Johan Hedbergc73eee92013-04-19 18:35:21 +03001792
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02001793 hci_dev_lock(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001794
1795 val = !!cp->val;
Gustavo Padovanffa88e02012-11-23 16:50:51 -02001796 enabled = lmp_host_le_capable(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001797
Florian Grandel847818d2015-06-18 03:16:46 +02001798 if (!val)
Johan Hedberg37d3a1f2016-08-28 20:53:34 +03001799 hci_req_clear_adv_instance(hdev, NULL, NULL, 0x00, true);
Florian Grandel847818d2015-06-18 03:16:46 +02001800
Johan Hedberg0b60eba2012-02-28 00:57:24 +02001801 if (!hdev_is_powered(hdev) || val == enabled) {
Johan Hedberg06199cf2012-02-22 16:37:11 +02001802 bool changed = false;
1803
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001804 if (val != hci_dev_test_flag(hdev, HCI_LE_ENABLED)) {
Marcel Holtmannce05d602015-03-13 02:11:03 -07001805 hci_dev_change_flag(hdev, HCI_LE_ENABLED);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001806 changed = true;
1807 }
1808
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001809 if (!val && hci_dev_test_flag(hdev, HCI_ADVERTISING)) {
Marcel Holtmanna358dc12015-03-13 02:11:02 -07001810 hci_dev_clear_flag(hdev, HCI_ADVERTISING);
Johan Hedbergeeca6f82013-09-25 13:26:09 +03001811 changed = true;
1812 }
1813
Johan Hedberg06199cf2012-02-22 16:37:11 +02001814 err = send_settings_rsp(sk, MGMT_OP_SET_LE, hdev);
1815 if (err < 0)
Johan Hedberg1de028c2012-02-29 19:55:35 -08001816 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001817
1818 if (changed)
1819 err = new_settings(hdev, sk);
1820
Johan Hedberg1de028c2012-02-29 19:55:35 -08001821 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001822 }
1823
Johan Hedberg333ae952015-03-17 13:48:47 +02001824 if (pending_find(MGMT_OP_SET_LE, hdev) ||
1825 pending_find(MGMT_OP_SET_ADVERTISING, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001826 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_LE,
1827 MGMT_STATUS_BUSY);
Johan Hedberg1de028c2012-02-29 19:55:35 -08001828 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001829 }
1830
1831 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LE, hdev, data, len);
1832 if (!cmd) {
1833 err = -ENOMEM;
Johan Hedberg1de028c2012-02-29 19:55:35 -08001834 goto unlock;
Johan Hedberg06199cf2012-02-22 16:37:11 +02001835 }
1836
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001837 hci_req_init(&req, hdev);
1838
Johan Hedberg06199cf2012-02-22 16:37:11 +02001839 memset(&hci_cp, 0, sizeof(hci_cp));
1840
1841 if (val) {
1842 hci_cp.le = val;
Marcel Holtmann32226e42014-07-24 20:04:16 +02001843 hci_cp.simul = 0x00;
Marcel Holtmann441ad2d2013-10-15 06:33:52 -07001844 } else {
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07001845 if (hci_dev_test_flag(hdev, HCI_LE_ADV))
Johan Hedbergf2252572015-11-18 12:49:20 +02001846 __hci_req_disable_advertising(&req);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001847 }
1848
Johan Hedberg416a4ae2013-09-25 13:26:08 +03001849 hci_req_add(&req, HCI_OP_WRITE_LE_HOST_SUPPORTED, sizeof(hci_cp),
1850 &hci_cp);
1851
1852 err = hci_req_run(&req, le_enable_complete);
Syam Sidhardhan0c01bc42012-04-12 20:33:21 +05301853 if (err < 0)
Johan Hedberg06199cf2012-02-22 16:37:11 +02001854 mgmt_pending_remove(cmd);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001855
Johan Hedberg1de028c2012-02-29 19:55:35 -08001856unlock:
1857 hci_dev_unlock(hdev);
Johan Hedberg06199cf2012-02-22 16:37:11 +02001858 return err;
1859}
1860
Johan Hedberg0cab9c82013-03-15 17:06:54 -05001861/* This is a helper function to test for pending mgmt commands that can
1862 * cause CoD or EIR HCI commands. We can only allow one such pending
1863 * mgmt command at a time since otherwise we cannot easily track what
1864 * the current values are, will be, and based on that calculate if a new
1865 * HCI command needs to be sent and if yes with what value.
1866 */
1867static bool pending_eir_or_class(struct hci_dev *hdev)
1868{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001869 struct mgmt_pending_cmd *cmd;
Johan Hedberg0cab9c82013-03-15 17:06:54 -05001870
1871 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
1872 switch (cmd->opcode) {
1873 case MGMT_OP_ADD_UUID:
1874 case MGMT_OP_REMOVE_UUID:
1875 case MGMT_OP_SET_DEV_CLASS:
1876 case MGMT_OP_SET_POWERED:
1877 return true;
1878 }
1879 }
1880
1881 return false;
1882}
1883
Johan Hedberg83be8ec2013-01-27 00:31:29 +02001884static const u8 bluetooth_base_uuid[] = {
1885 0xfb, 0x34, 0x9b, 0x5f, 0x80, 0x00, 0x00, 0x80,
1886 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1887};
1888
1889static u8 get_uuid_size(const u8 *uuid)
1890{
1891 u32 val;
1892
1893 if (memcmp(uuid, bluetooth_base_uuid, 12))
1894 return 128;
1895
1896 val = get_unaligned_le32(&uuid[12]);
1897 if (val > 0xffff)
1898 return 32;
1899
1900 return 16;
1901}
1902
Johan Hedberg92da6092013-03-15 17:06:55 -05001903static void mgmt_class_complete(struct hci_dev *hdev, u16 mgmt_op, u8 status)
1904{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001905 struct mgmt_pending_cmd *cmd;
Johan Hedberg92da6092013-03-15 17:06:55 -05001906
1907 hci_dev_lock(hdev);
1908
Johan Hedberg333ae952015-03-17 13:48:47 +02001909 cmd = pending_find(mgmt_op, hdev);
Johan Hedberg92da6092013-03-15 17:06:55 -05001910 if (!cmd)
1911 goto unlock;
1912
Johan Hedberg2a1afb52015-03-06 21:08:54 +02001913 mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode,
1914 mgmt_status(status), hdev->dev_class, 3);
Johan Hedberg92da6092013-03-15 17:06:55 -05001915
1916 mgmt_pending_remove(cmd);
1917
1918unlock:
1919 hci_dev_unlock(hdev);
1920}
1921
Marcel Holtmann1904a852015-01-11 13:50:44 -08001922static void add_uuid_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg92da6092013-03-15 17:06:55 -05001923{
1924 BT_DBG("status 0x%02x", status);
1925
1926 mgmt_class_complete(hdev, MGMT_OP_ADD_UUID, status);
1927}
1928
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001929static int add_uuid(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001930{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03001931 struct mgmt_cp_add_uuid *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02001932 struct mgmt_pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05001933 struct hci_request req;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001934 struct bt_uuid *uuid;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001935 int err;
1936
Johan Hedbergbdb6d972012-02-28 06:13:32 +02001937 BT_DBG("request for %s", hdev->name);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001938
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001939 hci_dev_lock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001940
Johan Hedberg0cab9c82013-03-15 17:06:54 -05001941 if (pending_eir_or_class(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02001942 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_UUID,
1943 MGMT_STATUS_BUSY);
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02001944 goto failed;
1945 }
1946
Andre Guedes92c4c202012-06-07 19:05:44 -03001947 uuid = kmalloc(sizeof(*uuid), GFP_KERNEL);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001948 if (!uuid) {
1949 err = -ENOMEM;
1950 goto failed;
1951 }
1952
1953 memcpy(uuid->uuid, cp->uuid, 16);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02001954 uuid->svc_hint = cp->svc_hint;
Johan Hedberg83be8ec2013-01-27 00:31:29 +02001955 uuid->size = get_uuid_size(cp->uuid);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001956
Johan Hedbergde66aa62013-01-27 00:31:27 +02001957 list_add_tail(&uuid->list, &hdev->uuids);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001958
Johan Hedberg890ea892013-03-15 17:06:52 -05001959 hci_req_init(&req, hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02001960
Johan Hedberg14bf5ea2015-11-22 19:00:22 +02001961 __hci_req_update_class(&req);
Johan Hedbergb1a89172015-11-25 16:15:42 +02001962 __hci_req_update_eir(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05001963
Johan Hedberg92da6092013-03-15 17:06:55 -05001964 err = hci_req_run(&req, add_uuid_complete);
1965 if (err < 0) {
1966 if (err != -ENODATA)
1967 goto failed;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +03001968
Johan Hedberg2a1afb52015-03-06 21:08:54 +02001969 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_UUID, 0,
1970 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02001971 goto failed;
1972 }
1973
1974 cmd = mgmt_pending_add(sk, MGMT_OP_ADD_UUID, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05001975 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02001976 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05001977 goto failed;
1978 }
1979
1980 err = 0;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001981
1982failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03001983 hci_dev_unlock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02001984 return err;
1985}
1986
Johan Hedberg24b78d02012-02-23 23:24:30 +02001987static bool enable_service_cache(struct hci_dev *hdev)
1988{
1989 if (!hdev_is_powered(hdev))
1990 return false;
1991
Marcel Holtmann238be782015-03-13 02:11:06 -07001992 if (!hci_dev_test_and_set_flag(hdev, HCI_SERVICE_CACHE)) {
Johan Hedberg46818ed2013-01-14 22:33:52 +02001993 queue_delayed_work(hdev->workqueue, &hdev->service_cache,
1994 CACHE_TIMEOUT);
Johan Hedberg24b78d02012-02-23 23:24:30 +02001995 return true;
1996 }
1997
1998 return false;
1999}
2000
Marcel Holtmann1904a852015-01-11 13:50:44 -08002001static void remove_uuid_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg92da6092013-03-15 17:06:55 -05002002{
2003 BT_DBG("status 0x%02x", status);
2004
2005 mgmt_class_complete(hdev, MGMT_OP_REMOVE_UUID, status);
2006}
2007
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002008static int remove_uuid(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002009 u16 len)
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002010{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002011 struct mgmt_cp_remove_uuid *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002012 struct mgmt_pending_cmd *cmd;
Johan Hedberg056341c2013-01-27 00:31:30 +02002013 struct bt_uuid *match, *tmp;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002014 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 -05002015 struct hci_request req;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002016 int err, found;
2017
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002018 BT_DBG("request for %s", hdev->name);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002019
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002020 hci_dev_lock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002021
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002022 if (pending_eir_or_class(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002023 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_REMOVE_UUID,
2024 MGMT_STATUS_BUSY);
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02002025 goto unlock;
2026 }
2027
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002028 if (memcmp(cp->uuid, bt_uuid_any, 16) == 0) {
Johan Hedberg35f74982014-02-18 17:14:32 +02002029 hci_uuids_clear(hdev);
Johan Hedberg4004b6d2012-02-23 21:30:12 +02002030
Johan Hedberg24b78d02012-02-23 23:24:30 +02002031 if (enable_service_cache(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002032 err = mgmt_cmd_complete(sk, hdev->id,
2033 MGMT_OP_REMOVE_UUID,
2034 0, hdev->dev_class, 3);
Johan Hedberg24b78d02012-02-23 23:24:30 +02002035 goto unlock;
2036 }
Johan Hedberg4004b6d2012-02-23 21:30:12 +02002037
Johan Hedberg9246a862012-02-23 21:33:16 +02002038 goto update_class;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002039 }
2040
2041 found = 0;
2042
Johan Hedberg056341c2013-01-27 00:31:30 +02002043 list_for_each_entry_safe(match, tmp, &hdev->uuids, list) {
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002044 if (memcmp(match->uuid, cp->uuid, 16) != 0)
2045 continue;
2046
2047 list_del(&match->list);
Johan Hedberg482049f2012-11-08 10:25:26 +01002048 kfree(match);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002049 found++;
2050 }
2051
2052 if (found == 0) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002053 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_REMOVE_UUID,
2054 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002055 goto unlock;
2056 }
2057
Johan Hedberg9246a862012-02-23 21:33:16 +02002058update_class:
Johan Hedberg890ea892013-03-15 17:06:52 -05002059 hci_req_init(&req, hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002060
Johan Hedberg14bf5ea2015-11-22 19:00:22 +02002061 __hci_req_update_class(&req);
Johan Hedbergb1a89172015-11-25 16:15:42 +02002062 __hci_req_update_eir(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05002063
Johan Hedberg92da6092013-03-15 17:06:55 -05002064 err = hci_req_run(&req, remove_uuid_complete);
2065 if (err < 0) {
2066 if (err != -ENODATA)
2067 goto unlock;
Johan Hedberg80a1e1d2011-03-28 14:07:23 +03002068
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002069 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_UUID, 0,
2070 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002071 goto unlock;
2072 }
2073
2074 cmd = mgmt_pending_add(sk, MGMT_OP_REMOVE_UUID, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002075 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002076 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002077 goto unlock;
2078 }
2079
2080 err = 0;
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002081
2082unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002083 hci_dev_unlock(hdev);
Johan Hedberg2aeb9a12011-01-04 12:08:51 +02002084 return err;
2085}
2086
Marcel Holtmann1904a852015-01-11 13:50:44 -08002087static void set_class_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg92da6092013-03-15 17:06:55 -05002088{
2089 BT_DBG("status 0x%02x", status);
2090
2091 mgmt_class_complete(hdev, MGMT_OP_SET_DEV_CLASS, status);
2092}
2093
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002094static int set_dev_class(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002095 u16 len)
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002096{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002097 struct mgmt_cp_set_dev_class *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002098 struct mgmt_pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05002099 struct hci_request req;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002100 int err;
2101
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002102 BT_DBG("request for %s", hdev->name);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002103
Marcel Holtmann6203fc92013-10-02 23:37:29 -07002104 if (!lmp_bredr_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02002105 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2106 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg13ecd8b2013-01-09 15:29:38 +02002107
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002108 hci_dev_lock(hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002109
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002110 if (pending_eir_or_class(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002111 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2112 MGMT_STATUS_BUSY);
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002113 goto unlock;
2114 }
2115
2116 if ((cp->minor & 0x03) != 0 || (cp->major & 0xe0) != 0) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002117 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DEV_CLASS,
2118 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg0cab9c82013-03-15 17:06:54 -05002119 goto unlock;
2120 }
2121
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002122 hdev->major_class = cp->major;
2123 hdev->minor_class = cp->minor;
2124
Johan Hedberg932f5ff2012-02-22 22:11:32 +02002125 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002126 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_DEV_CLASS, 0,
2127 hdev->dev_class, 3);
Johan Hedberg932f5ff2012-02-22 22:11:32 +02002128 goto unlock;
2129 }
2130
Johan Hedberg890ea892013-03-15 17:06:52 -05002131 hci_req_init(&req, hdev);
2132
Marcel Holtmanna69d8922015-03-13 02:11:05 -07002133 if (hci_dev_test_and_clear_flag(hdev, HCI_SERVICE_CACHE)) {
Johan Hedberg7d785252011-12-15 00:47:39 +02002134 hci_dev_unlock(hdev);
2135 cancel_delayed_work_sync(&hdev->service_cache);
2136 hci_dev_lock(hdev);
Johan Hedbergb1a89172015-11-25 16:15:42 +02002137 __hci_req_update_eir(&req);
Johan Hedberg7d785252011-12-15 00:47:39 +02002138 }
Johan Hedberg14c0b602011-12-15 00:47:37 +02002139
Johan Hedberg14bf5ea2015-11-22 19:00:22 +02002140 __hci_req_update_class(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05002141
Johan Hedberg92da6092013-03-15 17:06:55 -05002142 err = hci_req_run(&req, set_class_complete);
2143 if (err < 0) {
2144 if (err != -ENODATA)
2145 goto unlock;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002146
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002147 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_DEV_CLASS, 0,
2148 hdev->dev_class, 3);
Johan Hedberg90e70452012-02-23 23:09:40 +02002149 goto unlock;
2150 }
2151
2152 cmd = mgmt_pending_add(sk, MGMT_OP_SET_DEV_CLASS, hdev, data, len);
Johan Hedberg890ea892013-03-15 17:06:52 -05002153 if (!cmd) {
Johan Hedberg90e70452012-02-23 23:09:40 +02002154 err = -ENOMEM;
Johan Hedberg890ea892013-03-15 17:06:52 -05002155 goto unlock;
2156 }
2157
2158 err = 0;
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002159
Johan Hedbergb5235a62012-02-21 14:32:24 +02002160unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002161 hci_dev_unlock(hdev);
Johan Hedberg1aff6f02011-01-13 21:56:52 +02002162 return err;
2163}
2164
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002165static int load_link_keys(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002166 u16 len)
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002167{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002168 struct mgmt_cp_load_link_keys *cp = data;
Johan Hedbergba1d6932014-07-03 13:52:27 +03002169 const u16 max_key_count = ((U16_MAX - sizeof(*cp)) /
2170 sizeof(struct mgmt_link_key_info));
Szymon Janc4e51eae2011-02-25 19:05:48 +01002171 u16 key_count, expected_len;
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002172 bool changed;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002173 int i;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002174
Marcel Holtmann9060d5c2013-10-02 21:16:07 -07002175 BT_DBG("request for %s", hdev->name);
2176
2177 if (!lmp_bredr_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02002178 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2179 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmann9060d5c2013-10-02 21:16:07 -07002180
Marcel Holtmann1f350c82012-03-12 20:31:08 -07002181 key_count = __le16_to_cpu(cp->key_count);
Johan Hedbergba1d6932014-07-03 13:52:27 +03002182 if (key_count > max_key_count) {
2183 BT_ERR("load_link_keys: too big key_count value %u",
2184 key_count);
Johan Hedberga69e8372015-03-06 21:08:53 +02002185 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2186 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergba1d6932014-07-03 13:52:27 +03002187 }
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002188
Johan Hedberg86742e12011-11-07 23:13:38 +02002189 expected_len = sizeof(*cp) + key_count *
2190 sizeof(struct mgmt_link_key_info);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002191 if (expected_len != len) {
Johan Hedberg86742e12011-11-07 23:13:38 +02002192 BT_ERR("load_link_keys: expected %u bytes, got %u bytes",
Johan Hedberg2606ecb2014-03-07 15:04:13 +02002193 expected_len, len);
Johan Hedberga69e8372015-03-06 21:08:53 +02002194 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2195 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002196 }
2197
Johan Hedberg4ae14302013-01-20 14:27:13 +02002198 if (cp->debug_keys != 0x00 && cp->debug_keys != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02002199 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS,
2200 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg4ae14302013-01-20 14:27:13 +02002201
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002202 BT_DBG("%s debug_keys %u key_count %u", hdev->name, cp->debug_keys,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002203 key_count);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002204
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002205 for (i = 0; i < key_count; i++) {
2206 struct mgmt_link_key_info *key = &cp->keys[i];
2207
Marcel Holtmann8e991132014-01-10 02:07:25 -08002208 if (key->addr.type != BDADDR_BREDR || key->type > 0x08)
Johan Hedberga69e8372015-03-06 21:08:53 +02002209 return mgmt_cmd_status(sk, hdev->id,
2210 MGMT_OP_LOAD_LINK_KEYS,
2211 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002212 }
2213
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002214 hci_dev_lock(hdev);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002215
2216 hci_link_keys_clear(hdev);
2217
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002218 if (cp->debug_keys)
Marcel Holtmann238be782015-03-13 02:11:06 -07002219 changed = !hci_dev_test_and_set_flag(hdev, HCI_KEEP_DEBUG_KEYS);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002220 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07002221 changed = hci_dev_test_and_clear_flag(hdev,
2222 HCI_KEEP_DEBUG_KEYS);
Marcel Holtmannb1de97d2014-01-31 11:55:21 -08002223
2224 if (changed)
2225 new_settings(hdev, NULL);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002226
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002227 for (i = 0; i < key_count; i++) {
Johan Hedberg86742e12011-11-07 23:13:38 +02002228 struct mgmt_link_key_info *key = &cp->keys[i];
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002229
Johan Hedberg58e92932014-06-24 14:00:26 +03002230 /* Always ignore debug keys and require a new pairing if
2231 * the user wants to use them.
2232 */
2233 if (key->type == HCI_LK_DEBUG_COMBINATION)
2234 continue;
2235
Johan Hedberg7652ff62014-06-24 13:15:49 +03002236 hci_add_link_key(hdev, NULL, &key->addr.bdaddr, key->val,
2237 key->type, key->pin_len, NULL);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002238 }
2239
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002240 mgmt_cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LINK_KEYS, 0, NULL, 0);
Johan Hedberg0e5f8752011-11-11 16:18:54 +02002241
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002242 hci_dev_unlock(hdev);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002243
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03002244 return 0;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002245}
2246
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002247static int device_unpaired(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002248 u8 addr_type, struct sock *skip_sk)
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002249{
2250 struct mgmt_ev_device_unpaired ev;
2251
2252 bacpy(&ev.addr.bdaddr, bdaddr);
2253 ev.addr.type = addr_type;
2254
2255 return mgmt_event(MGMT_EV_DEVICE_UNPAIRED, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002256 skip_sk);
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002257}
2258
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002259static int unpair_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002260 u16 len)
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002261{
Johan Hedberg124f6e32012-02-09 13:50:12 +02002262 struct mgmt_cp_unpair_device *cp = data;
2263 struct mgmt_rp_unpair_device rp;
Johan Hedbergfc643612015-10-22 09:38:31 +03002264 struct hci_conn_params *params;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002265 struct mgmt_pending_cmd *cmd;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002266 struct hci_conn *conn;
Johan Hedbergec182f02015-10-21 18:03:03 +03002267 u8 addr_type;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002268 int err;
2269
Johan Hedberga8a1d192011-11-10 15:54:38 +02002270 memset(&rp, 0, sizeof(rp));
Johan Hedberg124f6e32012-02-09 13:50:12 +02002271 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2272 rp.addr.type = cp->addr.type;
Johan Hedberga8a1d192011-11-10 15:54:38 +02002273
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002274 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002275 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2276 MGMT_STATUS_INVALID_PARAMS,
2277 &rp, sizeof(rp));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002278
Johan Hedberg118da702013-01-20 14:27:20 +02002279 if (cp->disconnect != 0x00 && cp->disconnect != 0x01)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002280 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2281 MGMT_STATUS_INVALID_PARAMS,
2282 &rp, sizeof(rp));
Johan Hedberg118da702013-01-20 14:27:20 +02002283
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002284 hci_dev_lock(hdev);
2285
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002286 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002287 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2288 MGMT_STATUS_NOT_POWERED, &rp,
2289 sizeof(rp));
Johan Hedberg86a8cfc2012-02-22 22:53:34 +02002290 goto unlock;
2291 }
2292
Johan Hedberge0b2b272014-02-18 17:14:31 +02002293 if (cp->addr.type == BDADDR_BREDR) {
Alfonso Acosta89cbb062014-10-11 21:44:47 +00002294 /* If disconnection is requested, then look up the
2295 * connection. If the remote device is connected, it
2296 * will be later used to terminate the link.
2297 *
2298 * Setting it to NULL explicitly will cause no
2299 * termination of the link.
2300 */
2301 if (cp->disconnect)
2302 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
2303 &cp->addr.bdaddr);
2304 else
2305 conn = NULL;
2306
Johan Hedberg124f6e32012-02-09 13:50:12 +02002307 err = hci_remove_link_key(hdev, &cp->addr.bdaddr);
Johan Hedbergec182f02015-10-21 18:03:03 +03002308 if (err < 0) {
2309 err = mgmt_cmd_complete(sk, hdev->id,
2310 MGMT_OP_UNPAIR_DEVICE,
2311 MGMT_STATUS_NOT_PAIRED, &rp,
2312 sizeof(rp));
2313 goto unlock;
Alfonso Acosta89cbb062014-10-11 21:44:47 +00002314 }
2315
Johan Hedbergec182f02015-10-21 18:03:03 +03002316 goto done;
Johan Hedberge0b2b272014-02-18 17:14:31 +02002317 }
Vinicius Costa Gomesb0dbfb42012-02-02 21:08:03 -03002318
Johan Hedbergec182f02015-10-21 18:03:03 +03002319 /* LE address type */
2320 addr_type = le_addr_type(cp->addr.type);
2321
2322 hci_remove_irk(hdev, &cp->addr.bdaddr, addr_type);
2323
2324 err = hci_remove_ltk(hdev, &cp->addr.bdaddr, addr_type);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002325 if (err < 0) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002326 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE,
2327 MGMT_STATUS_NOT_PAIRED, &rp,
2328 sizeof(rp));
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002329 goto unlock;
2330 }
2331
Johan Hedbergec182f02015-10-21 18:03:03 +03002332 conn = hci_conn_hash_lookup_le(hdev, &cp->addr.bdaddr, addr_type);
2333 if (!conn) {
2334 hci_conn_params_del(hdev, &cp->addr.bdaddr, addr_type);
2335 goto done;
2336 }
2337
Johan Hedbergc81d5552015-10-22 09:38:35 +03002338 /* Abort any ongoing SMP pairing */
2339 smp_cancel_pairing(conn);
2340
Johan Hedbergec182f02015-10-21 18:03:03 +03002341 /* Defer clearing up the connection parameters until closing to
2342 * give a chance of keeping them if a repairing happens.
2343 */
2344 set_bit(HCI_CONN_PARAM_REMOVAL_PEND, &conn->flags);
2345
Johan Hedbergfc643612015-10-22 09:38:31 +03002346 /* Disable auto-connection parameters if present */
2347 params = hci_conn_params_lookup(hdev, &cp->addr.bdaddr, addr_type);
2348 if (params) {
2349 if (params->explicit_connect)
2350 params->auto_connect = HCI_AUTO_CONN_EXPLICIT;
2351 else
2352 params->auto_connect = HCI_AUTO_CONN_DISABLED;
2353 }
2354
Johan Hedbergec182f02015-10-21 18:03:03 +03002355 /* If disconnection is not requested, then clear the connection
2356 * variable so that the link is not terminated.
2357 */
2358 if (!cp->disconnect)
2359 conn = NULL;
2360
2361done:
Alfonso Acosta89cbb062014-10-11 21:44:47 +00002362 /* If the connection variable is set, then termination of the
2363 * link is requested.
2364 */
Johan Hedberga8a1d192011-11-10 15:54:38 +02002365 if (!conn) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002366 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNPAIR_DEVICE, 0,
2367 &rp, sizeof(rp));
Johan Hedbergb1078ad2012-02-09 17:21:16 +02002368 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, sk);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002369 goto unlock;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002370 }
2371
Johan Hedberg124f6e32012-02-09 13:50:12 +02002372 cmd = mgmt_pending_add(sk, MGMT_OP_UNPAIR_DEVICE, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002373 sizeof(*cp));
Johan Hedberga8a1d192011-11-10 15:54:38 +02002374 if (!cmd) {
2375 err = -ENOMEM;
2376 goto unlock;
2377 }
2378
Johan Hedbergd8b7b1e2014-12-05 13:36:05 +02002379 cmd->cmd_complete = addr_cmd_complete;
2380
Johan Hedberg89e0ccc2015-10-22 10:49:38 +03002381 err = hci_abort_conn(conn, HCI_ERROR_REMOTE_USER_TERM);
Johan Hedberga8a1d192011-11-10 15:54:38 +02002382 if (err < 0)
2383 mgmt_pending_remove(cmd);
2384
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002385unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002386 hci_dev_unlock(hdev);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02002387 return err;
2388}
2389
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002390static int disconnect(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002391 u16 len)
Johan Hedberg8962ee72011-01-20 12:40:27 +02002392{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002393 struct mgmt_cp_disconnect *cp = data;
Johan Hedberg06a63b12013-01-20 14:27:21 +02002394 struct mgmt_rp_disconnect rp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002395 struct mgmt_pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002396 struct hci_conn *conn;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002397 int err;
2398
2399 BT_DBG("");
2400
Johan Hedberg06a63b12013-01-20 14:27:21 +02002401 memset(&rp, 0, sizeof(rp));
2402 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2403 rp.addr.type = cp->addr.type;
2404
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002405 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002406 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2407 MGMT_STATUS_INVALID_PARAMS,
2408 &rp, sizeof(rp));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002409
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002410 hci_dev_lock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002411
2412 if (!test_bit(HCI_UP, &hdev->flags)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002413 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2414 MGMT_STATUS_NOT_POWERED, &rp,
2415 sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002416 goto failed;
2417 }
2418
Johan Hedberg333ae952015-03-17 13:48:47 +02002419 if (pending_find(MGMT_OP_DISCONNECT, hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002420 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2421 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002422 goto failed;
2423 }
2424
Andre Guedes591f47f2012-04-24 21:02:49 -03002425 if (cp->addr.type == BDADDR_BREDR)
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03002426 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
2427 &cp->addr.bdaddr);
Johan Hedberg88c3df12012-02-09 14:27:38 +02002428 else
Johan Hedberg9d4c1cc2015-10-21 18:03:01 +03002429 conn = hci_conn_hash_lookup_le(hdev, &cp->addr.bdaddr,
2430 le_addr_type(cp->addr.type));
Vinicius Costa Gomes365227e2011-05-06 18:41:44 -03002431
Vishal Agarwalf9607272012-06-13 05:32:43 +05302432 if (!conn || conn->state == BT_OPEN || conn->state == BT_CLOSED) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002433 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_DISCONNECT,
2434 MGMT_STATUS_NOT_CONNECTED, &rp,
2435 sizeof(rp));
Johan Hedberg8962ee72011-01-20 12:40:27 +02002436 goto failed;
2437 }
2438
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002439 cmd = mgmt_pending_add(sk, MGMT_OP_DISCONNECT, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002440 if (!cmd) {
2441 err = -ENOMEM;
Johan Hedberg8962ee72011-01-20 12:40:27 +02002442 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002443 }
Johan Hedberg8962ee72011-01-20 12:40:27 +02002444
Johan Hedbergf5818c22014-12-05 13:36:02 +02002445 cmd->cmd_complete = generic_cmd_complete;
2446
Johan Hedberge3f2f922014-08-18 20:33:33 +03002447 err = hci_disconnect(conn, HCI_ERROR_REMOTE_USER_TERM);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002448 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002449 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002450
2451failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002452 hci_dev_unlock(hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02002453 return err;
2454}
2455
Andre Guedes57c14772012-04-24 21:02:50 -03002456static u8 link_to_bdaddr(u8 link_type, u8 addr_type)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002457{
2458 switch (link_type) {
2459 case LE_LINK:
Johan Hedberg48264f02011-11-09 13:58:58 +02002460 switch (addr_type) {
2461 case ADDR_LE_DEV_PUBLIC:
Andre Guedes591f47f2012-04-24 21:02:49 -03002462 return BDADDR_LE_PUBLIC;
Andre Guedes0ed09142012-04-03 08:46:54 -03002463
Johan Hedberg48264f02011-11-09 13:58:58 +02002464 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002465 /* Fallback to LE Random address type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002466 return BDADDR_LE_RANDOM;
Johan Hedberg48264f02011-11-09 13:58:58 +02002467 }
Andre Guedes0ed09142012-04-03 08:46:54 -03002468
Johan Hedberg4c659c32011-11-07 23:13:39 +02002469 default:
Andre Guedes0ed09142012-04-03 08:46:54 -03002470 /* Fallback to BR/EDR type */
Andre Guedes591f47f2012-04-24 21:02:49 -03002471 return BDADDR_BREDR;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002472 }
2473}
2474
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002475static int get_connections(struct sock *sk, struct hci_dev *hdev, void *data,
2476 u16 data_len)
Johan Hedberg2784eb42011-01-21 13:56:35 +02002477{
Johan Hedberg2784eb42011-01-21 13:56:35 +02002478 struct mgmt_rp_get_connections *rp;
Luiz Augusto von Dentz8035ded2011-11-01 10:58:56 +02002479 struct hci_conn *c;
Johan Hedberga38528f2011-01-22 06:46:43 +02002480 size_t rp_len;
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002481 int err;
2482 u16 i;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002483
2484 BT_DBG("");
2485
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002486 hci_dev_lock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002487
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002488 if (!hdev_is_powered(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002489 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_GET_CONNECTIONS,
2490 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002491 goto unlock;
2492 }
2493
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002494 i = 0;
Johan Hedbergb644ba32012-01-17 21:48:47 +02002495 list_for_each_entry(c, &hdev->conn_hash.list, list) {
2496 if (test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002497 i++;
Johan Hedberg2784eb42011-01-21 13:56:35 +02002498 }
2499
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002500 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Andre Guedes92c4c202012-06-07 19:05:44 -03002501 rp = kmalloc(rp_len, GFP_KERNEL);
Johan Hedberga38528f2011-01-22 06:46:43 +02002502 if (!rp) {
Johan Hedberg2784eb42011-01-21 13:56:35 +02002503 err = -ENOMEM;
2504 goto unlock;
2505 }
2506
Johan Hedberg2784eb42011-01-21 13:56:35 +02002507 i = 0;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002508 list_for_each_entry(c, &hdev->conn_hash.list, list) {
Johan Hedbergb644ba32012-01-17 21:48:47 +02002509 if (!test_bit(HCI_CONN_MGMT_CONNECTED, &c->flags))
2510 continue;
Johan Hedberg4c659c32011-11-07 23:13:39 +02002511 bacpy(&rp->addr[i].bdaddr, &c->dst);
Andre Guedes57c14772012-04-24 21:02:50 -03002512 rp->addr[i].type = link_to_bdaddr(c->type, c->dst_type);
Andre Guedes0ed09142012-04-03 08:46:54 -03002513 if (c->type == SCO_LINK || c->type == ESCO_LINK)
Johan Hedberg4c659c32011-11-07 23:13:39 +02002514 continue;
2515 i++;
2516 }
2517
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02002518 rp->conn_count = cpu_to_le16(i);
Johan Hedberg60fc5fb62012-02-23 09:52:28 +02002519
Johan Hedberg4c659c32011-11-07 23:13:39 +02002520 /* Recalculate length in case of filtered SCO connections, etc */
2521 rp_len = sizeof(*rp) + (i * sizeof(struct mgmt_addr_info));
Johan Hedberg2784eb42011-01-21 13:56:35 +02002522
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002523 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONNECTIONS, 0, rp,
2524 rp_len);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002525
Johan Hedberga38528f2011-01-22 06:46:43 +02002526 kfree(rp);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002527
2528unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002529 hci_dev_unlock(hdev);
Johan Hedberg2784eb42011-01-21 13:56:35 +02002530 return err;
2531}
2532
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002533static int send_pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002534 struct mgmt_cp_pin_code_neg_reply *cp)
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002535{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002536 struct mgmt_pending_cmd *cmd;
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002537 int err;
2538
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002539 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_NEG_REPLY, hdev, cp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002540 sizeof(*cp));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002541 if (!cmd)
2542 return -ENOMEM;
2543
Johan Hedbergd8457692012-02-17 14:24:57 +02002544 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_NEG_REPLY,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002545 sizeof(cp->addr.bdaddr), &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002546 if (err < 0)
2547 mgmt_pending_remove(cmd);
2548
2549 return err;
2550}
2551
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002552static int pin_code_reply(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002553 u16 len)
Johan Hedberg980e1a52011-01-22 06:10:07 +02002554{
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002555 struct hci_conn *conn;
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002556 struct mgmt_cp_pin_code_reply *cp = data;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002557 struct hci_cp_pin_code_reply reply;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002558 struct mgmt_pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002559 int err;
2560
2561 BT_DBG("");
2562
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002563 hci_dev_lock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002564
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002565 if (!hdev_is_powered(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002566 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
2567 MGMT_STATUS_NOT_POWERED);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002568 goto failed;
2569 }
2570
Johan Hedbergd8457692012-02-17 14:24:57 +02002571 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &cp->addr.bdaddr);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002572 if (!conn) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002573 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
2574 MGMT_STATUS_NOT_CONNECTED);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002575 goto failed;
2576 }
2577
2578 if (conn->pending_sec_level == BT_SECURITY_HIGH && cp->pin_len != 16) {
Johan Hedbergd8457692012-02-17 14:24:57 +02002579 struct mgmt_cp_pin_code_neg_reply ncp;
2580
2581 memcpy(&ncp.addr, &cp->addr, sizeof(ncp.addr));
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002582
2583 BT_ERR("PIN code is not 16 bytes long");
2584
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002585 err = send_pin_code_neg_reply(sk, hdev, &ncp);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002586 if (err >= 0)
Johan Hedberga69e8372015-03-06 21:08:53 +02002587 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_PIN_CODE_REPLY,
2588 MGMT_STATUS_INVALID_PARAMS);
Waldemar Rymarkiewicz96d97a62011-06-01 17:28:48 +02002589
2590 goto failed;
2591 }
2592
Gustavo F. Padovan00abfe42012-03-01 00:37:10 -03002593 cmd = mgmt_pending_add(sk, MGMT_OP_PIN_CODE_REPLY, hdev, data, len);
Johan Hedberg366a0332011-02-19 12:05:55 -03002594 if (!cmd) {
2595 err = -ENOMEM;
Johan Hedberg980e1a52011-01-22 06:10:07 +02002596 goto failed;
Johan Hedberg366a0332011-02-19 12:05:55 -03002597 }
Johan Hedberg980e1a52011-01-22 06:10:07 +02002598
Johan Hedberg7776d1d2014-12-05 13:36:03 +02002599 cmd->cmd_complete = addr_cmd_complete;
2600
Johan Hedbergd8457692012-02-17 14:24:57 +02002601 bacpy(&reply.bdaddr, &cp->addr.bdaddr);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002602 reply.pin_len = cp->pin_len;
Waldemar Rymarkiewicz24718ca2011-06-01 17:28:47 +02002603 memcpy(reply.pin_code, cp->pin_code, sizeof(reply.pin_code));
Johan Hedberg980e1a52011-01-22 06:10:07 +02002604
2605 err = hci_send_cmd(hdev, HCI_OP_PIN_CODE_REPLY, sizeof(reply), &reply);
2606 if (err < 0)
Johan Hedberga664b5b2011-02-19 12:06:02 -03002607 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002608
2609failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002610 hci_dev_unlock(hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02002611 return err;
2612}
2613
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002614static int set_io_capability(struct sock *sk, struct hci_dev *hdev, void *data,
2615 u16 len)
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002616{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002617 struct mgmt_cp_set_io_capability *cp = data;
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002618
2619 BT_DBG("");
2620
Johan Hedberg4ec86d42014-06-17 15:14:48 +03002621 if (cp->io_capability > SMP_IO_KEYBOARD_DISPLAY)
Marcel Holtmann9db5c622016-08-29 06:31:57 +02002622 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_IO_CAPABILITY,
2623 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg4ec86d42014-06-17 15:14:48 +03002624
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002625 hci_dev_lock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002626
2627 hdev->io_capability = cp->io_capability;
2628
2629 BT_DBG("%s IO capability set to 0x%02x", hdev->name,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03002630 hdev->io_capability);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002631
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002632 hci_dev_unlock(hdev);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002633
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002634 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_IO_CAPABILITY, 0,
2635 NULL, 0);
Johan Hedberg17fa4b92011-01-25 13:28:33 +02002636}
2637
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002638static struct mgmt_pending_cmd *find_pairing(struct hci_conn *conn)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002639{
2640 struct hci_dev *hdev = conn->hdev;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002641 struct mgmt_pending_cmd *cmd;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002642
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002643 list_for_each_entry(cmd, &hdev->mgmt_pending, list) {
Johan Hedberge9a416b2011-02-19 12:05:56 -03002644 if (cmd->opcode != MGMT_OP_PAIR_DEVICE)
2645 continue;
2646
Johan Hedberge9a416b2011-02-19 12:05:56 -03002647 if (cmd->user_data != conn)
2648 continue;
2649
2650 return cmd;
2651 }
2652
2653 return NULL;
2654}
2655
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002656static int pairing_complete(struct mgmt_pending_cmd *cmd, u8 status)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002657{
2658 struct mgmt_rp_pair_device rp;
2659 struct hci_conn *conn = cmd->user_data;
Johan Hedberg9df74652014-12-19 22:26:03 +02002660 int err;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002661
Johan Hedberg61b1a7f2014-03-20 12:54:16 +02002662 bacpy(&rp.addr.bdaddr, &conn->dst);
2663 rp.addr.type = link_to_bdaddr(conn->type, conn->dst_type);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002664
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002665 err = mgmt_cmd_complete(cmd->sk, cmd->index, MGMT_OP_PAIR_DEVICE,
2666 status, &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002667
2668 /* So we don't get further callbacks for this connection */
2669 conn->connect_cfm_cb = NULL;
2670 conn->security_cfm_cb = NULL;
2671 conn->disconn_cfm_cb = NULL;
2672
David Herrmann76a68ba2013-04-06 20:28:37 +02002673 hci_conn_drop(conn);
Alfonso Acosta89cbb062014-10-11 21:44:47 +00002674
2675 /* The device is paired so there is no need to remove
2676 * its connection parameters anymore.
2677 */
2678 clear_bit(HCI_CONN_PARAM_REMOVAL_PEND, &conn->flags);
Johan Hedberg15013ae2014-12-11 21:45:44 +02002679
2680 hci_conn_put(conn);
Johan Hedberg9df74652014-12-19 22:26:03 +02002681
2682 return err;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002683}
2684
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002685void mgmt_smp_complete(struct hci_conn *conn, bool complete)
2686{
2687 u8 status = complete ? MGMT_STATUS_SUCCESS : MGMT_STATUS_FAILED;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002688 struct mgmt_pending_cmd *cmd;
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002689
2690 cmd = find_pairing(conn);
Johan Hedberga511b352014-12-11 21:45:45 +02002691 if (cmd) {
Johan Hedberg04ab2742014-12-05 13:36:04 +02002692 cmd->cmd_complete(cmd, status);
Johan Hedberga511b352014-12-11 21:45:45 +02002693 mgmt_pending_remove(cmd);
2694 }
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002695}
2696
Johan Hedberge9a416b2011-02-19 12:05:56 -03002697static void pairing_complete_cb(struct hci_conn *conn, u8 status)
2698{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002699 struct mgmt_pending_cmd *cmd;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002700
2701 BT_DBG("status %u", status);
2702
Johan Hedberg56e5cb82011-11-08 20:40:16 +02002703 cmd = find_pairing(conn);
Johan Hedberga511b352014-12-11 21:45:45 +02002704 if (!cmd) {
Johan Hedberg56e5cb82011-11-08 20:40:16 +02002705 BT_DBG("Unable to find a pending command");
Johan Hedberga511b352014-12-11 21:45:45 +02002706 return;
2707 }
2708
2709 cmd->cmd_complete(cmd, mgmt_status(status));
2710 mgmt_pending_remove(cmd);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002711}
2712
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002713static void le_pairing_complete_cb(struct hci_conn *conn, u8 status)
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302714{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002715 struct mgmt_pending_cmd *cmd;
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302716
2717 BT_DBG("status %u", status);
2718
2719 if (!status)
2720 return;
2721
2722 cmd = find_pairing(conn);
Johan Hedberga511b352014-12-11 21:45:45 +02002723 if (!cmd) {
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302724 BT_DBG("Unable to find a pending command");
Johan Hedberga511b352014-12-11 21:45:45 +02002725 return;
2726 }
2727
2728 cmd->cmd_complete(cmd, mgmt_status(status));
2729 mgmt_pending_remove(cmd);
Vishal Agarwal4c47d732012-06-07 20:27:35 +05302730}
2731
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002732static int pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002733 u16 len)
Johan Hedberge9a416b2011-02-19 12:05:56 -03002734{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002735 struct mgmt_cp_pair_device *cp = data;
Johan Hedberg1425acb2011-11-11 00:07:35 +02002736 struct mgmt_rp_pair_device rp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002737 struct mgmt_pending_cmd *cmd;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002738 u8 sec_level, auth_type;
2739 struct hci_conn *conn;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002740 int err;
2741
2742 BT_DBG("");
2743
Szymon Jancf950a30e2013-01-18 12:48:07 +01002744 memset(&rp, 0, sizeof(rp));
2745 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
2746 rp.addr.type = cp->addr.type;
2747
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002748 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002749 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2750 MGMT_STATUS_INVALID_PARAMS,
2751 &rp, sizeof(rp));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02002752
Johan Hedberg4ec86d42014-06-17 15:14:48 +03002753 if (cp->io_cap > SMP_IO_KEYBOARD_DISPLAY)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002754 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2755 MGMT_STATUS_INVALID_PARAMS,
2756 &rp, sizeof(rp));
Johan Hedberg4ec86d42014-06-17 15:14:48 +03002757
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002758 hci_dev_lock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002759
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002760 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002761 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2762 MGMT_STATUS_NOT_POWERED, &rp,
2763 sizeof(rp));
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002764 goto unlock;
2765 }
2766
Johan Hedberg55e76b32015-03-10 22:34:40 +02002767 if (hci_bdaddr_is_paired(hdev, &cp->addr.bdaddr, cp->addr.type)) {
2768 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2769 MGMT_STATUS_ALREADY_PAIRED, &rp,
2770 sizeof(rp));
2771 goto unlock;
2772 }
2773
Vinicius Costa Gomesc908df32011-09-02 14:51:22 -03002774 sec_level = BT_SECURITY_MEDIUM;
Mikel Astiz6fd6b912014-04-08 14:21:32 +02002775 auth_type = HCI_AT_DEDICATED_BONDING;
Johan Hedberge9a416b2011-02-19 12:05:56 -03002776
Andre Guedes6f77d8c2014-02-26 20:21:45 -03002777 if (cp->addr.type == BDADDR_BREDR) {
Andre Guedes04a6c582014-02-26 20:21:44 -03002778 conn = hci_connect_acl(hdev, &cp->addr.bdaddr, sec_level,
2779 auth_type);
Andre Guedes6f77d8c2014-02-26 20:21:45 -03002780 } else {
Johan Hedberg85813a72015-10-21 18:02:59 +03002781 u8 addr_type = le_addr_type(cp->addr.type);
Jakub Pawlowski5157b8a2015-10-16 10:07:54 +03002782 struct hci_conn_params *p;
Andre Guedes6f77d8c2014-02-26 20:21:45 -03002783
Marcel Holtmann7c264b12014-06-30 12:34:40 +02002784 /* When pairing a new device, it is expected to remember
2785 * this device for future connections. Adding the connection
2786 * parameter information ahead of time allows tracking
2787 * of the slave preferred values and will speed up any
2788 * further connection establishment.
2789 *
2790 * If connection parameters already exist, then they
2791 * will be kept and this function does nothing.
2792 */
Jakub Pawlowski5157b8a2015-10-16 10:07:54 +03002793 p = hci_conn_params_add(hdev, &cp->addr.bdaddr, addr_type);
2794
2795 if (p->auto_connect == HCI_AUTO_CONN_EXPLICIT)
2796 p->auto_connect = HCI_AUTO_CONN_DISABLED;
Marcel Holtmann7c264b12014-06-30 12:34:40 +02002797
Jakub Pawlowskifa142222015-08-07 20:22:56 +02002798 conn = hci_connect_le_scan(hdev, &cp->addr.bdaddr,
2799 addr_type, sec_level,
Johan Hedberg0ad06aa2015-11-11 14:44:57 +02002800 HCI_LE_CONN_TIMEOUT);
Andre Guedes6f77d8c2014-02-26 20:21:45 -03002801 }
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002802
Ville Tervo30e76272011-02-22 16:10:53 -03002803 if (IS_ERR(conn)) {
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002804 int status;
2805
2806 if (PTR_ERR(conn) == -EBUSY)
2807 status = MGMT_STATUS_BUSY;
Lukasz Rymanowskifaa81032015-02-11 12:31:42 +01002808 else if (PTR_ERR(conn) == -EOPNOTSUPP)
2809 status = MGMT_STATUS_NOT_SUPPORTED;
2810 else if (PTR_ERR(conn) == -ECONNREFUSED)
2811 status = MGMT_STATUS_REJECTED;
Andrzej Kaczmarek489dc482012-05-30 15:39:22 +02002812 else
2813 status = MGMT_STATUS_CONNECT_FAILED;
2814
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002815 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2816 status, &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002817 goto unlock;
2818 }
2819
2820 if (conn->connect_cfm_cb) {
David Herrmann76a68ba2013-04-06 20:28:37 +02002821 hci_conn_drop(conn);
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002822 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_PAIR_DEVICE,
2823 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberge9a416b2011-02-19 12:05:56 -03002824 goto unlock;
2825 }
2826
Johan Hedberg2e58ef32011-11-08 20:40:15 +02002827 cmd = mgmt_pending_add(sk, MGMT_OP_PAIR_DEVICE, hdev, data, len);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002828 if (!cmd) {
2829 err = -ENOMEM;
David Herrmann76a68ba2013-04-06 20:28:37 +02002830 hci_conn_drop(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002831 goto unlock;
2832 }
2833
Johan Hedberg04ab2742014-12-05 13:36:04 +02002834 cmd->cmd_complete = pairing_complete;
2835
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002836 /* For LE, just connecting isn't a proof that the pairing finished */
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002837 if (cp->addr.type == BDADDR_BREDR) {
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002838 conn->connect_cfm_cb = pairing_complete_cb;
Johan Hedbergf4a407b2014-02-18 21:41:34 +02002839 conn->security_cfm_cb = pairing_complete_cb;
2840 conn->disconn_cfm_cb = pairing_complete_cb;
2841 } else {
2842 conn->connect_cfm_cb = le_pairing_complete_cb;
2843 conn->security_cfm_cb = le_pairing_complete_cb;
2844 conn->disconn_cfm_cb = le_pairing_complete_cb;
2845 }
Vinicius Costa Gomes7a512d02011-08-19 21:06:54 -03002846
Johan Hedberge9a416b2011-02-19 12:05:56 -03002847 conn->io_capability = cp->io_cap;
Johan Hedbergf8aaf9b2014-08-17 23:28:57 +03002848 cmd->user_data = hci_conn_get(conn);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002849
Johan Hedberg6f78fd42014-07-30 08:35:48 +03002850 if ((conn->state == BT_CONNECTED || conn->state == BT_CONFIG) &&
Johan Hedberga511b352014-12-11 21:45:45 +02002851 hci_conn_security(conn, sec_level, auth_type, true)) {
2852 cmd->cmd_complete(cmd, 0);
2853 mgmt_pending_remove(cmd);
2854 }
Johan Hedberge9a416b2011-02-19 12:05:56 -03002855
2856 err = 0;
2857
2858unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002859 hci_dev_unlock(hdev);
Johan Hedberge9a416b2011-02-19 12:05:56 -03002860 return err;
2861}
2862
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002863static int cancel_pair_device(struct sock *sk, struct hci_dev *hdev, void *data,
2864 u16 len)
Johan Hedberg28424702012-02-02 04:02:29 +02002865{
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02002866 struct mgmt_addr_info *addr = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002867 struct mgmt_pending_cmd *cmd;
Johan Hedberg28424702012-02-02 04:02:29 +02002868 struct hci_conn *conn;
2869 int err;
2870
2871 BT_DBG("");
2872
Johan Hedberg28424702012-02-02 04:02:29 +02002873 hci_dev_lock(hdev);
2874
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002875 if (!hdev_is_powered(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002876 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
2877 MGMT_STATUS_NOT_POWERED);
Johan Hedberg5f97c1d2012-02-22 22:41:18 +02002878 goto unlock;
2879 }
2880
Johan Hedberg333ae952015-03-17 13:48:47 +02002881 cmd = pending_find(MGMT_OP_PAIR_DEVICE, hdev);
Johan Hedberg28424702012-02-02 04:02:29 +02002882 if (!cmd) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002883 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
2884 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002885 goto unlock;
2886 }
2887
2888 conn = cmd->user_data;
2889
2890 if (bacmp(&addr->bdaddr, &conn->dst) != 0) {
Johan Hedberga69e8372015-03-06 21:08:53 +02002891 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE,
2892 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg28424702012-02-02 04:02:29 +02002893 goto unlock;
2894 }
2895
Johan Hedberga511b352014-12-11 21:45:45 +02002896 cmd->cmd_complete(cmd, MGMT_STATUS_CANCELLED);
2897 mgmt_pending_remove(cmd);
Johan Hedberg28424702012-02-02 04:02:29 +02002898
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002899 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_CANCEL_PAIR_DEVICE, 0,
2900 addr, sizeof(*addr));
Johan Hedberg28424702012-02-02 04:02:29 +02002901unlock:
2902 hci_dev_unlock(hdev);
Johan Hedberg28424702012-02-02 04:02:29 +02002903 return err;
2904}
2905
Johan Hedbergbdb6d972012-02-28 06:13:32 +02002906static int user_pairing_resp(struct sock *sk, struct hci_dev *hdev,
Johan Hedberg1707c602013-03-15 17:07:15 -05002907 struct mgmt_addr_info *addr, u16 mgmt_op,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002908 u16 hci_op, __le32 passkey)
Johan Hedberga5c29682011-02-19 12:05:57 -03002909{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02002910 struct mgmt_pending_cmd *cmd;
Brian Gix0df4c182011-11-16 13:53:13 -08002911 struct hci_conn *conn;
Johan Hedberga5c29682011-02-19 12:05:57 -03002912 int err;
2913
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002914 hci_dev_lock(hdev);
Johan Hedberg08ba5382011-03-16 14:29:34 +02002915
Johan Hedberg4b34ee782012-02-21 14:13:02 +02002916 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002917 err = mgmt_cmd_complete(sk, hdev->id, mgmt_op,
2918 MGMT_STATUS_NOT_POWERED, addr,
2919 sizeof(*addr));
Brian Gix0df4c182011-11-16 13:53:13 -08002920 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002921 }
2922
Johan Hedberg1707c602013-03-15 17:07:15 -05002923 if (addr->type == BDADDR_BREDR)
2924 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK, &addr->bdaddr);
Johan Hedberg272d90d2012-02-09 15:26:12 +02002925 else
Johan Hedberg9d4c1cc2015-10-21 18:03:01 +03002926 conn = hci_conn_hash_lookup_le(hdev, &addr->bdaddr,
2927 le_addr_type(addr->type));
Brian Gix47c15e22011-11-16 13:53:14 -08002928
Johan Hedberg272d90d2012-02-09 15:26:12 +02002929 if (!conn) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002930 err = mgmt_cmd_complete(sk, hdev->id, mgmt_op,
2931 MGMT_STATUS_NOT_CONNECTED, addr,
2932 sizeof(*addr));
Johan Hedberg272d90d2012-02-09 15:26:12 +02002933 goto done;
2934 }
2935
Johan Hedberg1707c602013-03-15 17:07:15 -05002936 if (addr->type == BDADDR_LE_PUBLIC || addr->type == BDADDR_LE_RANDOM) {
Brian Gix5fe57d92011-12-21 16:12:13 -08002937 err = smp_user_confirm_reply(conn, mgmt_op, passkey);
Brian Gix5fe57d92011-12-21 16:12:13 -08002938 if (!err)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002939 err = mgmt_cmd_complete(sk, hdev->id, mgmt_op,
2940 MGMT_STATUS_SUCCESS, addr,
2941 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002942 else
Johan Hedberg2a1afb52015-03-06 21:08:54 +02002943 err = mgmt_cmd_complete(sk, hdev->id, mgmt_op,
2944 MGMT_STATUS_FAILED, addr,
2945 sizeof(*addr));
Brian Gix5fe57d92011-12-21 16:12:13 -08002946
Brian Gix47c15e22011-11-16 13:53:14 -08002947 goto done;
2948 }
2949
Johan Hedberg1707c602013-03-15 17:07:15 -05002950 cmd = mgmt_pending_add(sk, mgmt_op, hdev, addr, sizeof(*addr));
Johan Hedberga5c29682011-02-19 12:05:57 -03002951 if (!cmd) {
2952 err = -ENOMEM;
Brian Gix0df4c182011-11-16 13:53:13 -08002953 goto done;
Johan Hedberga5c29682011-02-19 12:05:57 -03002954 }
2955
Johan Hedberg7776d1d2014-12-05 13:36:03 +02002956 cmd->cmd_complete = addr_cmd_complete;
2957
Brian Gix0df4c182011-11-16 13:53:13 -08002958 /* Continue with pairing via HCI */
Brian Gix604086b2011-11-23 08:28:33 -08002959 if (hci_op == HCI_OP_USER_PASSKEY_REPLY) {
2960 struct hci_cp_user_passkey_reply cp;
2961
Johan Hedberg1707c602013-03-15 17:07:15 -05002962 bacpy(&cp.bdaddr, &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002963 cp.passkey = passkey;
2964 err = hci_send_cmd(hdev, hci_op, sizeof(cp), &cp);
2965 } else
Johan Hedberg1707c602013-03-15 17:07:15 -05002966 err = hci_send_cmd(hdev, hci_op, sizeof(addr->bdaddr),
2967 &addr->bdaddr);
Brian Gix604086b2011-11-23 08:28:33 -08002968
Johan Hedberga664b5b2011-02-19 12:06:02 -03002969 if (err < 0)
2970 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03002971
Brian Gix0df4c182011-11-16 13:53:13 -08002972done:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03002973 hci_dev_unlock(hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03002974 return err;
2975}
2976
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05302977static int pin_code_neg_reply(struct sock *sk, struct hci_dev *hdev,
2978 void *data, u16 len)
2979{
2980 struct mgmt_cp_pin_code_neg_reply *cp = data;
2981
2982 BT_DBG("");
2983
Johan Hedberg1707c602013-03-15 17:07:15 -05002984 return user_pairing_resp(sk, hdev, &cp->addr,
Jaganath Kanakkasseryafeb0192012-07-09 16:11:51 +05302985 MGMT_OP_PIN_CODE_NEG_REPLY,
2986 HCI_OP_PIN_CODE_NEG_REPLY, 0);
2987}
2988
Gustavo F. Padovan04124682012-03-08 01:25:00 -03002989static int user_confirm_reply(struct sock *sk, struct hci_dev *hdev, void *data,
2990 u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08002991{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03002992 struct mgmt_cp_user_confirm_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08002993
2994 BT_DBG("");
2995
2996 if (len != sizeof(*cp))
Johan Hedberga69e8372015-03-06 21:08:53 +02002997 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_USER_CONFIRM_REPLY,
2998 MGMT_STATUS_INVALID_PARAMS);
Brian Gix0df4c182011-11-16 13:53:13 -08002999
Johan Hedberg1707c602013-03-15 17:07:15 -05003000 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003001 MGMT_OP_USER_CONFIRM_REPLY,
3002 HCI_OP_USER_CONFIRM_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08003003}
3004
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003005static int user_confirm_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003006 void *data, u16 len)
Brian Gix0df4c182011-11-16 13:53:13 -08003007{
Johan Hedbergc9c26592011-12-15 00:47:41 +02003008 struct mgmt_cp_user_confirm_neg_reply *cp = data;
Brian Gix0df4c182011-11-16 13:53:13 -08003009
3010 BT_DBG("");
3011
Johan Hedberg1707c602013-03-15 17:07:15 -05003012 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003013 MGMT_OP_USER_CONFIRM_NEG_REPLY,
3014 HCI_OP_USER_CONFIRM_NEG_REPLY, 0);
Brian Gix0df4c182011-11-16 13:53:13 -08003015}
3016
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003017static int user_passkey_reply(struct sock *sk, struct hci_dev *hdev, void *data,
3018 u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08003019{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003020 struct mgmt_cp_user_passkey_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08003021
3022 BT_DBG("");
3023
Johan Hedberg1707c602013-03-15 17:07:15 -05003024 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003025 MGMT_OP_USER_PASSKEY_REPLY,
3026 HCI_OP_USER_PASSKEY_REPLY, cp->passkey);
Brian Gix604086b2011-11-23 08:28:33 -08003027}
3028
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003029static int user_passkey_neg_reply(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003030 void *data, u16 len)
Brian Gix604086b2011-11-23 08:28:33 -08003031{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003032 struct mgmt_cp_user_passkey_neg_reply *cp = data;
Brian Gix604086b2011-11-23 08:28:33 -08003033
3034 BT_DBG("");
3035
Johan Hedberg1707c602013-03-15 17:07:15 -05003036 return user_pairing_resp(sk, hdev, &cp->addr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003037 MGMT_OP_USER_PASSKEY_NEG_REPLY,
3038 HCI_OP_USER_PASSKEY_NEG_REPLY, 0);
Brian Gix604086b2011-11-23 08:28:33 -08003039}
3040
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02003041static void adv_expire(struct hci_dev *hdev, u32 flags)
3042{
3043 struct adv_info *adv_instance;
3044 struct hci_request req;
3045 int err;
3046
3047 adv_instance = hci_find_adv_instance(hdev, hdev->cur_adv_instance);
3048 if (!adv_instance)
3049 return;
3050
3051 /* stop if current instance doesn't need to be changed */
3052 if (!(adv_instance->flags & flags))
3053 return;
3054
3055 cancel_adv_timeout(hdev);
3056
3057 adv_instance = hci_get_next_instance(hdev, adv_instance->instance);
3058 if (!adv_instance)
3059 return;
3060
3061 hci_req_init(&req, hdev);
3062 err = __hci_req_schedule_adv_instance(&req, adv_instance->instance,
3063 true);
3064 if (err)
3065 return;
3066
3067 hci_req_run(&req, NULL);
3068}
3069
Marcel Holtmann1904a852015-01-11 13:50:44 -08003070static void set_name_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg13928972013-03-15 17:07:00 -05003071{
3072 struct mgmt_cp_set_local_name *cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003073 struct mgmt_pending_cmd *cmd;
Johan Hedberg13928972013-03-15 17:07:00 -05003074
3075 BT_DBG("status 0x%02x", status);
3076
3077 hci_dev_lock(hdev);
3078
Johan Hedberg333ae952015-03-17 13:48:47 +02003079 cmd = pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
Johan Hedberg13928972013-03-15 17:07:00 -05003080 if (!cmd)
3081 goto unlock;
3082
3083 cp = cmd->param;
3084
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02003085 if (status) {
Johan Hedberga69e8372015-03-06 21:08:53 +02003086 mgmt_cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME,
3087 mgmt_status(status));
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02003088 } else {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003089 mgmt_cmd_complete(cmd->sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
3090 cp, sizeof(*cp));
Johan Hedberg13928972013-03-15 17:07:00 -05003091
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02003092 if (hci_dev_test_flag(hdev, HCI_LE_ADV))
3093 adv_expire(hdev, MGMT_ADV_FLAG_LOCAL_NAME);
3094 }
3095
Johan Hedberg13928972013-03-15 17:07:00 -05003096 mgmt_pending_remove(cmd);
3097
3098unlock:
3099 hci_dev_unlock(hdev);
3100}
3101
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003102static int set_local_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003103 u16 len)
Johan Hedbergb312b1612011-03-16 14:29:37 +02003104{
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003105 struct mgmt_cp_set_local_name *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003106 struct mgmt_pending_cmd *cmd;
Johan Hedberg890ea892013-03-15 17:06:52 -05003107 struct hci_request req;
Johan Hedbergb312b1612011-03-16 14:29:37 +02003108 int err;
3109
3110 BT_DBG("");
3111
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003112 hci_dev_lock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003113
Johan Hedbergb3f2ca92013-03-15 17:07:03 -05003114 /* If the old values are the same as the new ones just return a
3115 * direct command complete event.
3116 */
3117 if (!memcmp(hdev->dev_name, cp->name, sizeof(hdev->dev_name)) &&
3118 !memcmp(hdev->short_name, cp->short_name,
3119 sizeof(hdev->short_name))) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003120 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
3121 data, len);
Johan Hedbergb3f2ca92013-03-15 17:07:03 -05003122 goto failed;
3123 }
3124
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003125 memcpy(hdev->short_name, cp->short_name, sizeof(hdev->short_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003126
Johan Hedbergb5235a62012-02-21 14:32:24 +02003127 if (!hdev_is_powered(hdev)) {
Johan Hedberg2b4bf392012-03-03 00:19:06 +02003128 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003129
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003130 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_LOCAL_NAME, 0,
3131 data, len);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003132 if (err < 0)
3133 goto failed;
3134
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02003135 err = mgmt_limited_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, data,
3136 len, HCI_MGMT_LOCAL_NAME_EVENTS, sk);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02003137 ext_info_changed(hdev, sk);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003138
Johan Hedbergb5235a62012-02-21 14:32:24 +02003139 goto failed;
3140 }
3141
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02003142 cmd = mgmt_pending_add(sk, MGMT_OP_SET_LOCAL_NAME, hdev, data, len);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003143 if (!cmd) {
3144 err = -ENOMEM;
3145 goto failed;
3146 }
3147
Johan Hedberg13928972013-03-15 17:07:00 -05003148 memcpy(hdev->dev_name, cp->name, sizeof(hdev->dev_name));
3149
Johan Hedberg890ea892013-03-15 17:06:52 -05003150 hci_req_init(&req, hdev);
Johan Hedberg3f985052013-03-15 17:07:02 -05003151
3152 if (lmp_bredr_capable(hdev)) {
Johan Hedberg00cf5042015-11-25 16:15:41 +02003153 __hci_req_update_name(&req);
Johan Hedbergb1a89172015-11-25 16:15:42 +02003154 __hci_req_update_eir(&req);
Johan Hedberg3f985052013-03-15 17:07:02 -05003155 }
3156
Marcel Holtmann7a5f4992013-10-16 00:16:49 -07003157 /* The name is stored in the scan response data and so
3158 * no need to udpate the advertising data here.
3159 */
Johan Hedberg3f985052013-03-15 17:07:02 -05003160 if (lmp_le_capable(hdev))
Johan Hedbergcab054a2015-11-30 11:21:45 +02003161 __hci_req_update_scan_rsp_data(&req, hdev->cur_adv_instance);
Johan Hedberg3f985052013-03-15 17:07:02 -05003162
Johan Hedberg13928972013-03-15 17:07:00 -05003163 err = hci_req_run(&req, set_name_complete);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003164 if (err < 0)
3165 mgmt_pending_remove(cmd);
3166
3167failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003168 hci_dev_unlock(hdev);
Johan Hedbergb312b1612011-03-16 14:29:37 +02003169 return err;
3170}
3171
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02003172static int set_appearance(struct sock *sk, struct hci_dev *hdev, void *data,
3173 u16 len)
3174{
3175 struct mgmt_cp_set_appearance *cp = data;
3176 u16 apperance;
3177 int err;
3178
3179 BT_DBG("");
3180
3181 apperance = le16_to_cpu(cp->appearance);
3182
3183 hci_dev_lock(hdev);
3184
3185 if (hdev->appearance != apperance) {
3186 hdev->appearance = apperance;
3187
3188 if (hci_dev_test_flag(hdev, HCI_LE_ADV))
3189 adv_expire(hdev, MGMT_ADV_FLAG_APPEARANCE);
MichaƂ Narajowskie74317f2016-09-19 20:25:56 +02003190
3191 ext_info_changed(hdev, sk);
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02003192 }
3193
3194 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_APPEARANCE, 0, NULL,
3195 0);
3196
3197 hci_dev_unlock(hdev);
3198
3199 return err;
3200}
3201
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003202static void read_local_oob_data_complete(struct hci_dev *hdev, u8 status,
3203 u16 opcode, struct sk_buff *skb)
3204{
3205 struct mgmt_rp_read_local_oob_data mgmt_rp;
3206 size_t rp_size = sizeof(mgmt_rp);
3207 struct mgmt_pending_cmd *cmd;
3208
3209 BT_DBG("%s status %u", hdev->name, status);
3210
3211 cmd = pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev);
3212 if (!cmd)
3213 return;
3214
3215 if (status || !skb) {
3216 mgmt_cmd_status(cmd->sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3217 status ? mgmt_status(status) : MGMT_STATUS_FAILED);
3218 goto remove;
3219 }
3220
3221 memset(&mgmt_rp, 0, sizeof(mgmt_rp));
3222
3223 if (opcode == HCI_OP_READ_LOCAL_OOB_DATA) {
3224 struct hci_rp_read_local_oob_data *rp = (void *) skb->data;
3225
3226 if (skb->len < sizeof(*rp)) {
3227 mgmt_cmd_status(cmd->sk, hdev->id,
3228 MGMT_OP_READ_LOCAL_OOB_DATA,
3229 MGMT_STATUS_FAILED);
3230 goto remove;
3231 }
3232
3233 memcpy(mgmt_rp.hash192, rp->hash, sizeof(rp->hash));
3234 memcpy(mgmt_rp.rand192, rp->rand, sizeof(rp->rand));
3235
3236 rp_size -= sizeof(mgmt_rp.hash256) + sizeof(mgmt_rp.rand256);
3237 } else {
3238 struct hci_rp_read_local_oob_ext_data *rp = (void *) skb->data;
3239
3240 if (skb->len < sizeof(*rp)) {
3241 mgmt_cmd_status(cmd->sk, hdev->id,
3242 MGMT_OP_READ_LOCAL_OOB_DATA,
3243 MGMT_STATUS_FAILED);
3244 goto remove;
3245 }
3246
3247 memcpy(mgmt_rp.hash192, rp->hash192, sizeof(rp->hash192));
3248 memcpy(mgmt_rp.rand192, rp->rand192, sizeof(rp->rand192));
3249
3250 memcpy(mgmt_rp.hash256, rp->hash256, sizeof(rp->hash256));
3251 memcpy(mgmt_rp.rand256, rp->rand256, sizeof(rp->rand256));
3252 }
3253
3254 mgmt_cmd_complete(cmd->sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3255 MGMT_STATUS_SUCCESS, &mgmt_rp, rp_size);
3256
3257remove:
3258 mgmt_pending_remove(cmd);
3259}
3260
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02003261static int read_local_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003262 void *data, u16 data_len)
Szymon Jancc35938b2011-03-22 13:12:21 +01003263{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003264 struct mgmt_pending_cmd *cmd;
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003265 struct hci_request req;
Szymon Jancc35938b2011-03-22 13:12:21 +01003266 int err;
3267
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003268 BT_DBG("%s", hdev->name);
Szymon Jancc35938b2011-03-22 13:12:21 +01003269
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003270 hci_dev_lock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003271
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003272 if (!hdev_is_powered(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02003273 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3274 MGMT_STATUS_NOT_POWERED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003275 goto unlock;
3276 }
3277
Andre Guedes9a1a1992012-07-24 15:03:48 -03003278 if (!lmp_ssp_capable(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02003279 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3280 MGMT_STATUS_NOT_SUPPORTED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003281 goto unlock;
3282 }
3283
Johan Hedberg333ae952015-03-17 13:48:47 +02003284 if (pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02003285 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3286 MGMT_STATUS_BUSY);
Szymon Jancc35938b2011-03-22 13:12:21 +01003287 goto unlock;
3288 }
3289
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003290 cmd = mgmt_pending_add(sk, MGMT_OP_READ_LOCAL_OOB_DATA, hdev, NULL, 0);
Szymon Jancc35938b2011-03-22 13:12:21 +01003291 if (!cmd) {
3292 err = -ENOMEM;
3293 goto unlock;
3294 }
3295
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003296 hci_req_init(&req, hdev);
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08003297
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003298 if (bredr_sc_enabled(hdev))
3299 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_EXT_DATA, 0, NULL);
3300 else
3301 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_DATA, 0, NULL);
3302
3303 err = hci_req_run_skb(&req, read_local_oob_data_complete);
Szymon Jancc35938b2011-03-22 13:12:21 +01003304 if (err < 0)
3305 mgmt_pending_remove(cmd);
3306
3307unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003308 hci_dev_unlock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003309 return err;
3310}
3311
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003312static int add_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003313 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003314{
Johan Hedberg5d57e792015-01-23 10:10:38 +02003315 struct mgmt_addr_info *addr = data;
Szymon Janc2763eda2011-03-22 13:12:22 +01003316 int err;
3317
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003318 BT_DBG("%s ", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003319
Johan Hedberg5d57e792015-01-23 10:10:38 +02003320 if (!bdaddr_type_is_valid(addr->type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003321 return mgmt_cmd_complete(sk, hdev->id,
3322 MGMT_OP_ADD_REMOTE_OOB_DATA,
3323 MGMT_STATUS_INVALID_PARAMS,
3324 addr, sizeof(*addr));
Johan Hedberg5d57e792015-01-23 10:10:38 +02003325
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003326 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003327
Marcel Holtmannec109112014-01-10 02:07:30 -08003328 if (len == MGMT_ADD_REMOTE_OOB_DATA_SIZE) {
3329 struct mgmt_cp_add_remote_oob_data *cp = data;
3330 u8 status;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003331
Johan Hedbergc19a4952014-11-17 20:52:19 +02003332 if (cp->addr.type != BDADDR_BREDR) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003333 err = mgmt_cmd_complete(sk, hdev->id,
3334 MGMT_OP_ADD_REMOTE_OOB_DATA,
3335 MGMT_STATUS_INVALID_PARAMS,
3336 &cp->addr, sizeof(cp->addr));
Johan Hedbergc19a4952014-11-17 20:52:19 +02003337 goto unlock;
3338 }
3339
Marcel Holtmannec109112014-01-10 02:07:30 -08003340 err = hci_add_remote_oob_data(hdev, &cp->addr.bdaddr,
Johan Hedberg6928a922014-10-26 20:46:09 +01003341 cp->addr.type, cp->hash,
3342 cp->rand, NULL, NULL);
Marcel Holtmannec109112014-01-10 02:07:30 -08003343 if (err < 0)
3344 status = MGMT_STATUS_FAILED;
3345 else
3346 status = MGMT_STATUS_SUCCESS;
3347
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003348 err = mgmt_cmd_complete(sk, hdev->id,
3349 MGMT_OP_ADD_REMOTE_OOB_DATA, status,
3350 &cp->addr, sizeof(cp->addr));
Marcel Holtmannec109112014-01-10 02:07:30 -08003351 } else if (len == MGMT_ADD_REMOTE_OOB_EXT_DATA_SIZE) {
3352 struct mgmt_cp_add_remote_oob_ext_data *cp = data;
Marcel Holtmann41bcfd52015-01-31 00:37:02 -08003353 u8 *rand192, *hash192, *rand256, *hash256;
Marcel Holtmannec109112014-01-10 02:07:30 -08003354 u8 status;
3355
Johan Hedberg86df9202014-10-26 20:52:27 +01003356 if (bdaddr_type_is_le(cp->addr.type)) {
Johan Hedbergd25b78e2015-01-27 12:55:52 +02003357 /* Enforce zero-valued 192-bit parameters as
3358 * long as legacy SMP OOB isn't implemented.
3359 */
3360 if (memcmp(cp->rand192, ZERO_KEY, 16) ||
3361 memcmp(cp->hash192, ZERO_KEY, 16)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003362 err = mgmt_cmd_complete(sk, hdev->id,
3363 MGMT_OP_ADD_REMOTE_OOB_DATA,
3364 MGMT_STATUS_INVALID_PARAMS,
3365 addr, sizeof(*addr));
Johan Hedbergd25b78e2015-01-27 12:55:52 +02003366 goto unlock;
3367 }
3368
Johan Hedberg86df9202014-10-26 20:52:27 +01003369 rand192 = NULL;
3370 hash192 = NULL;
3371 } else {
Marcel Holtmann41bcfd52015-01-31 00:37:02 -08003372 /* In case one of the P-192 values is set to zero,
3373 * then just disable OOB data for P-192.
3374 */
3375 if (!memcmp(cp->rand192, ZERO_KEY, 16) ||
3376 !memcmp(cp->hash192, ZERO_KEY, 16)) {
3377 rand192 = NULL;
3378 hash192 = NULL;
3379 } else {
3380 rand192 = cp->rand192;
3381 hash192 = cp->hash192;
3382 }
3383 }
3384
3385 /* In case one of the P-256 values is set to zero, then just
3386 * disable OOB data for P-256.
3387 */
3388 if (!memcmp(cp->rand256, ZERO_KEY, 16) ||
3389 !memcmp(cp->hash256, ZERO_KEY, 16)) {
3390 rand256 = NULL;
3391 hash256 = NULL;
3392 } else {
3393 rand256 = cp->rand256;
3394 hash256 = cp->hash256;
Johan Hedberg86df9202014-10-26 20:52:27 +01003395 }
3396
Johan Hedberg81328d52014-10-26 20:33:47 +01003397 err = hci_add_remote_oob_data(hdev, &cp->addr.bdaddr,
Johan Hedberg86df9202014-10-26 20:52:27 +01003398 cp->addr.type, hash192, rand192,
Marcel Holtmann41bcfd52015-01-31 00:37:02 -08003399 hash256, rand256);
Marcel Holtmannec109112014-01-10 02:07:30 -08003400 if (err < 0)
3401 status = MGMT_STATUS_FAILED;
3402 else
3403 status = MGMT_STATUS_SUCCESS;
3404
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003405 err = mgmt_cmd_complete(sk, hdev->id,
3406 MGMT_OP_ADD_REMOTE_OOB_DATA,
3407 status, &cp->addr, sizeof(cp->addr));
Marcel Holtmannec109112014-01-10 02:07:30 -08003408 } else {
3409 BT_ERR("add_remote_oob_data: invalid length of %u bytes", len);
Johan Hedberga69e8372015-03-06 21:08:53 +02003410 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3411 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmannec109112014-01-10 02:07:30 -08003412 }
Szymon Janc2763eda2011-03-22 13:12:22 +01003413
Johan Hedbergc19a4952014-11-17 20:52:19 +02003414unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003415 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003416 return err;
3417}
3418
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003419static int remove_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003420 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003421{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003422 struct mgmt_cp_remove_remote_oob_data *cp = data;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003423 u8 status;
Szymon Janc2763eda2011-03-22 13:12:22 +01003424 int err;
3425
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003426 BT_DBG("%s", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003427
Johan Hedbergc19a4952014-11-17 20:52:19 +02003428 if (cp->addr.type != BDADDR_BREDR)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003429 return mgmt_cmd_complete(sk, hdev->id,
3430 MGMT_OP_REMOVE_REMOTE_OOB_DATA,
3431 MGMT_STATUS_INVALID_PARAMS,
3432 &cp->addr, sizeof(cp->addr));
Johan Hedbergc19a4952014-11-17 20:52:19 +02003433
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003434 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003435
Johan Hedbergeedbd582014-11-15 09:34:23 +02003436 if (!bacmp(&cp->addr.bdaddr, BDADDR_ANY)) {
3437 hci_remote_oob_data_clear(hdev);
3438 status = MGMT_STATUS_SUCCESS;
3439 goto done;
3440 }
3441
Johan Hedberg6928a922014-10-26 20:46:09 +01003442 err = hci_remove_remote_oob_data(hdev, &cp->addr.bdaddr, cp->addr.type);
Szymon Janc2763eda2011-03-22 13:12:22 +01003443 if (err < 0)
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003444 status = MGMT_STATUS_INVALID_PARAMS;
Szymon Janc2763eda2011-03-22 13:12:22 +01003445 else
Szymon Janca6785be2012-12-13 15:11:21 +01003446 status = MGMT_STATUS_SUCCESS;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003447
Johan Hedbergeedbd582014-11-15 09:34:23 +02003448done:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003449 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_REMOTE_OOB_DATA,
3450 status, &cp->addr, sizeof(cp->addr));
Szymon Janc2763eda2011-03-22 13:12:22 +01003451
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003452 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003453 return err;
3454}
3455
Johan Hedberge68f0722015-11-11 08:30:30 +02003456void mgmt_start_discovery_complete(struct hci_dev *hdev, u8 status)
Andre Guedes7c307722013-04-30 15:29:28 -03003457{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003458 struct mgmt_pending_cmd *cmd;
Lukasz Rymanowskiae55f592014-03-27 20:55:19 +01003459
Andre Guedes7c307722013-04-30 15:29:28 -03003460 BT_DBG("status %d", status);
3461
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003462 hci_dev_lock(hdev);
3463
Johan Hedberg333ae952015-03-17 13:48:47 +02003464 cmd = pending_find(MGMT_OP_START_DISCOVERY, hdev);
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003465 if (!cmd)
Johan Hedberg333ae952015-03-17 13:48:47 +02003466 cmd = pending_find(MGMT_OP_START_SERVICE_DISCOVERY, hdev);
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003467
Johan Hedberg78b781c2016-01-05 13:19:32 +02003468 if (!cmd)
3469 cmd = pending_find(MGMT_OP_START_LIMITED_DISCOVERY, hdev);
3470
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003471 if (cmd) {
Johan Hedberg2922a942014-12-05 13:36:06 +02003472 cmd->cmd_complete(cmd, mgmt_status(status));
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003473 mgmt_pending_remove(cmd);
Andre Guedes7c307722013-04-30 15:29:28 -03003474 }
3475
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003476 hci_dev_unlock(hdev);
Andre Guedes7c307722013-04-30 15:29:28 -03003477}
3478
Johan Hedberg591752a2015-11-11 08:11:24 +02003479static bool discovery_type_is_valid(struct hci_dev *hdev, uint8_t type,
3480 uint8_t *mgmt_status)
3481{
3482 switch (type) {
3483 case DISCOV_TYPE_LE:
3484 *mgmt_status = mgmt_le_support(hdev);
3485 if (*mgmt_status)
3486 return false;
3487 break;
3488 case DISCOV_TYPE_INTERLEAVED:
3489 *mgmt_status = mgmt_le_support(hdev);
3490 if (*mgmt_status)
3491 return false;
3492 /* Intentional fall-through */
3493 case DISCOV_TYPE_BREDR:
3494 *mgmt_status = mgmt_bredr_support(hdev);
3495 if (*mgmt_status)
3496 return false;
3497 break;
3498 default:
3499 *mgmt_status = MGMT_STATUS_INVALID_PARAMS;
3500 return false;
3501 }
3502
3503 return true;
3504}
3505
Johan Hedberg78b781c2016-01-05 13:19:32 +02003506static int start_discovery_internal(struct sock *sk, struct hci_dev *hdev,
3507 u16 op, void *data, u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003508{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003509 struct mgmt_cp_start_discovery *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003510 struct mgmt_pending_cmd *cmd;
Marcel Holtmann80190442014-12-04 11:36:36 +01003511 u8 status;
Johan Hedberg14a53662011-04-27 10:29:56 -04003512 int err;
3513
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003514 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003515
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003516 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003517
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003518 if (!hdev_is_powered(hdev)) {
Johan Hedberg78b781c2016-01-05 13:19:32 +02003519 err = mgmt_cmd_complete(sk, hdev->id, op,
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003520 MGMT_STATUS_NOT_POWERED,
3521 &cp->type, sizeof(cp->type));
Johan Hedbergbd2d1332011-11-07 23:13:37 +02003522 goto failed;
3523 }
3524
Marcel Holtmannf5a969f2014-12-04 11:36:34 +01003525 if (hdev->discovery.state != DISCOVERY_STOPPED ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07003526 hci_dev_test_flag(hdev, HCI_PERIODIC_INQ)) {
Johan Hedberg78b781c2016-01-05 13:19:32 +02003527 err = mgmt_cmd_complete(sk, hdev->id, op, MGMT_STATUS_BUSY,
3528 &cp->type, sizeof(cp->type));
Andre Guedes642be6c2012-03-21 00:03:37 -03003529 goto failed;
3530 }
3531
Johan Hedberg591752a2015-11-11 08:11:24 +02003532 if (!discovery_type_is_valid(hdev, cp->type, &status)) {
Johan Hedberg78b781c2016-01-05 13:19:32 +02003533 err = mgmt_cmd_complete(sk, hdev->id, op, status,
3534 &cp->type, sizeof(cp->type));
Johan Hedberg591752a2015-11-11 08:11:24 +02003535 goto failed;
3536 }
3537
Marcel Holtmann22078802014-12-05 11:45:22 +01003538 /* Clear the discovery filter first to free any previously
3539 * allocated memory for the UUID list.
3540 */
3541 hci_discovery_filter_clear(hdev);
3542
Andre Guedes4aab14e2012-02-17 20:39:36 -03003543 hdev->discovery.type = cp->type;
Marcel Holtmannda25cf62014-12-05 13:03:35 +01003544 hdev->discovery.report_invalid_rssi = false;
Johan Hedberg78b781c2016-01-05 13:19:32 +02003545 if (op == MGMT_OP_START_LIMITED_DISCOVERY)
3546 hdev->discovery.limited = true;
3547 else
3548 hdev->discovery.limited = false;
Andre Guedes4aab14e2012-02-17 20:39:36 -03003549
Johan Hedberg78b781c2016-01-05 13:19:32 +02003550 cmd = mgmt_pending_add(sk, op, hdev, data, len);
Johan Hedberge68f0722015-11-11 08:30:30 +02003551 if (!cmd) {
3552 err = -ENOMEM;
Johan Hedberg04106752013-01-10 14:54:09 +02003553 goto failed;
Andre Guedesf39799f2012-02-17 20:39:35 -03003554 }
Andre Guedes3fd24152012-02-03 17:48:01 -03003555
Johan Hedberge68f0722015-11-11 08:30:30 +02003556 cmd->cmd_complete = generic_cmd_complete;
Marcel Holtmannf5a969f2014-12-04 11:36:34 +01003557
3558 hci_discovery_set_state(hdev, DISCOVERY_STARTING);
Johan Hedberge68f0722015-11-11 08:30:30 +02003559 queue_work(hdev->req_workqueue, &hdev->discov_update);
3560 err = 0;
Johan Hedberg14a53662011-04-27 10:29:56 -04003561
3562failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003563 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003564 return err;
3565}
3566
Johan Hedberg78b781c2016-01-05 13:19:32 +02003567static int start_discovery(struct sock *sk, struct hci_dev *hdev,
3568 void *data, u16 len)
3569{
3570 return start_discovery_internal(sk, hdev, MGMT_OP_START_DISCOVERY,
3571 data, len);
3572}
3573
3574static int start_limited_discovery(struct sock *sk, struct hci_dev *hdev,
3575 void *data, u16 len)
3576{
3577 return start_discovery_internal(sk, hdev,
3578 MGMT_OP_START_LIMITED_DISCOVERY,
3579 data, len);
3580}
3581
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003582static int service_discovery_cmd_complete(struct mgmt_pending_cmd *cmd,
3583 u8 status)
Andre Guedes1183fdc2013-04-30 15:29:35 -03003584{
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003585 return mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, status,
3586 cmd->param, 1);
Johan Hedberg2922a942014-12-05 13:36:06 +02003587}
3588
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003589static int start_service_discovery(struct sock *sk, struct hci_dev *hdev,
3590 void *data, u16 len)
3591{
3592 struct mgmt_cp_start_service_discovery *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003593 struct mgmt_pending_cmd *cmd;
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003594 const u16 max_uuid_count = ((U16_MAX - sizeof(*cp)) / 16);
3595 u16 uuid_count, expected_len;
3596 u8 status;
Andre Guedes1183fdc2013-04-30 15:29:35 -03003597 int err;
3598
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003599 BT_DBG("%s", hdev->name);
Andre Guedes1183fdc2013-04-30 15:29:35 -03003600
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003601 hci_dev_lock(hdev);
Andre Guedes1183fdc2013-04-30 15:29:35 -03003602
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003603 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003604 err = mgmt_cmd_complete(sk, hdev->id,
3605 MGMT_OP_START_SERVICE_DISCOVERY,
3606 MGMT_STATUS_NOT_POWERED,
3607 &cp->type, sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003608 goto failed;
3609 }
3610
3611 if (hdev->discovery.state != DISCOVERY_STOPPED ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07003612 hci_dev_test_flag(hdev, HCI_PERIODIC_INQ)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003613 err = mgmt_cmd_complete(sk, hdev->id,
3614 MGMT_OP_START_SERVICE_DISCOVERY,
3615 MGMT_STATUS_BUSY, &cp->type,
3616 sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003617 goto failed;
3618 }
3619
3620 uuid_count = __le16_to_cpu(cp->uuid_count);
3621 if (uuid_count > max_uuid_count) {
3622 BT_ERR("service_discovery: too big uuid_count value %u",
3623 uuid_count);
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003624 err = mgmt_cmd_complete(sk, hdev->id,
3625 MGMT_OP_START_SERVICE_DISCOVERY,
3626 MGMT_STATUS_INVALID_PARAMS, &cp->type,
3627 sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003628 goto failed;
3629 }
3630
3631 expected_len = sizeof(*cp) + uuid_count * 16;
3632 if (expected_len != len) {
3633 BT_ERR("service_discovery: expected %u bytes, got %u bytes",
3634 expected_len, len);
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003635 err = mgmt_cmd_complete(sk, hdev->id,
3636 MGMT_OP_START_SERVICE_DISCOVERY,
3637 MGMT_STATUS_INVALID_PARAMS, &cp->type,
3638 sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003639 goto failed;
3640 }
3641
Johan Hedberg591752a2015-11-11 08:11:24 +02003642 if (!discovery_type_is_valid(hdev, cp->type, &status)) {
3643 err = mgmt_cmd_complete(sk, hdev->id,
3644 MGMT_OP_START_SERVICE_DISCOVERY,
3645 status, &cp->type, sizeof(cp->type));
3646 goto failed;
3647 }
3648
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003649 cmd = mgmt_pending_add(sk, MGMT_OP_START_SERVICE_DISCOVERY,
Johan Hedberg2922a942014-12-05 13:36:06 +02003650 hdev, data, len);
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003651 if (!cmd) {
3652 err = -ENOMEM;
3653 goto failed;
3654 }
3655
Johan Hedberg2922a942014-12-05 13:36:06 +02003656 cmd->cmd_complete = service_discovery_cmd_complete;
3657
Marcel Holtmann22078802014-12-05 11:45:22 +01003658 /* Clear the discovery filter first to free any previously
3659 * allocated memory for the UUID list.
3660 */
3661 hci_discovery_filter_clear(hdev);
3662
Jakub Pawlowski82f8b652015-03-04 16:24:26 -08003663 hdev->discovery.result_filtering = true;
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003664 hdev->discovery.type = cp->type;
3665 hdev->discovery.rssi = cp->rssi;
3666 hdev->discovery.uuid_count = uuid_count;
3667
3668 if (uuid_count > 0) {
3669 hdev->discovery.uuids = kmemdup(cp->uuids, uuid_count * 16,
3670 GFP_KERNEL);
3671 if (!hdev->discovery.uuids) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003672 err = mgmt_cmd_complete(sk, hdev->id,
3673 MGMT_OP_START_SERVICE_DISCOVERY,
3674 MGMT_STATUS_FAILED,
3675 &cp->type, sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003676 mgmt_pending_remove(cmd);
3677 goto failed;
3678 }
3679 }
3680
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003681 hci_discovery_set_state(hdev, DISCOVERY_STARTING);
Johan Hedberge68f0722015-11-11 08:30:30 +02003682 queue_work(hdev->req_workqueue, &hdev->discov_update);
3683 err = 0;
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003684
3685failed:
3686 hci_dev_unlock(hdev);
Andre Guedes1183fdc2013-04-30 15:29:35 -03003687 return err;
3688}
3689
Johan Hedberg2154d3f2015-11-11 08:30:45 +02003690void mgmt_stop_discovery_complete(struct hci_dev *hdev, u8 status)
Andre Guedes0e05bba2013-04-30 15:29:33 -03003691{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003692 struct mgmt_pending_cmd *cmd;
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003693
Andre Guedes0e05bba2013-04-30 15:29:33 -03003694 BT_DBG("status %d", status);
3695
3696 hci_dev_lock(hdev);
3697
Johan Hedberg333ae952015-03-17 13:48:47 +02003698 cmd = pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003699 if (cmd) {
Johan Hedberg2922a942014-12-05 13:36:06 +02003700 cmd->cmd_complete(cmd, mgmt_status(status));
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003701 mgmt_pending_remove(cmd);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003702 }
3703
Andre Guedes0e05bba2013-04-30 15:29:33 -03003704 hci_dev_unlock(hdev);
3705}
3706
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003707static int stop_discovery(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003708 u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003709{
Johan Hedbergd9306502012-02-20 23:25:18 +02003710 struct mgmt_cp_stop_discovery *mgmt_cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003711 struct mgmt_pending_cmd *cmd;
Johan Hedberg14a53662011-04-27 10:29:56 -04003712 int err;
3713
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003714 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003715
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003716 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003717
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003718 if (!hci_discovery_active(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003719 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
3720 MGMT_STATUS_REJECTED, &mgmt_cp->type,
3721 sizeof(mgmt_cp->type));
Johan Hedbergd9306502012-02-20 23:25:18 +02003722 goto unlock;
3723 }
3724
3725 if (hdev->discovery.type != mgmt_cp->type) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003726 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
3727 MGMT_STATUS_INVALID_PARAMS,
3728 &mgmt_cp->type, sizeof(mgmt_cp->type));
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003729 goto unlock;
Johan Hedbergff9ef572012-01-04 14:23:45 +02003730 }
3731
Johan Hedberg2922a942014-12-05 13:36:06 +02003732 cmd = mgmt_pending_add(sk, MGMT_OP_STOP_DISCOVERY, hdev, data, len);
Johan Hedberg14a53662011-04-27 10:29:56 -04003733 if (!cmd) {
3734 err = -ENOMEM;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003735 goto unlock;
Johan Hedberg14a53662011-04-27 10:29:56 -04003736 }
3737
Johan Hedberg2922a942014-12-05 13:36:06 +02003738 cmd->cmd_complete = generic_cmd_complete;
3739
Johan Hedberg2154d3f2015-11-11 08:30:45 +02003740 hci_discovery_set_state(hdev, DISCOVERY_STOPPING);
3741 queue_work(hdev->req_workqueue, &hdev->discov_update);
3742 err = 0;
Johan Hedberg14a53662011-04-27 10:29:56 -04003743
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003744unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003745 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003746 return err;
3747}
3748
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003749static int confirm_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003750 u16 len)
Johan Hedberg561aafb2012-01-04 13:31:59 +02003751{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003752 struct mgmt_cp_confirm_name *cp = data;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003753 struct inquiry_entry *e;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003754 int err;
3755
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003756 BT_DBG("%s", hdev->name);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003757
Johan Hedberg561aafb2012-01-04 13:31:59 +02003758 hci_dev_lock(hdev);
3759
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003760 if (!hci_discovery_active(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003761 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
3762 MGMT_STATUS_FAILED, &cp->addr,
3763 sizeof(cp->addr));
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003764 goto failed;
3765 }
3766
Johan Hedberga198e7b2012-02-17 14:27:06 +02003767 e = hci_inquiry_cache_lookup_unknown(hdev, &cp->addr.bdaddr);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003768 if (!e) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003769 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
3770 MGMT_STATUS_INVALID_PARAMS, &cp->addr,
3771 sizeof(cp->addr));
Johan Hedberg561aafb2012-01-04 13:31:59 +02003772 goto failed;
3773 }
3774
3775 if (cp->name_known) {
3776 e->name_state = NAME_KNOWN;
3777 list_del(&e->list);
3778 } else {
3779 e->name_state = NAME_NEEDED;
Johan Hedberga3d4e202012-01-09 00:53:02 +02003780 hci_inquiry_cache_update_resolve(hdev, e);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003781 }
3782
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003783 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME, 0,
3784 &cp->addr, sizeof(cp->addr));
Johan Hedberg561aafb2012-01-04 13:31:59 +02003785
3786failed:
3787 hci_dev_unlock(hdev);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003788 return err;
3789}
3790
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003791static int block_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003792 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003793{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003794 struct mgmt_cp_block_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003795 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003796 int err;
3797
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003798 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003799
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003800 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003801 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE,
3802 MGMT_STATUS_INVALID_PARAMS,
3803 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003804
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003805 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003806
Johan Hedbergdcc36c12014-07-09 12:59:13 +03003807 err = hci_bdaddr_list_add(&hdev->blacklist, &cp->addr.bdaddr,
3808 cp->addr.type);
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003809 if (err < 0) {
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003810 status = MGMT_STATUS_FAILED;
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003811 goto done;
3812 }
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003813
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003814 mgmt_event(MGMT_EV_DEVICE_BLOCKED, hdev, &cp->addr, sizeof(cp->addr),
3815 sk);
3816 status = MGMT_STATUS_SUCCESS;
3817
3818done:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003819 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE, status,
3820 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003821
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003822 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003823
3824 return err;
3825}
3826
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003827static int unblock_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003828 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003829{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003830 struct mgmt_cp_unblock_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003831 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003832 int err;
3833
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003834 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003835
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003836 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003837 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE,
3838 MGMT_STATUS_INVALID_PARAMS,
3839 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003840
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003841 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003842
Johan Hedbergdcc36c12014-07-09 12:59:13 +03003843 err = hci_bdaddr_list_del(&hdev->blacklist, &cp->addr.bdaddr,
3844 cp->addr.type);
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003845 if (err < 0) {
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003846 status = MGMT_STATUS_INVALID_PARAMS;
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003847 goto done;
3848 }
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003849
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003850 mgmt_event(MGMT_EV_DEVICE_UNBLOCKED, hdev, &cp->addr, sizeof(cp->addr),
3851 sk);
3852 status = MGMT_STATUS_SUCCESS;
3853
3854done:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003855 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE, status,
3856 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003857
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003858 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003859
3860 return err;
3861}
3862
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003863static int set_device_id(struct sock *sk, struct hci_dev *hdev, void *data,
3864 u16 len)
3865{
3866 struct mgmt_cp_set_device_id *cp = data;
Johan Hedberg890ea892013-03-15 17:06:52 -05003867 struct hci_request req;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003868 int err;
Szymon Jancc72d4b82012-03-16 16:02:57 +01003869 __u16 source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003870
3871 BT_DBG("%s", hdev->name);
3872
Szymon Jancc72d4b82012-03-16 16:02:57 +01003873 source = __le16_to_cpu(cp->source);
3874
3875 if (source > 0x0002)
Johan Hedberga69e8372015-03-06 21:08:53 +02003876 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DEVICE_ID,
3877 MGMT_STATUS_INVALID_PARAMS);
Szymon Jancc72d4b82012-03-16 16:02:57 +01003878
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003879 hci_dev_lock(hdev);
3880
Szymon Jancc72d4b82012-03-16 16:02:57 +01003881 hdev->devid_source = source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003882 hdev->devid_vendor = __le16_to_cpu(cp->vendor);
3883 hdev->devid_product = __le16_to_cpu(cp->product);
3884 hdev->devid_version = __le16_to_cpu(cp->version);
3885
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003886 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_DEVICE_ID, 0,
3887 NULL, 0);
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003888
Johan Hedberg890ea892013-03-15 17:06:52 -05003889 hci_req_init(&req, hdev);
Johan Hedbergb1a89172015-11-25 16:15:42 +02003890 __hci_req_update_eir(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05003891 hci_req_run(&req, NULL);
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003892
3893 hci_dev_unlock(hdev);
3894
3895 return err;
3896}
3897
Arman Uguray24b4f382015-03-23 15:57:12 -07003898static void enable_advertising_instance(struct hci_dev *hdev, u8 status,
3899 u16 opcode)
3900{
3901 BT_DBG("status %d", status);
3902}
3903
Marcel Holtmann1904a852015-01-11 13:50:44 -08003904static void set_advertising_complete(struct hci_dev *hdev, u8 status,
3905 u16 opcode)
Johan Hedberg4375f102013-09-25 13:26:10 +03003906{
3907 struct cmd_lookup match = { NULL, hdev };
Arman Uguray24b4f382015-03-23 15:57:12 -07003908 struct hci_request req;
Florian Grandel7816b822015-06-18 03:16:45 +02003909 u8 instance;
3910 struct adv_info *adv_instance;
3911 int err;
Johan Hedberg4375f102013-09-25 13:26:10 +03003912
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303913 hci_dev_lock(hdev);
3914
Johan Hedberg4375f102013-09-25 13:26:10 +03003915 if (status) {
3916 u8 mgmt_err = mgmt_status(status);
3917
3918 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev,
3919 cmd_status_rsp, &mgmt_err);
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303920 goto unlock;
Johan Hedberg4375f102013-09-25 13:26:10 +03003921 }
3922
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07003923 if (hci_dev_test_flag(hdev, HCI_LE_ADV))
Marcel Holtmanna1536da2015-03-13 02:11:01 -07003924 hci_dev_set_flag(hdev, HCI_ADVERTISING);
Johan Hedbergc93bd152014-07-08 15:07:48 +03003925 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07003926 hci_dev_clear_flag(hdev, HCI_ADVERTISING);
Johan Hedbergc93bd152014-07-08 15:07:48 +03003927
Johan Hedberg4375f102013-09-25 13:26:10 +03003928 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev, settings_rsp,
3929 &match);
3930
3931 new_settings(hdev, match.sk);
3932
3933 if (match.sk)
3934 sock_put(match.sk);
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303935
Arman Uguray24b4f382015-03-23 15:57:12 -07003936 /* If "Set Advertising" was just disabled and instance advertising was
Florian Grandel7816b822015-06-18 03:16:45 +02003937 * set up earlier, then re-enable multi-instance advertising.
Arman Uguray24b4f382015-03-23 15:57:12 -07003938 */
3939 if (hci_dev_test_flag(hdev, HCI_ADVERTISING) ||
Florian Grandel7816b822015-06-18 03:16:45 +02003940 list_empty(&hdev->adv_instances))
Arman Uguray24b4f382015-03-23 15:57:12 -07003941 goto unlock;
3942
Florian Grandel7816b822015-06-18 03:16:45 +02003943 instance = hdev->cur_adv_instance;
3944 if (!instance) {
3945 adv_instance = list_first_entry_or_null(&hdev->adv_instances,
3946 struct adv_info, list);
3947 if (!adv_instance)
3948 goto unlock;
3949
3950 instance = adv_instance->instance;
3951 }
3952
Arman Uguray24b4f382015-03-23 15:57:12 -07003953 hci_req_init(&req, hdev);
3954
Johan Hedbergf2252572015-11-18 12:49:20 +02003955 err = __hci_req_schedule_adv_instance(&req, instance, true);
Arman Uguray24b4f382015-03-23 15:57:12 -07003956
Florian Grandel7816b822015-06-18 03:16:45 +02003957 if (!err)
3958 err = hci_req_run(&req, enable_advertising_instance);
3959
3960 if (err)
Arman Uguray24b4f382015-03-23 15:57:12 -07003961 BT_ERR("Failed to re-configure advertising");
3962
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303963unlock:
3964 hci_dev_unlock(hdev);
Johan Hedberg4375f102013-09-25 13:26:10 +03003965}
3966
Marcel Holtmann21b51872013-10-10 09:47:53 -07003967static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
3968 u16 len)
Johan Hedberg4375f102013-09-25 13:26:10 +03003969{
3970 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003971 struct mgmt_pending_cmd *cmd;
Johan Hedberg4375f102013-09-25 13:26:10 +03003972 struct hci_request req;
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07003973 u8 val, status;
Johan Hedberg4375f102013-09-25 13:26:10 +03003974 int err;
3975
3976 BT_DBG("request for %s", hdev->name);
3977
Johan Hedberge6fe7982013-10-02 15:45:22 +03003978 status = mgmt_le_support(hdev);
3979 if (status)
Johan Hedberga69e8372015-03-06 21:08:53 +02003980 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3981 status);
Johan Hedberg4375f102013-09-25 13:26:10 +03003982
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07003983 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02003984 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3985 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg4375f102013-09-25 13:26:10 +03003986
3987 hci_dev_lock(hdev);
3988
3989 val = !!cp->val;
Johan Hedberg4375f102013-09-25 13:26:10 +03003990
Johan Hedbergf74ca9b2013-10-08 15:52:18 +02003991 /* The following conditions are ones which mean that we should
3992 * not do any HCI communication but directly send a mgmt
3993 * response to user space (after toggling the flag if
3994 * necessary).
3995 */
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07003996 if (!hdev_is_powered(hdev) ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07003997 (val == hci_dev_test_flag(hdev, HCI_ADVERTISING) &&
3998 (cp->val == 0x02) == hci_dev_test_flag(hdev, HCI_ADVERTISING_CONNECTABLE)) ||
Johan Hedberge8bb6b92014-07-08 15:07:53 +03003999 hci_conn_num(hdev, LE_LINK) > 0 ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004000 (hci_dev_test_flag(hdev, HCI_LE_SCAN) &&
Johan Hedberge8bb6b92014-07-08 15:07:53 +03004001 hdev->le_scan_type == LE_SCAN_ACTIVE)) {
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004002 bool changed;
Johan Hedberg4375f102013-09-25 13:26:10 +03004003
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004004 if (cp->val) {
Johan Hedbergcab054a2015-11-30 11:21:45 +02004005 hdev->cur_adv_instance = 0x00;
Marcel Holtmann238be782015-03-13 02:11:06 -07004006 changed = !hci_dev_test_and_set_flag(hdev, HCI_ADVERTISING);
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004007 if (cp->val == 0x02)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004008 hci_dev_set_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004009 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004010 hci_dev_clear_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004011 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004012 changed = hci_dev_test_and_clear_flag(hdev, HCI_ADVERTISING);
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004013 hci_dev_clear_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Johan Hedberg4375f102013-09-25 13:26:10 +03004014 }
4015
4016 err = send_settings_rsp(sk, MGMT_OP_SET_ADVERTISING, hdev);
4017 if (err < 0)
4018 goto unlock;
4019
4020 if (changed)
4021 err = new_settings(hdev, sk);
4022
4023 goto unlock;
4024 }
4025
Johan Hedberg333ae952015-03-17 13:48:47 +02004026 if (pending_find(MGMT_OP_SET_ADVERTISING, hdev) ||
4027 pending_find(MGMT_OP_SET_LE, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004028 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
4029 MGMT_STATUS_BUSY);
Johan Hedberg4375f102013-09-25 13:26:10 +03004030 goto unlock;
4031 }
4032
4033 cmd = mgmt_pending_add(sk, MGMT_OP_SET_ADVERTISING, hdev, data, len);
4034 if (!cmd) {
4035 err = -ENOMEM;
4036 goto unlock;
4037 }
4038
4039 hci_req_init(&req, hdev);
4040
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004041 if (cp->val == 0x02)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004042 hci_dev_set_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004043 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004044 hci_dev_clear_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb02015-03-12 22:30:58 -07004045
Florian Grandel7816b822015-06-18 03:16:45 +02004046 cancel_adv_timeout(hdev);
4047
Arman Uguray24b4f382015-03-23 15:57:12 -07004048 if (val) {
Florian Grandel7816b822015-06-18 03:16:45 +02004049 /* Switch to instance "0" for the Set Advertising setting.
4050 * We cannot use update_[adv|scan_rsp]_data() here as the
4051 * HCI_ADVERTISING flag is not yet set.
4052 */
Johan Hedbergcab054a2015-11-30 11:21:45 +02004053 hdev->cur_adv_instance = 0x00;
Johan Hedbergf2252572015-11-18 12:49:20 +02004054 __hci_req_update_adv_data(&req, 0x00);
4055 __hci_req_update_scan_rsp_data(&req, 0x00);
4056 __hci_req_enable_advertising(&req);
Arman Uguray24b4f382015-03-23 15:57:12 -07004057 } else {
Johan Hedbergf2252572015-11-18 12:49:20 +02004058 __hci_req_disable_advertising(&req);
Arman Uguray24b4f382015-03-23 15:57:12 -07004059 }
Johan Hedberg4375f102013-09-25 13:26:10 +03004060
4061 err = hci_req_run(&req, set_advertising_complete);
4062 if (err < 0)
4063 mgmt_pending_remove(cmd);
4064
4065unlock:
4066 hci_dev_unlock(hdev);
4067 return err;
4068}
4069
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004070static int set_static_address(struct sock *sk, struct hci_dev *hdev,
4071 void *data, u16 len)
4072{
4073 struct mgmt_cp_set_static_address *cp = data;
4074 int err;
4075
4076 BT_DBG("%s", hdev->name);
4077
Marcel Holtmann62af4442013-10-02 22:10:32 -07004078 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004079 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
4080 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004081
4082 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004083 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
4084 MGMT_STATUS_REJECTED);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004085
4086 if (bacmp(&cp->bdaddr, BDADDR_ANY)) {
4087 if (!bacmp(&cp->bdaddr, BDADDR_NONE))
Johan Hedberga69e8372015-03-06 21:08:53 +02004088 return mgmt_cmd_status(sk, hdev->id,
4089 MGMT_OP_SET_STATIC_ADDRESS,
4090 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004091
4092 /* Two most significant bits shall be set */
4093 if ((cp->bdaddr.b[5] & 0xc0) != 0xc0)
Johan Hedberga69e8372015-03-06 21:08:53 +02004094 return mgmt_cmd_status(sk, hdev->id,
4095 MGMT_OP_SET_STATIC_ADDRESS,
4096 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004097 }
4098
4099 hci_dev_lock(hdev);
4100
4101 bacpy(&hdev->static_addr, &cp->bdaddr);
4102
Marcel Holtmann93690c22015-03-06 10:11:21 -08004103 err = send_settings_rsp(sk, MGMT_OP_SET_STATIC_ADDRESS, hdev);
4104 if (err < 0)
4105 goto unlock;
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004106
Marcel Holtmann93690c22015-03-06 10:11:21 -08004107 err = new_settings(hdev, sk);
4108
4109unlock:
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004110 hci_dev_unlock(hdev);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004111 return err;
4112}
4113
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004114static int set_scan_params(struct sock *sk, struct hci_dev *hdev,
4115 void *data, u16 len)
4116{
4117 struct mgmt_cp_set_scan_params *cp = data;
4118 __u16 interval, window;
4119 int err;
4120
4121 BT_DBG("%s", hdev->name);
4122
4123 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004124 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
4125 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004126
4127 interval = __le16_to_cpu(cp->interval);
4128
4129 if (interval < 0x0004 || interval > 0x4000)
Johan Hedberga69e8372015-03-06 21:08:53 +02004130 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
4131 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004132
4133 window = __le16_to_cpu(cp->window);
4134
4135 if (window < 0x0004 || window > 0x4000)
Johan Hedberga69e8372015-03-06 21:08:53 +02004136 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
4137 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004138
Marcel Holtmann899e1072013-10-14 09:55:32 -07004139 if (window > interval)
Johan Hedberga69e8372015-03-06 21:08:53 +02004140 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
4141 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann899e1072013-10-14 09:55:32 -07004142
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004143 hci_dev_lock(hdev);
4144
4145 hdev->le_scan_interval = interval;
4146 hdev->le_scan_window = window;
4147
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004148 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS, 0,
4149 NULL, 0);
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004150
Andre Guedesdd2ef8e2014-02-26 20:21:56 -03004151 /* If background scan is running, restart it so new parameters are
4152 * loaded.
4153 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004154 if (hci_dev_test_flag(hdev, HCI_LE_SCAN) &&
Andre Guedesdd2ef8e2014-02-26 20:21:56 -03004155 hdev->discovery.state == DISCOVERY_STOPPED) {
4156 struct hci_request req;
4157
4158 hci_req_init(&req, hdev);
4159
4160 hci_req_add_le_scan_disable(&req);
4161 hci_req_add_le_passive_scan(&req);
4162
4163 hci_req_run(&req, NULL);
4164 }
4165
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004166 hci_dev_unlock(hdev);
4167
4168 return err;
4169}
4170
Marcel Holtmann1904a852015-01-11 13:50:44 -08004171static void fast_connectable_complete(struct hci_dev *hdev, u8 status,
4172 u16 opcode)
Johan Hedberg33e38b32013-03-15 17:07:05 -05004173{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004174 struct mgmt_pending_cmd *cmd;
Johan Hedberg33e38b32013-03-15 17:07:05 -05004175
4176 BT_DBG("status 0x%02x", status);
4177
4178 hci_dev_lock(hdev);
4179
Johan Hedberg333ae952015-03-17 13:48:47 +02004180 cmd = pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004181 if (!cmd)
4182 goto unlock;
4183
4184 if (status) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004185 mgmt_cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4186 mgmt_status(status));
Johan Hedberg33e38b32013-03-15 17:07:05 -05004187 } else {
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004188 struct mgmt_mode *cp = cmd->param;
4189
4190 if (cp->val)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004191 hci_dev_set_flag(hdev, HCI_FAST_CONNECTABLE);
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004192 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004193 hci_dev_clear_flag(hdev, HCI_FAST_CONNECTABLE);
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004194
Johan Hedberg33e38b32013-03-15 17:07:05 -05004195 send_settings_rsp(cmd->sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev);
4196 new_settings(hdev, cmd->sk);
4197 }
4198
4199 mgmt_pending_remove(cmd);
4200
4201unlock:
4202 hci_dev_unlock(hdev);
4203}
4204
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004205static int set_fast_connectable(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004206 void *data, u16 len)
Antti Julkuf6422ec2011-06-22 13:11:56 +03004207{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004208 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004209 struct mgmt_pending_cmd *cmd;
Johan Hedberg33e38b32013-03-15 17:07:05 -05004210 struct hci_request req;
Antti Julkuf6422ec2011-06-22 13:11:56 +03004211 int err;
4212
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004213 BT_DBG("%s", hdev->name);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004214
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004215 if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) ||
Johan Hedberg56f87902013-10-02 13:43:13 +03004216 hdev->hci_ver < BLUETOOTH_VER_1_2)
Johan Hedberga69e8372015-03-06 21:08:53 +02004217 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4218 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03004219
Johan Hedberga7e80f22013-01-09 16:05:19 +02004220 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02004221 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4222 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02004223
Antti Julkuf6422ec2011-06-22 13:11:56 +03004224 hci_dev_lock(hdev);
4225
Johan Hedberg333ae952015-03-17 13:48:47 +02004226 if (pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004227 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4228 MGMT_STATUS_BUSY);
Johan Hedberg05cbf292013-03-15 17:07:07 -05004229 goto unlock;
4230 }
4231
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004232 if (!!cp->val == hci_dev_test_flag(hdev, HCI_FAST_CONNECTABLE)) {
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004233 err = send_settings_rsp(sk, MGMT_OP_SET_FAST_CONNECTABLE,
4234 hdev);
4235 goto unlock;
4236 }
4237
Johan Hedberg406ef2a2015-03-10 20:14:27 +02004238 if (!hdev_is_powered(hdev)) {
Marcel Holtmannce05d602015-03-13 02:11:03 -07004239 hci_dev_change_flag(hdev, HCI_FAST_CONNECTABLE);
Johan Hedberg406ef2a2015-03-10 20:14:27 +02004240 err = send_settings_rsp(sk, MGMT_OP_SET_FAST_CONNECTABLE,
4241 hdev);
4242 new_settings(hdev, sk);
4243 goto unlock;
4244 }
4245
Johan Hedberg33e38b32013-03-15 17:07:05 -05004246 cmd = mgmt_pending_add(sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev,
4247 data, len);
4248 if (!cmd) {
4249 err = -ENOMEM;
4250 goto unlock;
4251 }
4252
4253 hci_req_init(&req, hdev);
4254
Johan Hedbergbf943cb2015-11-25 16:15:43 +02004255 __hci_req_write_fast_connectable(&req, cp->val);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004256
4257 err = hci_req_run(&req, fast_connectable_complete);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004258 if (err < 0) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004259 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4260 MGMT_STATUS_FAILED);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004261 mgmt_pending_remove(cmd);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004262 }
4263
Johan Hedberg33e38b32013-03-15 17:07:05 -05004264unlock:
Antti Julkuf6422ec2011-06-22 13:11:56 +03004265 hci_dev_unlock(hdev);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004266
Antti Julkuf6422ec2011-06-22 13:11:56 +03004267 return err;
4268}
4269
Marcel Holtmann1904a852015-01-11 13:50:44 -08004270static void set_bredr_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg0663ca22013-10-02 13:43:14 +03004271{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004272 struct mgmt_pending_cmd *cmd;
Johan Hedberg0663ca22013-10-02 13:43:14 +03004273
4274 BT_DBG("status 0x%02x", status);
4275
4276 hci_dev_lock(hdev);
4277
Johan Hedberg333ae952015-03-17 13:48:47 +02004278 cmd = pending_find(MGMT_OP_SET_BREDR, hdev);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004279 if (!cmd)
4280 goto unlock;
4281
4282 if (status) {
4283 u8 mgmt_err = mgmt_status(status);
4284
4285 /* We need to restore the flag if related HCI commands
4286 * failed.
4287 */
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004288 hci_dev_clear_flag(hdev, HCI_BREDR_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004289
Johan Hedberga69e8372015-03-06 21:08:53 +02004290 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004291 } else {
4292 send_settings_rsp(cmd->sk, MGMT_OP_SET_BREDR, hdev);
4293 new_settings(hdev, cmd->sk);
4294 }
4295
4296 mgmt_pending_remove(cmd);
4297
4298unlock:
4299 hci_dev_unlock(hdev);
4300}
4301
4302static int set_bredr(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
4303{
4304 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004305 struct mgmt_pending_cmd *cmd;
Johan Hedberg0663ca22013-10-02 13:43:14 +03004306 struct hci_request req;
4307 int err;
4308
4309 BT_DBG("request for %s", hdev->name);
4310
4311 if (!lmp_bredr_capable(hdev) || !lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004312 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4313 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004314
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004315 if (!hci_dev_test_flag(hdev, HCI_LE_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02004316 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4317 MGMT_STATUS_REJECTED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004318
4319 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02004320 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4321 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004322
4323 hci_dev_lock(hdev);
4324
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004325 if (cp->val == hci_dev_test_flag(hdev, HCI_BREDR_ENABLED)) {
Johan Hedberg0663ca22013-10-02 13:43:14 +03004326 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4327 goto unlock;
4328 }
4329
4330 if (!hdev_is_powered(hdev)) {
4331 if (!cp->val) {
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004332 hci_dev_clear_flag(hdev, HCI_DISCOVERABLE);
4333 hci_dev_clear_flag(hdev, HCI_SSP_ENABLED);
4334 hci_dev_clear_flag(hdev, HCI_LINK_SECURITY);
4335 hci_dev_clear_flag(hdev, HCI_FAST_CONNECTABLE);
4336 hci_dev_clear_flag(hdev, HCI_HS_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004337 }
4338
Marcel Holtmannce05d602015-03-13 02:11:03 -07004339 hci_dev_change_flag(hdev, HCI_BREDR_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004340
4341 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4342 if (err < 0)
4343 goto unlock;
4344
4345 err = new_settings(hdev, sk);
4346 goto unlock;
4347 }
4348
4349 /* Reject disabling when powered on */
4350 if (!cp->val) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004351 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4352 MGMT_STATUS_REJECTED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004353 goto unlock;
Marcel Holtmann111e4bc2015-01-14 14:40:42 -08004354 } else {
4355 /* When configuring a dual-mode controller to operate
4356 * with LE only and using a static address, then switching
4357 * BR/EDR back on is not allowed.
4358 *
4359 * Dual-mode controllers shall operate with the public
4360 * address as its identity address for BR/EDR and LE. So
4361 * reject the attempt to create an invalid configuration.
Marcel Holtmann3a5486e2015-01-22 11:15:21 -08004362 *
4363 * The same restrictions applies when secure connections
4364 * has been enabled. For BR/EDR this is a controller feature
4365 * while for LE it is a host stack feature. This means that
4366 * switching BR/EDR back on when secure connections has been
4367 * enabled is not a supported transaction.
Marcel Holtmann111e4bc2015-01-14 14:40:42 -08004368 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004369 if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) &&
Marcel Holtmann3a5486e2015-01-22 11:15:21 -08004370 (bacmp(&hdev->static_addr, BDADDR_ANY) ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004371 hci_dev_test_flag(hdev, HCI_SC_ENABLED))) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004372 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4373 MGMT_STATUS_REJECTED);
Marcel Holtmann111e4bc2015-01-14 14:40:42 -08004374 goto unlock;
4375 }
Johan Hedberg0663ca22013-10-02 13:43:14 +03004376 }
4377
Johan Hedberg333ae952015-03-17 13:48:47 +02004378 if (pending_find(MGMT_OP_SET_BREDR, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004379 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4380 MGMT_STATUS_BUSY);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004381 goto unlock;
4382 }
4383
4384 cmd = mgmt_pending_add(sk, MGMT_OP_SET_BREDR, hdev, data, len);
4385 if (!cmd) {
4386 err = -ENOMEM;
4387 goto unlock;
4388 }
4389
Johan Hedbergf2252572015-11-18 12:49:20 +02004390 /* We need to flip the bit already here so that
4391 * hci_req_update_adv_data generates the correct flags.
Johan Hedberg0663ca22013-10-02 13:43:14 +03004392 */
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004393 hci_dev_set_flag(hdev, HCI_BREDR_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004394
4395 hci_req_init(&req, hdev);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004396
Johan Hedbergbf943cb2015-11-25 16:15:43 +02004397 __hci_req_write_fast_connectable(&req, false);
Johan Hedberg01b1cb82015-11-16 12:52:21 +02004398 __hci_req_update_scan(&req);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004399
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004400 /* Since only the advertising data flags will change, there
4401 * is no need to update the scan response data.
4402 */
Johan Hedbergcab054a2015-11-30 11:21:45 +02004403 __hci_req_update_adv_data(&req, hdev->cur_adv_instance);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004404
Johan Hedberg0663ca22013-10-02 13:43:14 +03004405 err = hci_req_run(&req, set_bredr_complete);
4406 if (err < 0)
4407 mgmt_pending_remove(cmd);
4408
4409unlock:
4410 hci_dev_unlock(hdev);
4411 return err;
4412}
4413
Johan Hedberga1443f52015-01-23 15:42:46 +02004414static void sc_enable_complete(struct hci_dev *hdev, u8 status, u16 opcode)
4415{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004416 struct mgmt_pending_cmd *cmd;
Johan Hedberga1443f52015-01-23 15:42:46 +02004417 struct mgmt_mode *cp;
4418
4419 BT_DBG("%s status %u", hdev->name, status);
4420
4421 hci_dev_lock(hdev);
4422
Johan Hedberg333ae952015-03-17 13:48:47 +02004423 cmd = pending_find(MGMT_OP_SET_SECURE_CONN, hdev);
Johan Hedberga1443f52015-01-23 15:42:46 +02004424 if (!cmd)
4425 goto unlock;
4426
4427 if (status) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004428 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode,
4429 mgmt_status(status));
Johan Hedberga1443f52015-01-23 15:42:46 +02004430 goto remove;
4431 }
4432
4433 cp = cmd->param;
4434
4435 switch (cp->val) {
4436 case 0x00:
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004437 hci_dev_clear_flag(hdev, HCI_SC_ENABLED);
4438 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Johan Hedberga1443f52015-01-23 15:42:46 +02004439 break;
4440 case 0x01:
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004441 hci_dev_set_flag(hdev, HCI_SC_ENABLED);
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004442 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Johan Hedberga1443f52015-01-23 15:42:46 +02004443 break;
4444 case 0x02:
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004445 hci_dev_set_flag(hdev, HCI_SC_ENABLED);
4446 hci_dev_set_flag(hdev, HCI_SC_ONLY);
Johan Hedberga1443f52015-01-23 15:42:46 +02004447 break;
4448 }
4449
4450 send_settings_rsp(cmd->sk, MGMT_OP_SET_SECURE_CONN, hdev);
4451 new_settings(hdev, cmd->sk);
4452
4453remove:
4454 mgmt_pending_remove(cmd);
4455unlock:
4456 hci_dev_unlock(hdev);
4457}
4458
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004459static int set_secure_conn(struct sock *sk, struct hci_dev *hdev,
4460 void *data, u16 len)
4461{
4462 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004463 struct mgmt_pending_cmd *cmd;
Johan Hedberga1443f52015-01-23 15:42:46 +02004464 struct hci_request req;
Johan Hedberga3209692014-05-26 11:23:35 +03004465 u8 val;
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004466 int err;
4467
4468 BT_DBG("request for %s", hdev->name);
4469
Marcel Holtmann05b3c3e2014-12-31 14:43:18 -08004470 if (!lmp_sc_capable(hdev) &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004471 !hci_dev_test_flag(hdev, HCI_LE_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02004472 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4473 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004474
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004475 if (hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) &&
Johan Hedberg59200282015-01-28 19:56:00 +02004476 lmp_sc_capable(hdev) &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004477 !hci_dev_test_flag(hdev, HCI_SSP_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02004478 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4479 MGMT_STATUS_REJECTED);
Marcel Holtmanned93ec62015-01-22 11:15:22 -08004480
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004481 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02004482 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004483 MGMT_STATUS_INVALID_PARAMS);
4484
4485 hci_dev_lock(hdev);
4486
Marcel Holtmann05b3c3e2014-12-31 14:43:18 -08004487 if (!hdev_is_powered(hdev) || !lmp_sc_capable(hdev) ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004488 !hci_dev_test_flag(hdev, HCI_BREDR_ENABLED)) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004489 bool changed;
4490
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004491 if (cp->val) {
Marcel Holtmann238be782015-03-13 02:11:06 -07004492 changed = !hci_dev_test_and_set_flag(hdev,
4493 HCI_SC_ENABLED);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004494 if (cp->val == 0x02)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004495 hci_dev_set_flag(hdev, HCI_SC_ONLY);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004496 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004497 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004498 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004499 changed = hci_dev_test_and_clear_flag(hdev,
4500 HCI_SC_ENABLED);
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004501 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004502 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004503
4504 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4505 if (err < 0)
4506 goto failed;
4507
4508 if (changed)
4509 err = new_settings(hdev, sk);
4510
4511 goto failed;
4512 }
4513
Johan Hedberg333ae952015-03-17 13:48:47 +02004514 if (pending_find(MGMT_OP_SET_SECURE_CONN, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004515 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4516 MGMT_STATUS_BUSY);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004517 goto failed;
4518 }
4519
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004520 val = !!cp->val;
4521
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004522 if (val == hci_dev_test_flag(hdev, HCI_SC_ENABLED) &&
4523 (cp->val == 0x02) == hci_dev_test_flag(hdev, HCI_SC_ONLY)) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004524 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4525 goto failed;
4526 }
4527
4528 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SECURE_CONN, hdev, data, len);
4529 if (!cmd) {
4530 err = -ENOMEM;
4531 goto failed;
4532 }
4533
Johan Hedberga1443f52015-01-23 15:42:46 +02004534 hci_req_init(&req, hdev);
4535 hci_req_add(&req, HCI_OP_WRITE_SC_SUPPORT, 1, &val);
4536 err = hci_req_run(&req, sc_enable_complete);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004537 if (err < 0) {
4538 mgmt_pending_remove(cmd);
4539 goto failed;
4540 }
4541
4542failed:
4543 hci_dev_unlock(hdev);
4544 return err;
4545}
4546
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004547static int set_debug_keys(struct sock *sk, struct hci_dev *hdev,
4548 void *data, u16 len)
4549{
4550 struct mgmt_mode *cp = data;
Johan Hedbergb97109792014-06-24 14:00:28 +03004551 bool changed, use_changed;
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004552 int err;
4553
4554 BT_DBG("request for %s", hdev->name);
4555
Johan Hedbergb97109792014-06-24 14:00:28 +03004556 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02004557 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DEBUG_KEYS,
4558 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004559
4560 hci_dev_lock(hdev);
4561
4562 if (cp->val)
Marcel Holtmann238be782015-03-13 02:11:06 -07004563 changed = !hci_dev_test_and_set_flag(hdev, HCI_KEEP_DEBUG_KEYS);
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004564 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004565 changed = hci_dev_test_and_clear_flag(hdev,
4566 HCI_KEEP_DEBUG_KEYS);
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004567
Johan Hedbergb97109792014-06-24 14:00:28 +03004568 if (cp->val == 0x02)
Marcel Holtmann238be782015-03-13 02:11:06 -07004569 use_changed = !hci_dev_test_and_set_flag(hdev,
4570 HCI_USE_DEBUG_KEYS);
Johan Hedbergb97109792014-06-24 14:00:28 +03004571 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004572 use_changed = hci_dev_test_and_clear_flag(hdev,
4573 HCI_USE_DEBUG_KEYS);
Johan Hedbergb97109792014-06-24 14:00:28 +03004574
4575 if (hdev_is_powered(hdev) && use_changed &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004576 hci_dev_test_flag(hdev, HCI_SSP_ENABLED)) {
Johan Hedbergb97109792014-06-24 14:00:28 +03004577 u8 mode = (cp->val == 0x02) ? 0x01 : 0x00;
4578 hci_send_cmd(hdev, HCI_OP_WRITE_SSP_DEBUG_MODE,
4579 sizeof(mode), &mode);
4580 }
4581
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004582 err = send_settings_rsp(sk, MGMT_OP_SET_DEBUG_KEYS, hdev);
4583 if (err < 0)
4584 goto unlock;
4585
4586 if (changed)
4587 err = new_settings(hdev, sk);
4588
4589unlock:
4590 hci_dev_unlock(hdev);
4591 return err;
4592}
4593
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004594static int set_privacy(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4595 u16 len)
4596{
4597 struct mgmt_cp_set_privacy *cp = cp_data;
4598 bool changed;
4599 int err;
4600
4601 BT_DBG("request for %s", hdev->name);
4602
4603 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004604 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4605 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004606
Johan Hedberg82a37ad2016-03-09 17:30:34 +02004607 if (cp->privacy != 0x00 && cp->privacy != 0x01 && cp->privacy != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02004608 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4609 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004610
4611 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004612 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4613 MGMT_STATUS_REJECTED);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004614
4615 hci_dev_lock(hdev);
4616
Johan Hedbergc21c0ea2014-02-24 11:10:30 +02004617 /* If user space supports this command it is also expected to
4618 * handle IRKs. Therefore, set the HCI_RPA_RESOLVING flag.
4619 */
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004620 hci_dev_set_flag(hdev, HCI_RPA_RESOLVING);
Johan Hedbergc21c0ea2014-02-24 11:10:30 +02004621
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004622 if (cp->privacy) {
Marcel Holtmann238be782015-03-13 02:11:06 -07004623 changed = !hci_dev_test_and_set_flag(hdev, HCI_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004624 memcpy(hdev->irk, cp->irk, sizeof(hdev->irk));
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004625 hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
Johan Hedberg82a37ad2016-03-09 17:30:34 +02004626 if (cp->privacy == 0x02)
4627 hci_dev_set_flag(hdev, HCI_LIMITED_PRIVACY);
4628 else
4629 hci_dev_clear_flag(hdev, HCI_LIMITED_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004630 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004631 changed = hci_dev_test_and_clear_flag(hdev, HCI_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004632 memset(hdev->irk, 0, sizeof(hdev->irk));
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004633 hci_dev_clear_flag(hdev, HCI_RPA_EXPIRED);
Johan Hedberg82a37ad2016-03-09 17:30:34 +02004634 hci_dev_clear_flag(hdev, HCI_LIMITED_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004635 }
4636
4637 err = send_settings_rsp(sk, MGMT_OP_SET_PRIVACY, hdev);
4638 if (err < 0)
4639 goto unlock;
4640
4641 if (changed)
4642 err = new_settings(hdev, sk);
4643
4644unlock:
4645 hci_dev_unlock(hdev);
4646 return err;
4647}
4648
Johan Hedberg41edf162014-02-18 10:19:35 +02004649static bool irk_is_valid(struct mgmt_irk_info *irk)
4650{
4651 switch (irk->addr.type) {
4652 case BDADDR_LE_PUBLIC:
4653 return true;
4654
4655 case BDADDR_LE_RANDOM:
4656 /* Two most significant bits shall be set */
4657 if ((irk->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4658 return false;
4659 return true;
4660 }
4661
4662 return false;
4663}
4664
4665static int load_irks(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4666 u16 len)
4667{
4668 struct mgmt_cp_load_irks *cp = cp_data;
Johan Hedbergba1d6932014-07-03 13:52:27 +03004669 const u16 max_irk_count = ((U16_MAX - sizeof(*cp)) /
4670 sizeof(struct mgmt_irk_info));
Johan Hedberg41edf162014-02-18 10:19:35 +02004671 u16 irk_count, expected_len;
4672 int i, err;
4673
4674 BT_DBG("request for %s", hdev->name);
4675
4676 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004677 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4678 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg41edf162014-02-18 10:19:35 +02004679
4680 irk_count = __le16_to_cpu(cp->irk_count);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004681 if (irk_count > max_irk_count) {
4682 BT_ERR("load_irks: too big irk_count value %u", irk_count);
Johan Hedberga69e8372015-03-06 21:08:53 +02004683 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4684 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004685 }
Johan Hedberg41edf162014-02-18 10:19:35 +02004686
4687 expected_len = sizeof(*cp) + irk_count * sizeof(struct mgmt_irk_info);
4688 if (expected_len != len) {
4689 BT_ERR("load_irks: expected %u bytes, got %u bytes",
Johan Hedberg2606ecb2014-03-07 15:04:13 +02004690 expected_len, len);
Johan Hedberga69e8372015-03-06 21:08:53 +02004691 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4692 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg41edf162014-02-18 10:19:35 +02004693 }
4694
4695 BT_DBG("%s irk_count %u", hdev->name, irk_count);
4696
4697 for (i = 0; i < irk_count; i++) {
4698 struct mgmt_irk_info *key = &cp->irks[i];
4699
4700 if (!irk_is_valid(key))
Johan Hedberga69e8372015-03-06 21:08:53 +02004701 return mgmt_cmd_status(sk, hdev->id,
4702 MGMT_OP_LOAD_IRKS,
4703 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg41edf162014-02-18 10:19:35 +02004704 }
4705
4706 hci_dev_lock(hdev);
4707
4708 hci_smp_irks_clear(hdev);
4709
4710 for (i = 0; i < irk_count; i++) {
4711 struct mgmt_irk_info *irk = &cp->irks[i];
Johan Hedberg41edf162014-02-18 10:19:35 +02004712
Johan Hedberg85813a72015-10-21 18:02:59 +03004713 hci_add_irk(hdev, &irk->addr.bdaddr,
4714 le_addr_type(irk->addr.type), irk->val,
Johan Hedberg41edf162014-02-18 10:19:35 +02004715 BDADDR_ANY);
4716 }
4717
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004718 hci_dev_set_flag(hdev, HCI_RPA_RESOLVING);
Johan Hedberg41edf162014-02-18 10:19:35 +02004719
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004720 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_LOAD_IRKS, 0, NULL, 0);
Johan Hedberg41edf162014-02-18 10:19:35 +02004721
4722 hci_dev_unlock(hdev);
4723
4724 return err;
4725}
4726
Johan Hedberg3f706b72013-01-20 14:27:16 +02004727static bool ltk_is_valid(struct mgmt_ltk_info *key)
4728{
4729 if (key->master != 0x00 && key->master != 0x01)
4730 return false;
Marcel Holtmann490cb0b2014-02-16 12:59:05 -08004731
4732 switch (key->addr.type) {
4733 case BDADDR_LE_PUBLIC:
4734 return true;
4735
4736 case BDADDR_LE_RANDOM:
4737 /* Two most significant bits shall be set */
4738 if ((key->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4739 return false;
4740 return true;
4741 }
4742
4743 return false;
Johan Hedberg3f706b72013-01-20 14:27:16 +02004744}
4745
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004746static int load_long_term_keys(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004747 void *cp_data, u16 len)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004748{
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004749 struct mgmt_cp_load_long_term_keys *cp = cp_data;
Johan Hedbergba1d6932014-07-03 13:52:27 +03004750 const u16 max_key_count = ((U16_MAX - sizeof(*cp)) /
4751 sizeof(struct mgmt_ltk_info));
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004752 u16 key_count, expected_len;
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004753 int i, err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004754
Marcel Holtmanncf99ba12013-10-02 21:16:08 -07004755 BT_DBG("request for %s", hdev->name);
4756
4757 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004758 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4759 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmanncf99ba12013-10-02 21:16:08 -07004760
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004761 key_count = __le16_to_cpu(cp->key_count);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004762 if (key_count > max_key_count) {
4763 BT_ERR("load_ltks: too big key_count value %u", key_count);
Johan Hedberga69e8372015-03-06 21:08:53 +02004764 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4765 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004766 }
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004767
4768 expected_len = sizeof(*cp) + key_count *
4769 sizeof(struct mgmt_ltk_info);
4770 if (expected_len != len) {
4771 BT_ERR("load_keys: expected %u bytes, got %u bytes",
Johan Hedberg2606ecb2014-03-07 15:04:13 +02004772 expected_len, len);
Johan Hedberga69e8372015-03-06 21:08:53 +02004773 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4774 MGMT_STATUS_INVALID_PARAMS);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004775 }
4776
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004777 BT_DBG("%s key_count %u", hdev->name, key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004778
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004779 for (i = 0; i < key_count; i++) {
4780 struct mgmt_ltk_info *key = &cp->keys[i];
4781
Johan Hedberg3f706b72013-01-20 14:27:16 +02004782 if (!ltk_is_valid(key))
Johan Hedberga69e8372015-03-06 21:08:53 +02004783 return mgmt_cmd_status(sk, hdev->id,
4784 MGMT_OP_LOAD_LONG_TERM_KEYS,
4785 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004786 }
4787
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004788 hci_dev_lock(hdev);
4789
4790 hci_smp_ltks_clear(hdev);
4791
4792 for (i = 0; i < key_count; i++) {
4793 struct mgmt_ltk_info *key = &cp->keys[i];
Johan Hedberg85813a72015-10-21 18:02:59 +03004794 u8 type, authenticated;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004795
Johan Hedberg61b43352014-05-29 19:36:53 +03004796 switch (key->type) {
4797 case MGMT_LTK_UNAUTHENTICATED:
Johan Hedbergd7b25452014-05-23 13:19:53 +03004798 authenticated = 0x00;
Johan Hedberg23fb8de2014-05-23 13:15:37 +03004799 type = key->master ? SMP_LTK : SMP_LTK_SLAVE;
Johan Hedberg61b43352014-05-29 19:36:53 +03004800 break;
4801 case MGMT_LTK_AUTHENTICATED:
Johan Hedbergd7b25452014-05-23 13:19:53 +03004802 authenticated = 0x01;
Johan Hedberg23fb8de2014-05-23 13:15:37 +03004803 type = key->master ? SMP_LTK : SMP_LTK_SLAVE;
Johan Hedberg61b43352014-05-29 19:36:53 +03004804 break;
Johan Hedberg23fb8de2014-05-23 13:15:37 +03004805 case MGMT_LTK_P256_UNAUTH:
4806 authenticated = 0x00;
4807 type = SMP_LTK_P256;
4808 break;
4809 case MGMT_LTK_P256_AUTH:
4810 authenticated = 0x01;
4811 type = SMP_LTK_P256;
4812 break;
4813 case MGMT_LTK_P256_DEBUG:
4814 authenticated = 0x00;
4815 type = SMP_LTK_P256_DEBUG;
Johan Hedberg61b43352014-05-29 19:36:53 +03004816 default:
4817 continue;
4818 }
Johan Hedbergd7b25452014-05-23 13:19:53 +03004819
Johan Hedberg85813a72015-10-21 18:02:59 +03004820 hci_add_ltk(hdev, &key->addr.bdaddr,
4821 le_addr_type(key->addr.type), type, authenticated,
4822 key->val, key->enc_size, key->ediv, key->rand);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004823 }
4824
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004825 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS, 0,
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004826 NULL, 0);
4827
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004828 hci_dev_unlock(hdev);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004829
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004830 return err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004831}
4832
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004833static int conn_info_cmd_complete(struct mgmt_pending_cmd *cmd, u8 status)
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004834{
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004835 struct hci_conn *conn = cmd->user_data;
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004836 struct mgmt_rp_get_conn_info rp;
Johan Hedberg9df74652014-12-19 22:26:03 +02004837 int err;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004838
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004839 memcpy(&rp.addr, cmd->param, sizeof(rp.addr));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004840
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004841 if (status == MGMT_STATUS_SUCCESS) {
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004842 rp.rssi = conn->rssi;
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004843 rp.tx_power = conn->tx_power;
4844 rp.max_tx_power = conn->max_tx_power;
4845 } else {
4846 rp.rssi = HCI_RSSI_INVALID;
4847 rp.tx_power = HCI_TX_POWER_INVALID;
4848 rp.max_tx_power = HCI_TX_POWER_INVALID;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004849 }
4850
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004851 err = mgmt_cmd_complete(cmd->sk, cmd->index, MGMT_OP_GET_CONN_INFO,
4852 status, &rp, sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004853
4854 hci_conn_drop(conn);
Johan Hedbergf8aaf9b2014-08-17 23:28:57 +03004855 hci_conn_put(conn);
Johan Hedberg9df74652014-12-19 22:26:03 +02004856
4857 return err;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004858}
4859
Marcel Holtmann1904a852015-01-11 13:50:44 -08004860static void conn_info_refresh_complete(struct hci_dev *hdev, u8 hci_status,
4861 u16 opcode)
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004862{
4863 struct hci_cp_read_rssi *cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004864 struct mgmt_pending_cmd *cmd;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004865 struct hci_conn *conn;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004866 u16 handle;
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004867 u8 status;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004868
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004869 BT_DBG("status 0x%02x", hci_status);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004870
4871 hci_dev_lock(hdev);
4872
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004873 /* Commands sent in request are either Read RSSI or Read Transmit Power
4874 * Level so we check which one was last sent to retrieve connection
4875 * handle. Both commands have handle as first parameter so it's safe to
4876 * cast data on the same command struct.
4877 *
4878 * First command sent is always Read RSSI and we fail only if it fails.
4879 * In other case we simply override error to indicate success as we
4880 * already remembered if TX power value is actually valid.
4881 */
4882 cp = hci_sent_cmd_data(hdev, HCI_OP_READ_RSSI);
4883 if (!cp) {
4884 cp = hci_sent_cmd_data(hdev, HCI_OP_READ_TX_POWER);
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004885 status = MGMT_STATUS_SUCCESS;
4886 } else {
4887 status = mgmt_status(hci_status);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004888 }
4889
4890 if (!cp) {
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004891 BT_ERR("invalid sent_cmd in conn_info response");
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004892 goto unlock;
4893 }
4894
4895 handle = __le16_to_cpu(cp->handle);
4896 conn = hci_conn_hash_lookup_handle(hdev, handle);
4897 if (!conn) {
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004898 BT_ERR("unknown handle (%d) in conn_info response", handle);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004899 goto unlock;
4900 }
4901
Johan Hedberg333ae952015-03-17 13:48:47 +02004902 cmd = pending_find_data(MGMT_OP_GET_CONN_INFO, hdev, conn);
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004903 if (!cmd)
4904 goto unlock;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004905
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004906 cmd->cmd_complete(cmd, status);
4907 mgmt_pending_remove(cmd);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004908
4909unlock:
4910 hci_dev_unlock(hdev);
4911}
4912
4913static int get_conn_info(struct sock *sk, struct hci_dev *hdev, void *data,
4914 u16 len)
4915{
4916 struct mgmt_cp_get_conn_info *cp = data;
4917 struct mgmt_rp_get_conn_info rp;
4918 struct hci_conn *conn;
4919 unsigned long conn_info_age;
4920 int err = 0;
4921
4922 BT_DBG("%s", hdev->name);
4923
4924 memset(&rp, 0, sizeof(rp));
4925 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
4926 rp.addr.type = cp->addr.type;
4927
4928 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004929 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4930 MGMT_STATUS_INVALID_PARAMS,
4931 &rp, sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004932
4933 hci_dev_lock(hdev);
4934
4935 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004936 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4937 MGMT_STATUS_NOT_POWERED, &rp,
4938 sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004939 goto unlock;
4940 }
4941
4942 if (cp->addr.type == BDADDR_BREDR)
4943 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
4944 &cp->addr.bdaddr);
4945 else
4946 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &cp->addr.bdaddr);
4947
4948 if (!conn || conn->state != BT_CONNECTED) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004949 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4950 MGMT_STATUS_NOT_CONNECTED, &rp,
4951 sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004952 goto unlock;
4953 }
4954
Johan Hedberg333ae952015-03-17 13:48:47 +02004955 if (pending_find_data(MGMT_OP_GET_CONN_INFO, hdev, conn)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004956 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4957 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004958 goto unlock;
4959 }
4960
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004961 /* To avoid client trying to guess when to poll again for information we
4962 * calculate conn info age as random value between min/max set in hdev.
4963 */
4964 conn_info_age = hdev->conn_info_min_age +
4965 prandom_u32_max(hdev->conn_info_max_age -
4966 hdev->conn_info_min_age);
4967
4968 /* Query controller to refresh cached values if they are too old or were
4969 * never read.
4970 */
Andrzej Kaczmarekf4e2dd52014-05-16 16:48:57 +02004971 if (time_after(jiffies, conn->conn_info_timestamp +
4972 msecs_to_jiffies(conn_info_age)) ||
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004973 !conn->conn_info_timestamp) {
4974 struct hci_request req;
4975 struct hci_cp_read_tx_power req_txp_cp;
4976 struct hci_cp_read_rssi req_rssi_cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004977 struct mgmt_pending_cmd *cmd;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004978
4979 hci_req_init(&req, hdev);
4980 req_rssi_cp.handle = cpu_to_le16(conn->handle);
4981 hci_req_add(&req, HCI_OP_READ_RSSI, sizeof(req_rssi_cp),
4982 &req_rssi_cp);
4983
Andrzej Kaczmarekf7faab02014-05-14 13:43:04 +02004984 /* For LE links TX power does not change thus we don't need to
4985 * query for it once value is known.
4986 */
4987 if (!bdaddr_type_is_le(cp->addr.type) ||
4988 conn->tx_power == HCI_TX_POWER_INVALID) {
4989 req_txp_cp.handle = cpu_to_le16(conn->handle);
4990 req_txp_cp.type = 0x00;
4991 hci_req_add(&req, HCI_OP_READ_TX_POWER,
4992 sizeof(req_txp_cp), &req_txp_cp);
4993 }
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004994
Andrzej Kaczmarekeed5daf2014-05-14 13:43:06 +02004995 /* Max TX power needs to be read only once per connection */
4996 if (conn->max_tx_power == HCI_TX_POWER_INVALID) {
4997 req_txp_cp.handle = cpu_to_le16(conn->handle);
4998 req_txp_cp.type = 0x01;
4999 hci_req_add(&req, HCI_OP_READ_TX_POWER,
5000 sizeof(req_txp_cp), &req_txp_cp);
5001 }
5002
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005003 err = hci_req_run(&req, conn_info_refresh_complete);
5004 if (err < 0)
5005 goto unlock;
5006
5007 cmd = mgmt_pending_add(sk, MGMT_OP_GET_CONN_INFO, hdev,
5008 data, len);
5009 if (!cmd) {
5010 err = -ENOMEM;
5011 goto unlock;
5012 }
5013
5014 hci_conn_hold(conn);
Johan Hedbergf8aaf9b2014-08-17 23:28:57 +03005015 cmd->user_data = hci_conn_get(conn);
Johan Hedberg9981bdb2014-12-05 13:42:57 +02005016 cmd->cmd_complete = conn_info_cmd_complete;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005017
5018 conn->conn_info_timestamp = jiffies;
5019 } else {
5020 /* Cache is valid, just reply with values cached in hci_conn */
5021 rp.rssi = conn->rssi;
5022 rp.tx_power = conn->tx_power;
Andrzej Kaczmarekeed5daf2014-05-14 13:43:06 +02005023 rp.max_tx_power = conn->max_tx_power;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005024
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005025 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
5026 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005027 }
5028
5029unlock:
5030 hci_dev_unlock(hdev);
5031 return err;
5032}
5033
Johan Hedberg3b0602c2015-03-06 21:08:55 +02005034static int clock_info_cmd_complete(struct mgmt_pending_cmd *cmd, u8 status)
Johan Hedberg69487372014-12-05 13:36:07 +02005035{
5036 struct hci_conn *conn = cmd->user_data;
5037 struct mgmt_rp_get_clock_info rp;
5038 struct hci_dev *hdev;
Johan Hedberg9df74652014-12-19 22:26:03 +02005039 int err;
Johan Hedberg69487372014-12-05 13:36:07 +02005040
5041 memset(&rp, 0, sizeof(rp));
Marcel Holtmann56f787c2016-08-29 06:19:47 +02005042 memcpy(&rp.addr, cmd->param, sizeof(rp.addr));
Johan Hedberg69487372014-12-05 13:36:07 +02005043
5044 if (status)
5045 goto complete;
5046
5047 hdev = hci_dev_get(cmd->index);
5048 if (hdev) {
5049 rp.local_clock = cpu_to_le32(hdev->clock);
5050 hci_dev_put(hdev);
5051 }
5052
5053 if (conn) {
5054 rp.piconet_clock = cpu_to_le32(conn->clock);
5055 rp.accuracy = cpu_to_le16(conn->clock_accuracy);
5056 }
5057
5058complete:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005059 err = mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, status, &rp,
5060 sizeof(rp));
Johan Hedberg69487372014-12-05 13:36:07 +02005061
5062 if (conn) {
5063 hci_conn_drop(conn);
5064 hci_conn_put(conn);
5065 }
Johan Hedberg9df74652014-12-19 22:26:03 +02005066
5067 return err;
Johan Hedberg69487372014-12-05 13:36:07 +02005068}
5069
Marcel Holtmann1904a852015-01-11 13:50:44 -08005070static void get_clock_info_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg95868422014-06-28 17:54:07 +03005071{
Johan Hedberg95868422014-06-28 17:54:07 +03005072 struct hci_cp_read_clock *hci_cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02005073 struct mgmt_pending_cmd *cmd;
Johan Hedberg95868422014-06-28 17:54:07 +03005074 struct hci_conn *conn;
5075
5076 BT_DBG("%s status %u", hdev->name, status);
5077
5078 hci_dev_lock(hdev);
5079
5080 hci_cp = hci_sent_cmd_data(hdev, HCI_OP_READ_CLOCK);
5081 if (!hci_cp)
5082 goto unlock;
5083
5084 if (hci_cp->which) {
5085 u16 handle = __le16_to_cpu(hci_cp->handle);
5086 conn = hci_conn_hash_lookup_handle(hdev, handle);
5087 } else {
5088 conn = NULL;
5089 }
5090
Johan Hedberg333ae952015-03-17 13:48:47 +02005091 cmd = pending_find_data(MGMT_OP_GET_CLOCK_INFO, hdev, conn);
Johan Hedberg95868422014-06-28 17:54:07 +03005092 if (!cmd)
5093 goto unlock;
5094
Johan Hedberg69487372014-12-05 13:36:07 +02005095 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberg95868422014-06-28 17:54:07 +03005096 mgmt_pending_remove(cmd);
Johan Hedberg95868422014-06-28 17:54:07 +03005097
5098unlock:
5099 hci_dev_unlock(hdev);
5100}
5101
5102static int get_clock_info(struct sock *sk, struct hci_dev *hdev, void *data,
5103 u16 len)
5104{
5105 struct mgmt_cp_get_clock_info *cp = data;
5106 struct mgmt_rp_get_clock_info rp;
5107 struct hci_cp_read_clock hci_cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02005108 struct mgmt_pending_cmd *cmd;
Johan Hedberg95868422014-06-28 17:54:07 +03005109 struct hci_request req;
5110 struct hci_conn *conn;
5111 int err;
5112
5113 BT_DBG("%s", hdev->name);
5114
5115 memset(&rp, 0, sizeof(rp));
5116 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
5117 rp.addr.type = cp->addr.type;
5118
5119 if (cp->addr.type != BDADDR_BREDR)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005120 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CLOCK_INFO,
5121 MGMT_STATUS_INVALID_PARAMS,
5122 &rp, sizeof(rp));
Johan Hedberg95868422014-06-28 17:54:07 +03005123
5124 hci_dev_lock(hdev);
5125
5126 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005127 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CLOCK_INFO,
5128 MGMT_STATUS_NOT_POWERED, &rp,
5129 sizeof(rp));
Johan Hedberg95868422014-06-28 17:54:07 +03005130 goto unlock;
5131 }
5132
5133 if (bacmp(&cp->addr.bdaddr, BDADDR_ANY)) {
5134 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
5135 &cp->addr.bdaddr);
5136 if (!conn || conn->state != BT_CONNECTED) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005137 err = mgmt_cmd_complete(sk, hdev->id,
5138 MGMT_OP_GET_CLOCK_INFO,
5139 MGMT_STATUS_NOT_CONNECTED,
5140 &rp, sizeof(rp));
Johan Hedberg95868422014-06-28 17:54:07 +03005141 goto unlock;
5142 }
5143 } else {
5144 conn = NULL;
5145 }
5146
5147 cmd = mgmt_pending_add(sk, MGMT_OP_GET_CLOCK_INFO, hdev, data, len);
5148 if (!cmd) {
5149 err = -ENOMEM;
5150 goto unlock;
5151 }
5152
Johan Hedberg69487372014-12-05 13:36:07 +02005153 cmd->cmd_complete = clock_info_cmd_complete;
5154
Johan Hedberg95868422014-06-28 17:54:07 +03005155 hci_req_init(&req, hdev);
5156
5157 memset(&hci_cp, 0, sizeof(hci_cp));
5158 hci_req_add(&req, HCI_OP_READ_CLOCK, sizeof(hci_cp), &hci_cp);
5159
5160 if (conn) {
5161 hci_conn_hold(conn);
Johan Hedbergf8aaf9b2014-08-17 23:28:57 +03005162 cmd->user_data = hci_conn_get(conn);
Johan Hedberg95868422014-06-28 17:54:07 +03005163
5164 hci_cp.handle = cpu_to_le16(conn->handle);
5165 hci_cp.which = 0x01; /* Piconet clock */
5166 hci_req_add(&req, HCI_OP_READ_CLOCK, sizeof(hci_cp), &hci_cp);
5167 }
5168
5169 err = hci_req_run(&req, get_clock_info_complete);
5170 if (err < 0)
5171 mgmt_pending_remove(cmd);
5172
5173unlock:
5174 hci_dev_unlock(hdev);
5175 return err;
5176}
5177
Johan Hedberg5a154e62014-12-19 22:26:02 +02005178static bool is_connected(struct hci_dev *hdev, bdaddr_t *addr, u8 type)
5179{
5180 struct hci_conn *conn;
5181
5182 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, addr);
5183 if (!conn)
5184 return false;
5185
5186 if (conn->dst_type != type)
5187 return false;
5188
5189 if (conn->state != BT_CONNECTED)
5190 return false;
5191
5192 return true;
5193}
5194
5195/* This function requires the caller holds hdev->lock */
Johan Hedberg51d7a942015-11-11 08:11:18 +02005196static int hci_conn_params_set(struct hci_dev *hdev, bdaddr_t *addr,
Johan Hedberg5a154e62014-12-19 22:26:02 +02005197 u8 addr_type, u8 auto_connect)
5198{
Johan Hedberg5a154e62014-12-19 22:26:02 +02005199 struct hci_conn_params *params;
5200
5201 params = hci_conn_params_add(hdev, addr, addr_type);
5202 if (!params)
5203 return -EIO;
5204
5205 if (params->auto_connect == auto_connect)
5206 return 0;
5207
5208 list_del_init(&params->action);
5209
5210 switch (auto_connect) {
5211 case HCI_AUTO_CONN_DISABLED:
5212 case HCI_AUTO_CONN_LINK_LOSS:
Jakub Pawlowski28a667c2015-08-07 20:22:54 +02005213 /* If auto connect is being disabled when we're trying to
5214 * connect to device, keep connecting.
5215 */
5216 if (params->explicit_connect)
5217 list_add(&params->action, &hdev->pend_le_conns);
Johan Hedberg5a154e62014-12-19 22:26:02 +02005218 break;
5219 case HCI_AUTO_CONN_REPORT:
Johan Hedberg49c50922015-10-16 10:07:51 +03005220 if (params->explicit_connect)
5221 list_add(&params->action, &hdev->pend_le_conns);
5222 else
5223 list_add(&params->action, &hdev->pend_le_reports);
Johan Hedberg5a154e62014-12-19 22:26:02 +02005224 break;
5225 case HCI_AUTO_CONN_DIRECT:
5226 case HCI_AUTO_CONN_ALWAYS:
Johan Hedberg51d7a942015-11-11 08:11:18 +02005227 if (!is_connected(hdev, addr, addr_type))
Johan Hedberg5a154e62014-12-19 22:26:02 +02005228 list_add(&params->action, &hdev->pend_le_conns);
Johan Hedberg5a154e62014-12-19 22:26:02 +02005229 break;
5230 }
5231
5232 params->auto_connect = auto_connect;
5233
5234 BT_DBG("addr %pMR (type %u) auto_connect %u", addr, addr_type,
5235 auto_connect);
5236
5237 return 0;
5238}
5239
Marcel Holtmann8afef092014-06-29 22:28:34 +02005240static void device_added(struct sock *sk, struct hci_dev *hdev,
5241 bdaddr_t *bdaddr, u8 type, u8 action)
5242{
5243 struct mgmt_ev_device_added ev;
5244
5245 bacpy(&ev.addr.bdaddr, bdaddr);
5246 ev.addr.type = type;
5247 ev.action = action;
5248
5249 mgmt_event(MGMT_EV_DEVICE_ADDED, hdev, &ev, sizeof(ev), sk);
5250}
5251
Marcel Holtmann2faade52014-06-29 19:44:03 +02005252static int add_device(struct sock *sk, struct hci_dev *hdev,
5253 void *data, u16 len)
5254{
5255 struct mgmt_cp_add_device *cp = data;
5256 u8 auto_conn, addr_type;
5257 int err;
5258
5259 BT_DBG("%s", hdev->name);
5260
Johan Hedberg66593582014-07-09 12:59:14 +03005261 if (!bdaddr_type_is_valid(cp->addr.type) ||
Marcel Holtmann2faade52014-06-29 19:44:03 +02005262 !bacmp(&cp->addr.bdaddr, BDADDR_ANY))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005263 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5264 MGMT_STATUS_INVALID_PARAMS,
5265 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005266
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005267 if (cp->action != 0x00 && cp->action != 0x01 && cp->action != 0x02)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005268 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5269 MGMT_STATUS_INVALID_PARAMS,
5270 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005271
5272 hci_dev_lock(hdev);
5273
Johan Hedberg66593582014-07-09 12:59:14 +03005274 if (cp->addr.type == BDADDR_BREDR) {
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005275 /* Only incoming connections action is supported for now */
Johan Hedberg66593582014-07-09 12:59:14 +03005276 if (cp->action != 0x01) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005277 err = mgmt_cmd_complete(sk, hdev->id,
5278 MGMT_OP_ADD_DEVICE,
5279 MGMT_STATUS_INVALID_PARAMS,
5280 &cp->addr, sizeof(cp->addr));
Johan Hedberg66593582014-07-09 12:59:14 +03005281 goto unlock;
5282 }
5283
5284 err = hci_bdaddr_list_add(&hdev->whitelist, &cp->addr.bdaddr,
5285 cp->addr.type);
5286 if (err)
5287 goto unlock;
Johan Hedberga3974072014-07-09 12:59:15 +03005288
Johan Hedberg01b1cb82015-11-16 12:52:21 +02005289 hci_req_update_scan(hdev);
Johan Hedberga3974072014-07-09 12:59:15 +03005290
Johan Hedberg66593582014-07-09 12:59:14 +03005291 goto added;
5292 }
5293
Johan Hedberg85813a72015-10-21 18:02:59 +03005294 addr_type = le_addr_type(cp->addr.type);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005295
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005296 if (cp->action == 0x02)
Marcel Holtmann2faade52014-06-29 19:44:03 +02005297 auto_conn = HCI_AUTO_CONN_ALWAYS;
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005298 else if (cp->action == 0x01)
5299 auto_conn = HCI_AUTO_CONN_DIRECT;
Marcel Holtmann2faade52014-06-29 19:44:03 +02005300 else
Johan Hedberga3451d22014-07-02 17:37:27 +03005301 auto_conn = HCI_AUTO_CONN_REPORT;
Marcel Holtmann2faade52014-06-29 19:44:03 +02005302
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005303 /* Kernel internally uses conn_params with resolvable private
5304 * address, but Add Device allows only identity addresses.
5305 * Make sure it is enforced before calling
5306 * hci_conn_params_lookup.
5307 */
5308 if (!hci_is_identity_address(&cp->addr.bdaddr, addr_type)) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005309 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5310 MGMT_STATUS_INVALID_PARAMS,
5311 &cp->addr, sizeof(cp->addr));
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005312 goto unlock;
5313 }
5314
Marcel Holtmannbf5b3c82014-06-30 12:34:39 +02005315 /* If the connection parameters don't exist for this device,
5316 * they will be created and configured with defaults.
5317 */
Johan Hedberg51d7a942015-11-11 08:11:18 +02005318 if (hci_conn_params_set(hdev, &cp->addr.bdaddr, addr_type,
Marcel Holtmannd06b50c2014-07-01 12:11:06 +02005319 auto_conn) < 0) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005320 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5321 MGMT_STATUS_FAILED, &cp->addr,
5322 sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005323 goto unlock;
5324 }
5325
Johan Hedberg51d7a942015-11-11 08:11:18 +02005326 hci_update_background_scan(hdev);
5327
Johan Hedberg66593582014-07-09 12:59:14 +03005328added:
Marcel Holtmann8afef092014-06-29 22:28:34 +02005329 device_added(sk, hdev, &cp->addr.bdaddr, cp->addr.type, cp->action);
5330
Johan Hedberg51d7a942015-11-11 08:11:18 +02005331 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5332 MGMT_STATUS_SUCCESS, &cp->addr,
5333 sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005334
5335unlock:
5336 hci_dev_unlock(hdev);
5337 return err;
5338}
5339
Marcel Holtmann8afef092014-06-29 22:28:34 +02005340static void device_removed(struct sock *sk, struct hci_dev *hdev,
5341 bdaddr_t *bdaddr, u8 type)
5342{
5343 struct mgmt_ev_device_removed ev;
5344
5345 bacpy(&ev.addr.bdaddr, bdaddr);
5346 ev.addr.type = type;
5347
5348 mgmt_event(MGMT_EV_DEVICE_REMOVED, hdev, &ev, sizeof(ev), sk);
5349}
5350
Marcel Holtmann2faade52014-06-29 19:44:03 +02005351static int remove_device(struct sock *sk, struct hci_dev *hdev,
5352 void *data, u16 len)
5353{
5354 struct mgmt_cp_remove_device *cp = data;
5355 int err;
5356
5357 BT_DBG("%s", hdev->name);
5358
5359 hci_dev_lock(hdev);
5360
5361 if (bacmp(&cp->addr.bdaddr, BDADDR_ANY)) {
Johan Hedbergc71593d2014-07-02 17:37:28 +03005362 struct hci_conn_params *params;
Marcel Holtmann2faade52014-06-29 19:44:03 +02005363 u8 addr_type;
5364
Johan Hedberg66593582014-07-09 12:59:14 +03005365 if (!bdaddr_type_is_valid(cp->addr.type)) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005366 err = mgmt_cmd_complete(sk, hdev->id,
5367 MGMT_OP_REMOVE_DEVICE,
5368 MGMT_STATUS_INVALID_PARAMS,
5369 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005370 goto unlock;
5371 }
5372
Johan Hedberg66593582014-07-09 12:59:14 +03005373 if (cp->addr.type == BDADDR_BREDR) {
5374 err = hci_bdaddr_list_del(&hdev->whitelist,
5375 &cp->addr.bdaddr,
5376 cp->addr.type);
5377 if (err) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005378 err = mgmt_cmd_complete(sk, hdev->id,
5379 MGMT_OP_REMOVE_DEVICE,
5380 MGMT_STATUS_INVALID_PARAMS,
5381 &cp->addr,
5382 sizeof(cp->addr));
Johan Hedberg66593582014-07-09 12:59:14 +03005383 goto unlock;
5384 }
5385
Johan Hedberg01b1cb82015-11-16 12:52:21 +02005386 hci_req_update_scan(hdev);
Johan Hedberga3974072014-07-09 12:59:15 +03005387
Johan Hedberg66593582014-07-09 12:59:14 +03005388 device_removed(sk, hdev, &cp->addr.bdaddr,
5389 cp->addr.type);
5390 goto complete;
5391 }
5392
Johan Hedberg85813a72015-10-21 18:02:59 +03005393 addr_type = le_addr_type(cp->addr.type);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005394
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005395 /* Kernel internally uses conn_params with resolvable private
5396 * address, but Remove Device allows only identity addresses.
5397 * Make sure it is enforced before calling
5398 * hci_conn_params_lookup.
5399 */
5400 if (!hci_is_identity_address(&cp->addr.bdaddr, addr_type)) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005401 err = mgmt_cmd_complete(sk, hdev->id,
5402 MGMT_OP_REMOVE_DEVICE,
5403 MGMT_STATUS_INVALID_PARAMS,
5404 &cp->addr, sizeof(cp->addr));
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005405 goto unlock;
5406 }
5407
Johan Hedbergc71593d2014-07-02 17:37:28 +03005408 params = hci_conn_params_lookup(hdev, &cp->addr.bdaddr,
5409 addr_type);
5410 if (!params) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005411 err = mgmt_cmd_complete(sk, hdev->id,
5412 MGMT_OP_REMOVE_DEVICE,
5413 MGMT_STATUS_INVALID_PARAMS,
5414 &cp->addr, sizeof(cp->addr));
Johan Hedbergc71593d2014-07-02 17:37:28 +03005415 goto unlock;
5416 }
5417
Johan Hedberg679d2b62015-10-16 10:07:52 +03005418 if (params->auto_connect == HCI_AUTO_CONN_DISABLED ||
5419 params->auto_connect == HCI_AUTO_CONN_EXPLICIT) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005420 err = mgmt_cmd_complete(sk, hdev->id,
5421 MGMT_OP_REMOVE_DEVICE,
5422 MGMT_STATUS_INVALID_PARAMS,
5423 &cp->addr, sizeof(cp->addr));
Johan Hedbergc71593d2014-07-02 17:37:28 +03005424 goto unlock;
5425 }
5426
Johan Hedbergd1dbf122014-07-04 16:17:23 +03005427 list_del(&params->action);
Johan Hedbergc71593d2014-07-02 17:37:28 +03005428 list_del(&params->list);
5429 kfree(params);
Johan Hedberg51d7a942015-11-11 08:11:18 +02005430 hci_update_background_scan(hdev);
Marcel Holtmann8afef092014-06-29 22:28:34 +02005431
5432 device_removed(sk, hdev, &cp->addr.bdaddr, cp->addr.type);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005433 } else {
Johan Hedberg19de0822014-07-06 13:06:51 +03005434 struct hci_conn_params *p, *tmp;
Johan Hedberg66593582014-07-09 12:59:14 +03005435 struct bdaddr_list *b, *btmp;
Johan Hedberg19de0822014-07-06 13:06:51 +03005436
Marcel Holtmann2faade52014-06-29 19:44:03 +02005437 if (cp->addr.type) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005438 err = mgmt_cmd_complete(sk, hdev->id,
5439 MGMT_OP_REMOVE_DEVICE,
5440 MGMT_STATUS_INVALID_PARAMS,
5441 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005442 goto unlock;
5443 }
5444
Johan Hedberg66593582014-07-09 12:59:14 +03005445 list_for_each_entry_safe(b, btmp, &hdev->whitelist, list) {
5446 device_removed(sk, hdev, &b->bdaddr, b->bdaddr_type);
5447 list_del(&b->list);
5448 kfree(b);
5449 }
5450
Johan Hedberg01b1cb82015-11-16 12:52:21 +02005451 hci_req_update_scan(hdev);
Johan Hedberga3974072014-07-09 12:59:15 +03005452
Johan Hedberg19de0822014-07-06 13:06:51 +03005453 list_for_each_entry_safe(p, tmp, &hdev->le_conn_params, list) {
5454 if (p->auto_connect == HCI_AUTO_CONN_DISABLED)
5455 continue;
5456 device_removed(sk, hdev, &p->addr, p->addr_type);
Johan Hedberg679d2b62015-10-16 10:07:52 +03005457 if (p->explicit_connect) {
5458 p->auto_connect = HCI_AUTO_CONN_EXPLICIT;
5459 continue;
5460 }
Johan Hedberg19de0822014-07-06 13:06:51 +03005461 list_del(&p->action);
5462 list_del(&p->list);
5463 kfree(p);
5464 }
5465
5466 BT_DBG("All LE connection parameters were removed");
5467
Johan Hedberg51d7a942015-11-11 08:11:18 +02005468 hci_update_background_scan(hdev);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005469 }
5470
Johan Hedberg66593582014-07-09 12:59:14 +03005471complete:
Johan Hedberg51d7a942015-11-11 08:11:18 +02005472 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_DEVICE,
5473 MGMT_STATUS_SUCCESS, &cp->addr,
5474 sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005475unlock:
5476 hci_dev_unlock(hdev);
5477 return err;
5478}
5479
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005480static int load_conn_param(struct sock *sk, struct hci_dev *hdev, void *data,
5481 u16 len)
5482{
5483 struct mgmt_cp_load_conn_param *cp = data;
Johan Hedbergba1d6932014-07-03 13:52:27 +03005484 const u16 max_param_count = ((U16_MAX - sizeof(*cp)) /
5485 sizeof(struct mgmt_conn_param));
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005486 u16 param_count, expected_len;
5487 int i;
5488
5489 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02005490 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM,
5491 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005492
5493 param_count = __le16_to_cpu(cp->param_count);
Johan Hedbergba1d6932014-07-03 13:52:27 +03005494 if (param_count > max_param_count) {
5495 BT_ERR("load_conn_param: too big param_count value %u",
5496 param_count);
Johan Hedberga69e8372015-03-06 21:08:53 +02005497 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM,
5498 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergba1d6932014-07-03 13:52:27 +03005499 }
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005500
5501 expected_len = sizeof(*cp) + param_count *
5502 sizeof(struct mgmt_conn_param);
5503 if (expected_len != len) {
5504 BT_ERR("load_conn_param: expected %u bytes, got %u bytes",
5505 expected_len, len);
Johan Hedberga69e8372015-03-06 21:08:53 +02005506 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM,
5507 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005508 }
5509
5510 BT_DBG("%s param_count %u", hdev->name, param_count);
5511
5512 hci_dev_lock(hdev);
5513
5514 hci_conn_params_clear_disabled(hdev);
5515
5516 for (i = 0; i < param_count; i++) {
5517 struct mgmt_conn_param *param = &cp->params[i];
5518 struct hci_conn_params *hci_param;
5519 u16 min, max, latency, timeout;
5520 u8 addr_type;
5521
5522 BT_DBG("Adding %pMR (type %u)", &param->addr.bdaddr,
5523 param->addr.type);
5524
5525 if (param->addr.type == BDADDR_LE_PUBLIC) {
5526 addr_type = ADDR_LE_DEV_PUBLIC;
5527 } else if (param->addr.type == BDADDR_LE_RANDOM) {
5528 addr_type = ADDR_LE_DEV_RANDOM;
5529 } else {
5530 BT_ERR("Ignoring invalid connection parameters");
5531 continue;
5532 }
5533
5534 min = le16_to_cpu(param->min_interval);
5535 max = le16_to_cpu(param->max_interval);
5536 latency = le16_to_cpu(param->latency);
5537 timeout = le16_to_cpu(param->timeout);
5538
5539 BT_DBG("min 0x%04x max 0x%04x latency 0x%04x timeout 0x%04x",
5540 min, max, latency, timeout);
5541
5542 if (hci_check_conn_params(min, max, latency, timeout) < 0) {
5543 BT_ERR("Ignoring invalid connection parameters");
5544 continue;
5545 }
5546
5547 hci_param = hci_conn_params_add(hdev, &param->addr.bdaddr,
5548 addr_type);
5549 if (!hci_param) {
5550 BT_ERR("Failed to add connection parameters");
5551 continue;
5552 }
5553
5554 hci_param->conn_min_interval = min;
5555 hci_param->conn_max_interval = max;
5556 hci_param->conn_latency = latency;
5557 hci_param->supervision_timeout = timeout;
5558 }
5559
5560 hci_dev_unlock(hdev);
5561
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005562 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM, 0,
5563 NULL, 0);
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005564}
5565
Marcel Holtmanndbece372014-07-04 18:11:55 +02005566static int set_external_config(struct sock *sk, struct hci_dev *hdev,
5567 void *data, u16 len)
5568{
5569 struct mgmt_cp_set_external_config *cp = data;
5570 bool changed;
5571 int err;
5572
5573 BT_DBG("%s", hdev->name);
5574
5575 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02005576 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_EXTERNAL_CONFIG,
5577 MGMT_STATUS_REJECTED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005578
5579 if (cp->config != 0x00 && cp->config != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02005580 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_EXTERNAL_CONFIG,
5581 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005582
5583 if (!test_bit(HCI_QUIRK_EXTERNAL_CONFIG, &hdev->quirks))
Johan Hedberga69e8372015-03-06 21:08:53 +02005584 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_EXTERNAL_CONFIG,
5585 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005586
5587 hci_dev_lock(hdev);
5588
5589 if (cp->config)
Marcel Holtmann238be782015-03-13 02:11:06 -07005590 changed = !hci_dev_test_and_set_flag(hdev, HCI_EXT_CONFIGURED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005591 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07005592 changed = hci_dev_test_and_clear_flag(hdev, HCI_EXT_CONFIGURED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005593
5594 err = send_options_rsp(sk, MGMT_OP_SET_EXTERNAL_CONFIG, hdev);
5595 if (err < 0)
5596 goto unlock;
5597
5598 if (!changed)
5599 goto unlock;
5600
Marcel Holtmannf4537c02014-07-04 19:06:23 +02005601 err = new_options(hdev, sk);
5602
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07005603 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) == is_configured(hdev)) {
Marcel Holtmanndbece372014-07-04 18:11:55 +02005604 mgmt_index_removed(hdev);
Marcel Holtmannd603b762014-07-06 12:11:14 +02005605
Marcel Holtmann516018a2015-03-13 02:11:04 -07005606 if (hci_dev_test_and_change_flag(hdev, HCI_UNCONFIGURED)) {
Marcel Holtmanna1536da2015-03-13 02:11:01 -07005607 hci_dev_set_flag(hdev, HCI_CONFIG);
5608 hci_dev_set_flag(hdev, HCI_AUTO_OFF);
Marcel Holtmannd603b762014-07-06 12:11:14 +02005609
5610 queue_work(hdev->req_workqueue, &hdev->power_on);
5611 } else {
Marcel Holtmann5ea234d2014-07-06 12:11:16 +02005612 set_bit(HCI_RAW, &hdev->flags);
Marcel Holtmannd603b762014-07-06 12:11:14 +02005613 mgmt_index_added(hdev);
5614 }
Marcel Holtmanndbece372014-07-04 18:11:55 +02005615 }
5616
5617unlock:
5618 hci_dev_unlock(hdev);
5619 return err;
5620}
5621
Marcel Holtmann9713c172014-07-06 12:11:15 +02005622static int set_public_address(struct sock *sk, struct hci_dev *hdev,
5623 void *data, u16 len)
5624{
5625 struct mgmt_cp_set_public_address *cp = data;
5626 bool changed;
5627 int err;
5628
5629 BT_DBG("%s", hdev->name);
5630
5631 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02005632 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PUBLIC_ADDRESS,
5633 MGMT_STATUS_REJECTED);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005634
5635 if (!bacmp(&cp->bdaddr, BDADDR_ANY))
Johan Hedberga69e8372015-03-06 21:08:53 +02005636 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PUBLIC_ADDRESS,
5637 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005638
5639 if (!hdev->set_bdaddr)
Johan Hedberga69e8372015-03-06 21:08:53 +02005640 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PUBLIC_ADDRESS,
5641 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005642
5643 hci_dev_lock(hdev);
5644
5645 changed = !!bacmp(&hdev->public_addr, &cp->bdaddr);
5646 bacpy(&hdev->public_addr, &cp->bdaddr);
5647
5648 err = send_options_rsp(sk, MGMT_OP_SET_PUBLIC_ADDRESS, hdev);
5649 if (err < 0)
5650 goto unlock;
5651
5652 if (!changed)
5653 goto unlock;
5654
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07005655 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
Marcel Holtmann9713c172014-07-06 12:11:15 +02005656 err = new_options(hdev, sk);
5657
5658 if (is_configured(hdev)) {
5659 mgmt_index_removed(hdev);
5660
Marcel Holtmanna358dc12015-03-13 02:11:02 -07005661 hci_dev_clear_flag(hdev, HCI_UNCONFIGURED);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005662
Marcel Holtmanna1536da2015-03-13 02:11:01 -07005663 hci_dev_set_flag(hdev, HCI_CONFIG);
5664 hci_dev_set_flag(hdev, HCI_AUTO_OFF);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005665
5666 queue_work(hdev->req_workqueue, &hdev->power_on);
5667 }
5668
5669unlock:
5670 hci_dev_unlock(hdev);
5671 return err;
5672}
5673
Johan Hedberg40f66c02015-04-07 21:52:22 +03005674static void read_local_oob_ext_data_complete(struct hci_dev *hdev, u8 status,
5675 u16 opcode, struct sk_buff *skb)
5676{
5677 const struct mgmt_cp_read_local_oob_ext_data *mgmt_cp;
5678 struct mgmt_rp_read_local_oob_ext_data *mgmt_rp;
5679 u8 *h192, *r192, *h256, *r256;
5680 struct mgmt_pending_cmd *cmd;
5681 u16 eir_len;
5682 int err;
5683
5684 BT_DBG("%s status %u", hdev->name, status);
5685
5686 cmd = pending_find(MGMT_OP_READ_LOCAL_OOB_EXT_DATA, hdev);
5687 if (!cmd)
5688 return;
5689
5690 mgmt_cp = cmd->param;
5691
5692 if (status) {
5693 status = mgmt_status(status);
5694 eir_len = 0;
5695
5696 h192 = NULL;
5697 r192 = NULL;
5698 h256 = NULL;
5699 r256 = NULL;
5700 } else if (opcode == HCI_OP_READ_LOCAL_OOB_DATA) {
5701 struct hci_rp_read_local_oob_data *rp;
5702
5703 if (skb->len != sizeof(*rp)) {
5704 status = MGMT_STATUS_FAILED;
5705 eir_len = 0;
5706 } else {
5707 status = MGMT_STATUS_SUCCESS;
5708 rp = (void *)skb->data;
5709
5710 eir_len = 5 + 18 + 18;
5711 h192 = rp->hash;
5712 r192 = rp->rand;
5713 h256 = NULL;
5714 r256 = NULL;
5715 }
5716 } else {
5717 struct hci_rp_read_local_oob_ext_data *rp;
5718
5719 if (skb->len != sizeof(*rp)) {
5720 status = MGMT_STATUS_FAILED;
5721 eir_len = 0;
5722 } else {
5723 status = MGMT_STATUS_SUCCESS;
5724 rp = (void *)skb->data;
5725
5726 if (hci_dev_test_flag(hdev, HCI_SC_ONLY)) {
5727 eir_len = 5 + 18 + 18;
5728 h192 = NULL;
5729 r192 = NULL;
5730 } else {
5731 eir_len = 5 + 18 + 18 + 18 + 18;
5732 h192 = rp->hash192;
5733 r192 = rp->rand192;
5734 }
5735
5736 h256 = rp->hash256;
5737 r256 = rp->rand256;
5738 }
5739 }
5740
5741 mgmt_rp = kmalloc(sizeof(*mgmt_rp) + eir_len, GFP_KERNEL);
5742 if (!mgmt_rp)
5743 goto done;
5744
5745 if (status)
5746 goto send_rsp;
5747
5748 eir_len = eir_append_data(mgmt_rp->eir, 0, EIR_CLASS_OF_DEV,
5749 hdev->dev_class, 3);
5750
5751 if (h192 && r192) {
5752 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5753 EIR_SSP_HASH_C192, h192, 16);
5754 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5755 EIR_SSP_RAND_R192, r192, 16);
5756 }
5757
5758 if (h256 && r256) {
5759 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5760 EIR_SSP_HASH_C256, h256, 16);
5761 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5762 EIR_SSP_RAND_R256, r256, 16);
5763 }
5764
5765send_rsp:
5766 mgmt_rp->type = mgmt_cp->type;
5767 mgmt_rp->eir_len = cpu_to_le16(eir_len);
5768
5769 err = mgmt_cmd_complete(cmd->sk, hdev->id,
5770 MGMT_OP_READ_LOCAL_OOB_EXT_DATA, status,
5771 mgmt_rp, sizeof(*mgmt_rp) + eir_len);
5772 if (err < 0 || status)
5773 goto done;
5774
5775 hci_sock_set_flag(cmd->sk, HCI_MGMT_OOB_DATA_EVENTS);
5776
5777 err = mgmt_limited_event(MGMT_EV_LOCAL_OOB_DATA_UPDATED, hdev,
5778 mgmt_rp, sizeof(*mgmt_rp) + eir_len,
5779 HCI_MGMT_OOB_DATA_EVENTS, cmd->sk);
5780done:
5781 kfree(mgmt_rp);
5782 mgmt_pending_remove(cmd);
5783}
5784
5785static int read_local_ssp_oob_req(struct hci_dev *hdev, struct sock *sk,
5786 struct mgmt_cp_read_local_oob_ext_data *cp)
5787{
5788 struct mgmt_pending_cmd *cmd;
5789 struct hci_request req;
5790 int err;
5791
5792 cmd = mgmt_pending_add(sk, MGMT_OP_READ_LOCAL_OOB_EXT_DATA, hdev,
5793 cp, sizeof(*cp));
5794 if (!cmd)
5795 return -ENOMEM;
5796
5797 hci_req_init(&req, hdev);
5798
5799 if (bredr_sc_enabled(hdev))
5800 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_EXT_DATA, 0, NULL);
5801 else
5802 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_DATA, 0, NULL);
5803
5804 err = hci_req_run_skb(&req, read_local_oob_ext_data_complete);
5805 if (err < 0) {
5806 mgmt_pending_remove(cmd);
5807 return err;
5808 }
5809
5810 return 0;
5811}
5812
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005813static int read_local_oob_ext_data(struct sock *sk, struct hci_dev *hdev,
5814 void *data, u16 data_len)
5815{
5816 struct mgmt_cp_read_local_oob_ext_data *cp = data;
5817 struct mgmt_rp_read_local_oob_ext_data *rp;
5818 size_t rp_len;
5819 u16 eir_len;
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005820 u8 status, flags, role, addr[7], hash[16], rand[16];
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005821 int err;
5822
5823 BT_DBG("%s", hdev->name);
5824
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005825 if (hdev_is_powered(hdev)) {
5826 switch (cp->type) {
5827 case BIT(BDADDR_BREDR):
5828 status = mgmt_bredr_support(hdev);
5829 if (status)
5830 eir_len = 0;
5831 else
5832 eir_len = 5;
5833 break;
5834 case (BIT(BDADDR_LE_PUBLIC) | BIT(BDADDR_LE_RANDOM)):
5835 status = mgmt_le_support(hdev);
5836 if (status)
5837 eir_len = 0;
5838 else
5839 eir_len = 9 + 3 + 18 + 18 + 3;
5840 break;
5841 default:
5842 status = MGMT_STATUS_INVALID_PARAMS;
5843 eir_len = 0;
5844 break;
5845 }
5846 } else {
5847 status = MGMT_STATUS_NOT_POWERED;
5848 eir_len = 0;
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005849 }
5850
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005851 rp_len = sizeof(*rp) + eir_len;
5852 rp = kmalloc(rp_len, GFP_ATOMIC);
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005853 if (!rp)
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005854 return -ENOMEM;
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005855
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005856 if (status)
5857 goto complete;
5858
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005859 hci_dev_lock(hdev);
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005860
5861 eir_len = 0;
5862 switch (cp->type) {
5863 case BIT(BDADDR_BREDR):
Johan Hedberg40f66c02015-04-07 21:52:22 +03005864 if (hci_dev_test_flag(hdev, HCI_SSP_ENABLED)) {
5865 err = read_local_ssp_oob_req(hdev, sk, cp);
5866 hci_dev_unlock(hdev);
5867 if (!err)
5868 goto done;
5869
5870 status = MGMT_STATUS_FAILED;
5871 goto complete;
5872 } else {
5873 eir_len = eir_append_data(rp->eir, eir_len,
5874 EIR_CLASS_OF_DEV,
5875 hdev->dev_class, 3);
5876 }
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005877 break;
5878 case (BIT(BDADDR_LE_PUBLIC) | BIT(BDADDR_LE_RANDOM)):
Marcel Holtmann5082a592015-03-16 12:39:00 -07005879 if (hci_dev_test_flag(hdev, HCI_SC_ENABLED) &&
5880 smp_generate_oob(hdev, hash, rand) < 0) {
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005881 hci_dev_unlock(hdev);
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005882 status = MGMT_STATUS_FAILED;
5883 goto complete;
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005884 }
5885
Marcel Holtmanne2135682015-04-02 12:00:58 -07005886 /* This should return the active RPA, but since the RPA
5887 * is only programmed on demand, it is really hard to fill
5888 * this in at the moment. For now disallow retrieving
5889 * local out-of-band data when privacy is in use.
5890 *
5891 * Returning the identity address will not help here since
5892 * pairing happens before the identity resolving key is
5893 * known and thus the connection establishment happens
5894 * based on the RPA and not the identity address.
5895 */
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005896 if (hci_dev_test_flag(hdev, HCI_PRIVACY)) {
Marcel Holtmanne2135682015-04-02 12:00:58 -07005897 hci_dev_unlock(hdev);
5898 status = MGMT_STATUS_REJECTED;
5899 goto complete;
5900 }
5901
5902 if (hci_dev_test_flag(hdev, HCI_FORCE_STATIC_ADDR) ||
5903 !bacmp(&hdev->bdaddr, BDADDR_ANY) ||
5904 (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) &&
5905 bacmp(&hdev->static_addr, BDADDR_ANY))) {
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005906 memcpy(addr, &hdev->static_addr, 6);
5907 addr[6] = 0x01;
5908 } else {
5909 memcpy(addr, &hdev->bdaddr, 6);
5910 addr[6] = 0x00;
5911 }
5912
5913 eir_len = eir_append_data(rp->eir, eir_len, EIR_LE_BDADDR,
5914 addr, sizeof(addr));
5915
5916 if (hci_dev_test_flag(hdev, HCI_ADVERTISING))
5917 role = 0x02;
5918 else
5919 role = 0x01;
5920
5921 eir_len = eir_append_data(rp->eir, eir_len, EIR_LE_ROLE,
5922 &role, sizeof(role));
5923
Marcel Holtmann5082a592015-03-16 12:39:00 -07005924 if (hci_dev_test_flag(hdev, HCI_SC_ENABLED)) {
5925 eir_len = eir_append_data(rp->eir, eir_len,
5926 EIR_LE_SC_CONFIRM,
5927 hash, sizeof(hash));
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005928
Marcel Holtmann5082a592015-03-16 12:39:00 -07005929 eir_len = eir_append_data(rp->eir, eir_len,
5930 EIR_LE_SC_RANDOM,
5931 rand, sizeof(rand));
5932 }
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005933
Johan Hedbergf2252572015-11-18 12:49:20 +02005934 flags = mgmt_get_adv_discov_flags(hdev);
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005935
5936 if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED))
5937 flags |= LE_AD_NO_BREDR;
5938
5939 eir_len = eir_append_data(rp->eir, eir_len, EIR_FLAGS,
5940 &flags, sizeof(flags));
5941 break;
5942 }
5943
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005944 hci_dev_unlock(hdev);
5945
Marcel Holtmann72000df2015-03-16 16:11:21 -07005946 hci_sock_set_flag(sk, HCI_MGMT_OOB_DATA_EVENTS);
5947
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005948 status = MGMT_STATUS_SUCCESS;
5949
5950complete:
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005951 rp->type = cp->type;
5952 rp->eir_len = cpu_to_le16(eir_len);
5953
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005954 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_EXT_DATA,
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005955 status, rp, sizeof(*rp) + eir_len);
5956 if (err < 0 || status)
Marcel Holtmann72000df2015-03-16 16:11:21 -07005957 goto done;
5958
5959 err = mgmt_limited_event(MGMT_EV_LOCAL_OOB_DATA_UPDATED, hdev,
5960 rp, sizeof(*rp) + eir_len,
5961 HCI_MGMT_OOB_DATA_EVENTS, sk);
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005962
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005963done:
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005964 kfree(rp);
5965
5966 return err;
5967}
5968
Arman Uguray089fa8c2015-03-25 18:53:45 -07005969static u32 get_supported_adv_flags(struct hci_dev *hdev)
5970{
5971 u32 flags = 0;
5972
5973 flags |= MGMT_ADV_FLAG_CONNECTABLE;
5974 flags |= MGMT_ADV_FLAG_DISCOV;
5975 flags |= MGMT_ADV_FLAG_LIMITED_DISCOV;
5976 flags |= MGMT_ADV_FLAG_MANAGED_FLAGS;
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02005977 flags |= MGMT_ADV_FLAG_APPEARANCE;
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02005978 flags |= MGMT_ADV_FLAG_LOCAL_NAME;
Arman Uguray089fa8c2015-03-25 18:53:45 -07005979
5980 if (hdev->adv_tx_power != HCI_TX_POWER_INVALID)
5981 flags |= MGMT_ADV_FLAG_TX_POWER;
5982
5983 return flags;
5984}
5985
Marcel Holtmannd3d53052015-03-14 20:53:25 -07005986static int read_adv_features(struct sock *sk, struct hci_dev *hdev,
5987 void *data, u16 data_len)
5988{
5989 struct mgmt_rp_read_adv_features *rp;
5990 size_t rp_len;
Johan Hedberg02c04af2015-11-26 12:15:58 +02005991 int err;
Florian Grandel286e0c82015-06-18 03:16:38 +02005992 struct adv_info *adv_instance;
Arman Uguray089fa8c2015-03-25 18:53:45 -07005993 u32 supported_flags;
Johan Hedberg02c04af2015-11-26 12:15:58 +02005994 u8 *instance;
Marcel Holtmannd3d53052015-03-14 20:53:25 -07005995
5996 BT_DBG("%s", hdev->name);
5997
Arman Uguray089fa8c2015-03-25 18:53:45 -07005998 if (!lmp_le_capable(hdev))
5999 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_ADV_FEATURES,
6000 MGMT_STATUS_REJECTED);
6001
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006002 hci_dev_lock(hdev);
6003
Johan Hedberg02c04af2015-11-26 12:15:58 +02006004 rp_len = sizeof(*rp) + hdev->adv_instance_cnt;
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006005 rp = kmalloc(rp_len, GFP_ATOMIC);
6006 if (!rp) {
6007 hci_dev_unlock(hdev);
6008 return -ENOMEM;
6009 }
6010
Arman Uguray089fa8c2015-03-25 18:53:45 -07006011 supported_flags = get_supported_adv_flags(hdev);
6012
6013 rp->supported_flags = cpu_to_le32(supported_flags);
Marcel Holtmanndc5d82a2015-03-19 17:22:25 -07006014 rp->max_adv_data_len = HCI_MAX_AD_LENGTH;
6015 rp->max_scan_rsp_len = HCI_MAX_AD_LENGTH;
Florian Grandeld2609b32015-06-18 03:16:34 +02006016 rp->max_instances = HCI_MAX_ADV_INSTANCES;
Johan Hedberg02c04af2015-11-26 12:15:58 +02006017 rp->num_instances = hdev->adv_instance_cnt;
Arman Uguray24b4f382015-03-23 15:57:12 -07006018
Johan Hedberg02c04af2015-11-26 12:15:58 +02006019 instance = rp->instance;
6020 list_for_each_entry(adv_instance, &hdev->adv_instances, list) {
6021 *instance = adv_instance->instance;
6022 instance++;
Arman Uguray24b4f382015-03-23 15:57:12 -07006023 }
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006024
6025 hci_dev_unlock(hdev);
6026
6027 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_READ_ADV_FEATURES,
6028 MGMT_STATUS_SUCCESS, rp, rp_len);
6029
6030 kfree(rp);
6031
6032 return err;
6033}
6034
Szymon Janc2bb368702016-09-18 12:50:05 +02006035static u8 tlv_data_max_len(u32 adv_flags, bool is_adv_data)
Arman Uguray24b4f382015-03-23 15:57:12 -07006036{
Arman Uguray4117ed72015-03-23 15:57:14 -07006037 u8 max_len = HCI_MAX_AD_LENGTH;
Arman Uguray24b4f382015-03-23 15:57:12 -07006038
Marcel Holtmann31a32482015-11-19 16:16:42 +01006039 if (is_adv_data) {
6040 if (adv_flags & (MGMT_ADV_FLAG_DISCOV |
6041 MGMT_ADV_FLAG_LIMITED_DISCOV |
Szymon Janc2bb368702016-09-18 12:50:05 +02006042 MGMT_ADV_FLAG_MANAGED_FLAGS))
Marcel Holtmann31a32482015-11-19 16:16:42 +01006043 max_len -= 3;
Arman Uguray24b4f382015-03-23 15:57:12 -07006044
Szymon Janc2bb368702016-09-18 12:50:05 +02006045 if (adv_flags & MGMT_ADV_FLAG_TX_POWER)
Marcel Holtmann31a32482015-11-19 16:16:42 +01006046 max_len -= 3;
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02006047 } else {
6048 /* at least 1 byte of name should fit in */
6049 if (adv_flags & MGMT_ADV_FLAG_LOCAL_NAME)
6050 max_len -= 3;
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02006051
Szymon Janc2bb368702016-09-18 12:50:05 +02006052 if (adv_flags & (MGMT_ADV_FLAG_APPEARANCE))
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02006053 max_len -= 4;
Arman Uguray5507e352015-03-25 18:53:44 -07006054 }
6055
Szymon Janc2bb368702016-09-18 12:50:05 +02006056 return max_len;
6057}
6058
6059static bool flags_managed(u32 adv_flags)
6060{
6061 return adv_flags & (MGMT_ADV_FLAG_DISCOV |
6062 MGMT_ADV_FLAG_LIMITED_DISCOV |
6063 MGMT_ADV_FLAG_MANAGED_FLAGS);
6064}
6065
6066static bool tx_power_managed(u32 adv_flags)
6067{
6068 return adv_flags & MGMT_ADV_FLAG_TX_POWER;
6069}
6070
6071static bool name_managed(u32 adv_flags)
6072{
6073 return adv_flags & MGMT_ADV_FLAG_LOCAL_NAME;
6074}
6075
6076static bool appearance_managed(u32 adv_flags)
6077{
6078 return adv_flags & MGMT_ADV_FLAG_APPEARANCE;
6079}
6080
6081static bool tlv_data_is_valid(u32 adv_flags, u8 *data, u8 len, bool is_adv_data)
6082{
6083 int i, cur_len;
6084 u8 max_len;
6085
6086 max_len = tlv_data_max_len(adv_flags, is_adv_data);
6087
Arman Uguray4117ed72015-03-23 15:57:14 -07006088 if (len > max_len)
Arman Uguray24b4f382015-03-23 15:57:12 -07006089 return false;
6090
Arman Uguray4117ed72015-03-23 15:57:14 -07006091 /* Make sure that the data is correctly formatted. */
6092 for (i = 0, cur_len = 0; i < len; i += (cur_len + 1)) {
6093 cur_len = data[i];
Arman Uguray24b4f382015-03-23 15:57:12 -07006094
Szymon Janc9c9db782016-09-18 12:50:06 +02006095 if (data[i + 1] == EIR_FLAGS &&
6096 (!is_adv_data || flags_managed(adv_flags)))
Arman Ugurayb44133f2015-03-25 18:53:41 -07006097 return false;
6098
Szymon Janc2bb368702016-09-18 12:50:05 +02006099 if (data[i + 1] == EIR_TX_POWER && tx_power_managed(adv_flags))
6100 return false;
6101
6102 if (data[i + 1] == EIR_NAME_COMPLETE && name_managed(adv_flags))
6103 return false;
6104
6105 if (data[i + 1] == EIR_NAME_SHORT && name_managed(adv_flags))
6106 return false;
6107
6108 if (data[i + 1] == EIR_APPEARANCE &&
6109 appearance_managed(adv_flags))
Arman Uguray5507e352015-03-25 18:53:44 -07006110 return false;
6111
Arman Uguray24b4f382015-03-23 15:57:12 -07006112 /* If the current field length would exceed the total data
6113 * length, then it's invalid.
6114 */
Arman Uguray4117ed72015-03-23 15:57:14 -07006115 if (i + cur_len >= len)
Arman Uguray24b4f382015-03-23 15:57:12 -07006116 return false;
6117 }
6118
6119 return true;
6120}
6121
Arman Uguray24b4f382015-03-23 15:57:12 -07006122static void add_advertising_complete(struct hci_dev *hdev, u8 status,
6123 u16 opcode)
6124{
6125 struct mgmt_pending_cmd *cmd;
Florian Grandelfffd38b2015-06-18 03:16:47 +02006126 struct mgmt_cp_add_advertising *cp;
Arman Uguray24b4f382015-03-23 15:57:12 -07006127 struct mgmt_rp_add_advertising rp;
Florian Grandelfffd38b2015-06-18 03:16:47 +02006128 struct adv_info *adv_instance, *n;
6129 u8 instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006130
6131 BT_DBG("status %d", status);
6132
6133 hci_dev_lock(hdev);
6134
6135 cmd = pending_find(MGMT_OP_ADD_ADVERTISING, hdev);
6136
Florian Grandelfffd38b2015-06-18 03:16:47 +02006137 list_for_each_entry_safe(adv_instance, n, &hdev->adv_instances, list) {
6138 if (!adv_instance->pending)
6139 continue;
6140
6141 if (!status) {
6142 adv_instance->pending = false;
6143 continue;
6144 }
6145
6146 instance = adv_instance->instance;
6147
6148 if (hdev->cur_adv_instance == instance)
6149 cancel_adv_timeout(hdev);
6150
6151 hci_remove_adv_instance(hdev, instance);
Johan Hedbergf2252572015-11-18 12:49:20 +02006152 mgmt_advertising_removed(cmd ? cmd->sk : NULL, hdev, instance);
Arman Uguray24b4f382015-03-23 15:57:12 -07006153 }
6154
6155 if (!cmd)
6156 goto unlock;
6157
Florian Grandelfffd38b2015-06-18 03:16:47 +02006158 cp = cmd->param;
6159 rp.instance = cp->instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006160
6161 if (status)
6162 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode,
6163 mgmt_status(status));
6164 else
6165 mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode,
6166 mgmt_status(status), &rp, sizeof(rp));
6167
6168 mgmt_pending_remove(cmd);
6169
6170unlock:
6171 hci_dev_unlock(hdev);
6172}
6173
6174static int add_advertising(struct sock *sk, struct hci_dev *hdev,
6175 void *data, u16 data_len)
6176{
6177 struct mgmt_cp_add_advertising *cp = data;
6178 struct mgmt_rp_add_advertising rp;
6179 u32 flags;
Arman Uguray089fa8c2015-03-25 18:53:45 -07006180 u32 supported_flags;
Arman Uguray24b4f382015-03-23 15:57:12 -07006181 u8 status;
Florian Grandelfffd38b2015-06-18 03:16:47 +02006182 u16 timeout, duration;
6183 unsigned int prev_instance_cnt = hdev->adv_instance_cnt;
6184 u8 schedule_instance = 0;
6185 struct adv_info *next_instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006186 int err;
6187 struct mgmt_pending_cmd *cmd;
6188 struct hci_request req;
6189
6190 BT_DBG("%s", hdev->name);
6191
6192 status = mgmt_le_support(hdev);
6193 if (status)
6194 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6195 status);
6196
Marcel Holtmannceff86a2015-11-19 16:16:41 +01006197 if (cp->instance < 1 || cp->instance > HCI_MAX_ADV_INSTANCES)
6198 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6199 MGMT_STATUS_INVALID_PARAMS);
6200
Johan Hedberg6a0e7802016-03-11 09:56:33 +02006201 if (data_len != sizeof(*cp) + cp->adv_data_len + cp->scan_rsp_len)
6202 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6203 MGMT_STATUS_INVALID_PARAMS);
6204
Arman Uguray24b4f382015-03-23 15:57:12 -07006205 flags = __le32_to_cpu(cp->flags);
Arman Uguray912098a2015-03-23 15:57:15 -07006206 timeout = __le16_to_cpu(cp->timeout);
Florian Grandelfffd38b2015-06-18 03:16:47 +02006207 duration = __le16_to_cpu(cp->duration);
Arman Uguray24b4f382015-03-23 15:57:12 -07006208
Florian Grandelfffd38b2015-06-18 03:16:47 +02006209 /* The current implementation only supports a subset of the specified
6210 * flags.
Arman Uguray089fa8c2015-03-25 18:53:45 -07006211 */
6212 supported_flags = get_supported_adv_flags(hdev);
Florian Grandelfffd38b2015-06-18 03:16:47 +02006213 if (flags & ~supported_flags)
Arman Uguray24b4f382015-03-23 15:57:12 -07006214 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6215 MGMT_STATUS_INVALID_PARAMS);
6216
6217 hci_dev_lock(hdev);
6218
Arman Uguray912098a2015-03-23 15:57:15 -07006219 if (timeout && !hdev_is_powered(hdev)) {
6220 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6221 MGMT_STATUS_REJECTED);
6222 goto unlock;
6223 }
6224
Arman Uguray24b4f382015-03-23 15:57:12 -07006225 if (pending_find(MGMT_OP_ADD_ADVERTISING, hdev) ||
Arman Ugurayda9293352015-03-23 15:57:13 -07006226 pending_find(MGMT_OP_REMOVE_ADVERTISING, hdev) ||
Arman Uguray24b4f382015-03-23 15:57:12 -07006227 pending_find(MGMT_OP_SET_LE, hdev)) {
6228 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6229 MGMT_STATUS_BUSY);
6230 goto unlock;
6231 }
6232
Szymon Janc5e2c59e2016-09-18 12:50:04 +02006233 if (!tlv_data_is_valid(flags, cp->data, cp->adv_data_len, true) ||
6234 !tlv_data_is_valid(flags, cp->data + cp->adv_data_len,
Arman Ugurayb44133f2015-03-25 18:53:41 -07006235 cp->scan_rsp_len, false)) {
Arman Uguray24b4f382015-03-23 15:57:12 -07006236 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6237 MGMT_STATUS_INVALID_PARAMS);
6238 goto unlock;
6239 }
6240
Florian Grandelfffd38b2015-06-18 03:16:47 +02006241 err = hci_add_adv_instance(hdev, cp->instance, flags,
6242 cp->adv_data_len, cp->data,
6243 cp->scan_rsp_len,
6244 cp->data + cp->adv_data_len,
6245 timeout, duration);
6246 if (err < 0) {
6247 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6248 MGMT_STATUS_FAILED);
6249 goto unlock;
6250 }
Arman Uguray24b4f382015-03-23 15:57:12 -07006251
Florian Grandelfffd38b2015-06-18 03:16:47 +02006252 /* Only trigger an advertising added event if a new instance was
6253 * actually added.
6254 */
6255 if (hdev->adv_instance_cnt > prev_instance_cnt)
Johan Hedbergf2252572015-11-18 12:49:20 +02006256 mgmt_advertising_added(sk, hdev, cp->instance);
Arman Uguray24b4f382015-03-23 15:57:12 -07006257
Florian Grandelfffd38b2015-06-18 03:16:47 +02006258 if (hdev->cur_adv_instance == cp->instance) {
6259 /* If the currently advertised instance is being changed then
6260 * cancel the current advertising and schedule the next
6261 * instance. If there is only one instance then the overridden
6262 * advertising data will be visible right away.
6263 */
6264 cancel_adv_timeout(hdev);
Arman Uguray912098a2015-03-23 15:57:15 -07006265
Florian Grandelfffd38b2015-06-18 03:16:47 +02006266 next_instance = hci_get_next_instance(hdev, cp->instance);
6267 if (next_instance)
6268 schedule_instance = next_instance->instance;
6269 } else if (!hdev->adv_instance_timeout) {
6270 /* Immediately advertise the new instance if no other
6271 * instance is currently being advertised.
6272 */
6273 schedule_instance = cp->instance;
6274 }
Arman Uguray912098a2015-03-23 15:57:15 -07006275
Florian Grandelfffd38b2015-06-18 03:16:47 +02006276 /* If the HCI_ADVERTISING flag is set or the device isn't powered or
6277 * there is no instance to be advertised then we have no HCI
6278 * communication to make. Simply return.
Arman Uguray24b4f382015-03-23 15:57:12 -07006279 */
6280 if (!hdev_is_powered(hdev) ||
Florian Grandelfffd38b2015-06-18 03:16:47 +02006281 hci_dev_test_flag(hdev, HCI_ADVERTISING) ||
6282 !schedule_instance) {
6283 rp.instance = cp->instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006284 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6285 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
6286 goto unlock;
6287 }
6288
6289 /* We're good to go, update advertising data, parameters, and start
6290 * advertising.
6291 */
6292 cmd = mgmt_pending_add(sk, MGMT_OP_ADD_ADVERTISING, hdev, data,
6293 data_len);
6294 if (!cmd) {
6295 err = -ENOMEM;
6296 goto unlock;
6297 }
6298
6299 hci_req_init(&req, hdev);
6300
Johan Hedbergf2252572015-11-18 12:49:20 +02006301 err = __hci_req_schedule_adv_instance(&req, schedule_instance, true);
Arman Uguray24b4f382015-03-23 15:57:12 -07006302
Florian Grandelfffd38b2015-06-18 03:16:47 +02006303 if (!err)
6304 err = hci_req_run(&req, add_advertising_complete);
6305
Arman Uguray24b4f382015-03-23 15:57:12 -07006306 if (err < 0)
6307 mgmt_pending_remove(cmd);
6308
6309unlock:
6310 hci_dev_unlock(hdev);
6311
6312 return err;
6313}
6314
Arman Ugurayda9293352015-03-23 15:57:13 -07006315static void remove_advertising_complete(struct hci_dev *hdev, u8 status,
6316 u16 opcode)
6317{
6318 struct mgmt_pending_cmd *cmd;
Florian Grandel01948332015-06-18 03:16:48 +02006319 struct mgmt_cp_remove_advertising *cp;
Arman Ugurayda9293352015-03-23 15:57:13 -07006320 struct mgmt_rp_remove_advertising rp;
6321
6322 BT_DBG("status %d", status);
6323
6324 hci_dev_lock(hdev);
6325
6326 /* A failure status here only means that we failed to disable
6327 * advertising. Otherwise, the advertising instance has been removed,
6328 * so report success.
6329 */
6330 cmd = pending_find(MGMT_OP_REMOVE_ADVERTISING, hdev);
6331 if (!cmd)
6332 goto unlock;
6333
Florian Grandel01948332015-06-18 03:16:48 +02006334 cp = cmd->param;
6335 rp.instance = cp->instance;
Arman Ugurayda9293352015-03-23 15:57:13 -07006336
6337 mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, MGMT_STATUS_SUCCESS,
6338 &rp, sizeof(rp));
6339 mgmt_pending_remove(cmd);
6340
6341unlock:
6342 hci_dev_unlock(hdev);
6343}
6344
6345static int remove_advertising(struct sock *sk, struct hci_dev *hdev,
6346 void *data, u16 data_len)
6347{
6348 struct mgmt_cp_remove_advertising *cp = data;
6349 struct mgmt_rp_remove_advertising rp;
Arman Ugurayda9293352015-03-23 15:57:13 -07006350 struct mgmt_pending_cmd *cmd;
6351 struct hci_request req;
Johan Hedberg952497b2015-06-18 21:05:31 +03006352 int err;
Arman Ugurayda9293352015-03-23 15:57:13 -07006353
6354 BT_DBG("%s", hdev->name);
6355
Arman Ugurayda9293352015-03-23 15:57:13 -07006356 hci_dev_lock(hdev);
6357
Johan Hedberg952497b2015-06-18 21:05:31 +03006358 if (cp->instance && !hci_find_adv_instance(hdev, cp->instance)) {
Florian Grandel01948332015-06-18 03:16:48 +02006359 err = mgmt_cmd_status(sk, hdev->id,
6360 MGMT_OP_REMOVE_ADVERTISING,
6361 MGMT_STATUS_INVALID_PARAMS);
6362 goto unlock;
6363 }
6364
Arman Ugurayda9293352015-03-23 15:57:13 -07006365 if (pending_find(MGMT_OP_ADD_ADVERTISING, hdev) ||
6366 pending_find(MGMT_OP_REMOVE_ADVERTISING, hdev) ||
6367 pending_find(MGMT_OP_SET_LE, hdev)) {
6368 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_REMOVE_ADVERTISING,
6369 MGMT_STATUS_BUSY);
6370 goto unlock;
6371 }
6372
Johan Hedberg17fd08f2015-11-26 12:15:59 +02006373 if (list_empty(&hdev->adv_instances)) {
Arman Ugurayda9293352015-03-23 15:57:13 -07006374 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_REMOVE_ADVERTISING,
6375 MGMT_STATUS_INVALID_PARAMS);
6376 goto unlock;
6377 }
6378
Florian Grandel01948332015-06-18 03:16:48 +02006379 hci_req_init(&req, hdev);
Arman Uguray912098a2015-03-23 15:57:15 -07006380
Johan Hedberg37d3a1f2016-08-28 20:53:34 +03006381 hci_req_clear_adv_instance(hdev, sk, &req, cp->instance, true);
Arman Ugurayda9293352015-03-23 15:57:13 -07006382
Florian Grandel01948332015-06-18 03:16:48 +02006383 if (list_empty(&hdev->adv_instances))
Johan Hedbergf2252572015-11-18 12:49:20 +02006384 __hci_req_disable_advertising(&req);
Arman Ugurayda9293352015-03-23 15:57:13 -07006385
Florian Grandel01948332015-06-18 03:16:48 +02006386 /* If no HCI commands have been collected so far or the HCI_ADVERTISING
6387 * flag is set or the device isn't powered then we have no HCI
6388 * communication to make. Simply return.
Arman Ugurayda9293352015-03-23 15:57:13 -07006389 */
Florian Grandel01948332015-06-18 03:16:48 +02006390 if (skb_queue_empty(&req.cmd_q) ||
6391 !hdev_is_powered(hdev) ||
Arman Ugurayda9293352015-03-23 15:57:13 -07006392 hci_dev_test_flag(hdev, HCI_ADVERTISING)) {
Florian Grandel01948332015-06-18 03:16:48 +02006393 rp.instance = cp->instance;
Arman Ugurayda9293352015-03-23 15:57:13 -07006394 err = mgmt_cmd_complete(sk, hdev->id,
6395 MGMT_OP_REMOVE_ADVERTISING,
6396 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
6397 goto unlock;
6398 }
6399
6400 cmd = mgmt_pending_add(sk, MGMT_OP_REMOVE_ADVERTISING, hdev, data,
6401 data_len);
6402 if (!cmd) {
6403 err = -ENOMEM;
6404 goto unlock;
6405 }
6406
Arman Ugurayda9293352015-03-23 15:57:13 -07006407 err = hci_req_run(&req, remove_advertising_complete);
6408 if (err < 0)
6409 mgmt_pending_remove(cmd);
6410
6411unlock:
6412 hci_dev_unlock(hdev);
6413
6414 return err;
6415}
6416
Marcel Holtmann40b25fe2015-11-19 16:16:43 +01006417static int get_adv_size_info(struct sock *sk, struct hci_dev *hdev,
6418 void *data, u16 data_len)
6419{
6420 struct mgmt_cp_get_adv_size_info *cp = data;
6421 struct mgmt_rp_get_adv_size_info rp;
6422 u32 flags, supported_flags;
6423 int err;
6424
6425 BT_DBG("%s", hdev->name);
6426
6427 if (!lmp_le_capable(hdev))
6428 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6429 MGMT_STATUS_REJECTED);
6430
6431 if (cp->instance < 1 || cp->instance > HCI_MAX_ADV_INSTANCES)
6432 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6433 MGMT_STATUS_INVALID_PARAMS);
6434
6435 flags = __le32_to_cpu(cp->flags);
6436
6437 /* The current implementation only supports a subset of the specified
6438 * flags.
6439 */
6440 supported_flags = get_supported_adv_flags(hdev);
6441 if (flags & ~supported_flags)
6442 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6443 MGMT_STATUS_INVALID_PARAMS);
6444
6445 rp.instance = cp->instance;
6446 rp.flags = cp->flags;
6447 rp.max_adv_data_len = tlv_data_max_len(flags, true);
6448 rp.max_scan_rsp_len = tlv_data_max_len(flags, false);
6449
6450 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6451 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
6452
6453 return err;
6454}
6455
Johan Hedberg6d785aa32015-03-06 21:08:51 +02006456static const struct hci_mgmt_handler mgmt_handlers[] = {
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02006457 { NULL }, /* 0x0000 (no command) */
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006458 { read_version, MGMT_READ_VERSION_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006459 HCI_MGMT_NO_HDEV |
6460 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006461 { read_commands, MGMT_READ_COMMANDS_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006462 HCI_MGMT_NO_HDEV |
6463 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006464 { read_index_list, MGMT_READ_INDEX_LIST_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006465 HCI_MGMT_NO_HDEV |
6466 HCI_MGMT_UNTRUSTED },
6467 { read_controller_info, MGMT_READ_INFO_SIZE,
6468 HCI_MGMT_UNTRUSTED },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006469 { set_powered, MGMT_SETTING_SIZE },
6470 { set_discoverable, MGMT_SET_DISCOVERABLE_SIZE },
6471 { set_connectable, MGMT_SETTING_SIZE },
6472 { set_fast_connectable, MGMT_SETTING_SIZE },
6473 { set_bondable, MGMT_SETTING_SIZE },
6474 { set_link_security, MGMT_SETTING_SIZE },
6475 { set_ssp, MGMT_SETTING_SIZE },
6476 { set_hs, MGMT_SETTING_SIZE },
6477 { set_le, MGMT_SETTING_SIZE },
6478 { set_dev_class, MGMT_SET_DEV_CLASS_SIZE },
6479 { set_local_name, MGMT_SET_LOCAL_NAME_SIZE },
6480 { add_uuid, MGMT_ADD_UUID_SIZE },
6481 { remove_uuid, MGMT_REMOVE_UUID_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006482 { load_link_keys, MGMT_LOAD_LINK_KEYS_SIZE,
6483 HCI_MGMT_VAR_LEN },
6484 { load_long_term_keys, MGMT_LOAD_LONG_TERM_KEYS_SIZE,
6485 HCI_MGMT_VAR_LEN },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006486 { disconnect, MGMT_DISCONNECT_SIZE },
6487 { get_connections, MGMT_GET_CONNECTIONS_SIZE },
6488 { pin_code_reply, MGMT_PIN_CODE_REPLY_SIZE },
6489 { pin_code_neg_reply, MGMT_PIN_CODE_NEG_REPLY_SIZE },
6490 { set_io_capability, MGMT_SET_IO_CAPABILITY_SIZE },
6491 { pair_device, MGMT_PAIR_DEVICE_SIZE },
6492 { cancel_pair_device, MGMT_CANCEL_PAIR_DEVICE_SIZE },
6493 { unpair_device, MGMT_UNPAIR_DEVICE_SIZE },
6494 { user_confirm_reply, MGMT_USER_CONFIRM_REPLY_SIZE },
6495 { user_confirm_neg_reply, MGMT_USER_CONFIRM_NEG_REPLY_SIZE },
6496 { user_passkey_reply, MGMT_USER_PASSKEY_REPLY_SIZE },
6497 { user_passkey_neg_reply, MGMT_USER_PASSKEY_NEG_REPLY_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006498 { read_local_oob_data, MGMT_READ_LOCAL_OOB_DATA_SIZE },
6499 { add_remote_oob_data, MGMT_ADD_REMOTE_OOB_DATA_SIZE,
6500 HCI_MGMT_VAR_LEN },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006501 { remove_remote_oob_data, MGMT_REMOVE_REMOTE_OOB_DATA_SIZE },
6502 { start_discovery, MGMT_START_DISCOVERY_SIZE },
6503 { stop_discovery, MGMT_STOP_DISCOVERY_SIZE },
6504 { confirm_name, MGMT_CONFIRM_NAME_SIZE },
6505 { block_device, MGMT_BLOCK_DEVICE_SIZE },
6506 { unblock_device, MGMT_UNBLOCK_DEVICE_SIZE },
6507 { set_device_id, MGMT_SET_DEVICE_ID_SIZE },
6508 { set_advertising, MGMT_SETTING_SIZE },
6509 { set_bredr, MGMT_SETTING_SIZE },
6510 { set_static_address, MGMT_SET_STATIC_ADDRESS_SIZE },
6511 { set_scan_params, MGMT_SET_SCAN_PARAMS_SIZE },
6512 { set_secure_conn, MGMT_SETTING_SIZE },
6513 { set_debug_keys, MGMT_SETTING_SIZE },
6514 { set_privacy, MGMT_SET_PRIVACY_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006515 { load_irks, MGMT_LOAD_IRKS_SIZE,
6516 HCI_MGMT_VAR_LEN },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006517 { get_conn_info, MGMT_GET_CONN_INFO_SIZE },
6518 { get_clock_info, MGMT_GET_CLOCK_INFO_SIZE },
6519 { add_device, MGMT_ADD_DEVICE_SIZE },
6520 { remove_device, MGMT_REMOVE_DEVICE_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006521 { load_conn_param, MGMT_LOAD_CONN_PARAM_SIZE,
6522 HCI_MGMT_VAR_LEN },
6523 { read_unconf_index_list, MGMT_READ_UNCONF_INDEX_LIST_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006524 HCI_MGMT_NO_HDEV |
6525 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006526 { read_config_info, MGMT_READ_CONFIG_INFO_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006527 HCI_MGMT_UNCONFIGURED |
6528 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006529 { set_external_config, MGMT_SET_EXTERNAL_CONFIG_SIZE,
6530 HCI_MGMT_UNCONFIGURED },
6531 { set_public_address, MGMT_SET_PUBLIC_ADDRESS_SIZE,
6532 HCI_MGMT_UNCONFIGURED },
6533 { start_service_discovery, MGMT_START_SERVICE_DISCOVERY_SIZE,
6534 HCI_MGMT_VAR_LEN },
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07006535 { read_local_oob_ext_data, MGMT_READ_LOCAL_OOB_EXT_DATA_SIZE },
Marcel Holtmann96f14742015-03-14 19:27:57 -07006536 { read_ext_index_list, MGMT_READ_EXT_INDEX_LIST_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006537 HCI_MGMT_NO_HDEV |
6538 HCI_MGMT_UNTRUSTED },
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006539 { read_adv_features, MGMT_READ_ADV_FEATURES_SIZE },
Arman Uguray24b4f382015-03-23 15:57:12 -07006540 { add_advertising, MGMT_ADD_ADVERTISING_SIZE,
6541 HCI_MGMT_VAR_LEN },
Arman Ugurayda9293352015-03-23 15:57:13 -07006542 { remove_advertising, MGMT_REMOVE_ADVERTISING_SIZE },
Marcel Holtmann40b25fe2015-11-19 16:16:43 +01006543 { get_adv_size_info, MGMT_GET_ADV_SIZE_INFO_SIZE },
Johan Hedberg78b781c2016-01-05 13:19:32 +02006544 { start_limited_discovery, MGMT_START_DISCOVERY_SIZE },
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02006545 { read_ext_controller_info,MGMT_READ_EXT_INFO_SIZE,
6546 HCI_MGMT_UNTRUSTED },
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02006547 { set_appearance, MGMT_SET_APPEARANCE_SIZE },
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02006548};
6549
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07006550void mgmt_index_added(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02006551{
Marcel Holtmannced85542015-03-14 19:27:56 -07006552 struct mgmt_ev_ext_index ev;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03006553
Marcel Holtmann0602a8a2014-07-02 21:30:54 +02006554 if (test_bit(HCI_QUIRK_RAW_DEVICE, &hdev->quirks))
6555 return;
6556
Marcel Holtmannf9207332015-03-14 19:27:55 -07006557 switch (hdev->dev_type) {
Marcel Holtmannca8bee52016-07-05 14:30:14 +02006558 case HCI_PRIMARY:
Marcel Holtmannf9207332015-03-14 19:27:55 -07006559 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) {
6560 mgmt_index_event(MGMT_EV_UNCONF_INDEX_ADDED, hdev,
6561 NULL, 0, HCI_MGMT_UNCONF_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006562 ev.type = 0x01;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006563 } else {
6564 mgmt_index_event(MGMT_EV_INDEX_ADDED, hdev, NULL, 0,
6565 HCI_MGMT_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006566 ev.type = 0x00;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006567 }
6568 break;
Marcel Holtmannced85542015-03-14 19:27:56 -07006569 case HCI_AMP:
6570 ev.type = 0x02;
6571 break;
6572 default:
6573 return;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006574 }
Marcel Holtmannced85542015-03-14 19:27:56 -07006575
6576 ev.bus = hdev->bus;
6577
6578 mgmt_index_event(MGMT_EV_EXT_INDEX_ADDED, hdev, &ev, sizeof(ev),
6579 HCI_MGMT_EXT_INDEX_EVENTS);
Johan Hedbergc71e97b2010-12-13 21:07:07 +02006580}
6581
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07006582void mgmt_index_removed(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02006583{
Marcel Holtmannced85542015-03-14 19:27:56 -07006584 struct mgmt_ev_ext_index ev;
Johan Hedberg5f159032012-03-02 03:13:19 +02006585 u8 status = MGMT_STATUS_INVALID_INDEX;
Johan Hedbergb24752f2011-11-03 14:40:33 +02006586
Marcel Holtmann0602a8a2014-07-02 21:30:54 +02006587 if (test_bit(HCI_QUIRK_RAW_DEVICE, &hdev->quirks))
6588 return;
6589
Marcel Holtmannf9207332015-03-14 19:27:55 -07006590 switch (hdev->dev_type) {
Marcel Holtmannca8bee52016-07-05 14:30:14 +02006591 case HCI_PRIMARY:
Marcel Holtmannf9207332015-03-14 19:27:55 -07006592 mgmt_pending_foreach(0, hdev, cmd_complete_rsp, &status);
Johan Hedbergb24752f2011-11-03 14:40:33 +02006593
Marcel Holtmannf9207332015-03-14 19:27:55 -07006594 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) {
6595 mgmt_index_event(MGMT_EV_UNCONF_INDEX_REMOVED, hdev,
6596 NULL, 0, HCI_MGMT_UNCONF_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006597 ev.type = 0x01;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006598 } else {
6599 mgmt_index_event(MGMT_EV_INDEX_REMOVED, hdev, NULL, 0,
6600 HCI_MGMT_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006601 ev.type = 0x00;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006602 }
6603 break;
Marcel Holtmannced85542015-03-14 19:27:56 -07006604 case HCI_AMP:
6605 ev.type = 0x02;
6606 break;
6607 default:
6608 return;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006609 }
Marcel Holtmannced85542015-03-14 19:27:56 -07006610
6611 ev.bus = hdev->bus;
6612
6613 mgmt_index_event(MGMT_EV_EXT_INDEX_REMOVED, hdev, &ev, sizeof(ev),
6614 HCI_MGMT_EXT_INDEX_EVENTS);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02006615}
6616
Andre Guedes6046dc32014-02-26 20:21:51 -03006617/* This function requires the caller holds hdev->lock */
Johan Hedbergaf02dd42015-11-11 08:11:21 +02006618static void restart_le_actions(struct hci_dev *hdev)
Andre Guedes6046dc32014-02-26 20:21:51 -03006619{
6620 struct hci_conn_params *p;
6621
6622 list_for_each_entry(p, &hdev->le_conn_params, list) {
Johan Hedbergd7347f32014-07-04 12:37:23 +03006623 /* Needed for AUTO_OFF case where might not "really"
6624 * have been powered off.
6625 */
6626 list_del_init(&p->action);
6627
6628 switch (p->auto_connect) {
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02006629 case HCI_AUTO_CONN_DIRECT:
Johan Hedbergd7347f32014-07-04 12:37:23 +03006630 case HCI_AUTO_CONN_ALWAYS:
6631 list_add(&p->action, &hdev->pend_le_conns);
6632 break;
6633 case HCI_AUTO_CONN_REPORT:
6634 list_add(&p->action, &hdev->pend_le_reports);
6635 break;
6636 default:
6637 break;
Marcel Holtmannc83ed192014-07-01 19:28:24 +02006638 }
Andre Guedes6046dc32014-02-26 20:21:51 -03006639 }
6640}
6641
Johan Hedberg2ff13892015-11-25 16:15:44 +02006642void mgmt_power_on(struct hci_dev *hdev, int err)
Johan Hedberg229ab392013-03-15 17:06:53 -05006643{
6644 struct cmd_lookup match = { NULL, hdev };
6645
Johan Hedberg2ff13892015-11-25 16:15:44 +02006646 BT_DBG("err %d", err);
Johan Hedberg229ab392013-03-15 17:06:53 -05006647
Johan Hedberg2ff13892015-11-25 16:15:44 +02006648 hci_dev_lock(hdev);
6649
6650 if (!err) {
Johan Hedbergaf02dd42015-11-11 08:11:21 +02006651 restart_le_actions(hdev);
6652 hci_update_background_scan(hdev);
Marcel Holtmann162a3ba2015-01-14 15:43:11 -08006653 }
6654
Johan Hedberg229ab392013-03-15 17:06:53 -05006655 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
6656
6657 new_settings(hdev, match.sk);
6658
Johan Hedberg229ab392013-03-15 17:06:53 -05006659 if (match.sk)
6660 sock_put(match.sk);
Johan Hedberg2ff13892015-11-25 16:15:44 +02006661
6662 hci_dev_unlock(hdev);
Johan Hedberg229ab392013-03-15 17:06:53 -05006663}
6664
Johan Hedberg2ff13892015-11-25 16:15:44 +02006665void __mgmt_power_off(struct hci_dev *hdev)
Johan Hedberg5add6af2010-12-16 10:00:37 +02006666{
Johan Hedberg76a7f3a2012-02-17 00:34:40 +02006667 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg98459042014-12-12 11:15:21 +02006668 u8 status, zero_cod[] = { 0, 0, 0 };
Johan Hedbergb24752f2011-11-03 14:40:33 +02006669
Johan Hedberg229ab392013-03-15 17:06:53 -05006670 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
Johan Hedberg98459042014-12-12 11:15:21 +02006671
6672 /* If the power off is because of hdev unregistration let
6673 * use the appropriate INVALID_INDEX status. Otherwise use
6674 * NOT_POWERED. We cover both scenarios here since later in
6675 * mgmt_index_removed() any hci_conn callbacks will have already
6676 * been triggered, potentially causing misleading DISCONNECTED
6677 * status responses.
6678 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07006679 if (hci_dev_test_flag(hdev, HCI_UNREGISTER))
Johan Hedberg98459042014-12-12 11:15:21 +02006680 status = MGMT_STATUS_INVALID_INDEX;
6681 else
6682 status = MGMT_STATUS_NOT_POWERED;
6683
6684 mgmt_pending_foreach(0, hdev, cmd_complete_rsp, &status);
Johan Hedberg229ab392013-03-15 17:06:53 -05006685
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02006686 if (memcmp(hdev->dev_class, zero_cod, sizeof(zero_cod)) != 0) {
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02006687 mgmt_limited_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev,
6688 zero_cod, sizeof(zero_cod),
6689 HCI_MGMT_DEV_CLASS_EVENTS, NULL);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02006690 ext_info_changed(hdev, NULL);
6691 }
Johan Hedberg229ab392013-03-15 17:06:53 -05006692
Johan Hedberg2ff13892015-11-25 16:15:44 +02006693 new_settings(hdev, match.sk);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02006694
6695 if (match.sk)
6696 sock_put(match.sk);
Johan Hedberg5add6af2010-12-16 10:00:37 +02006697}
Johan Hedberg73f22f62010-12-29 16:00:25 +02006698
Marcel Holtmann3eec7052013-10-06 23:55:46 -07006699void mgmt_set_powered_failed(struct hci_dev *hdev, int err)
Johan Hedberg96570ff2013-05-29 09:51:29 +03006700{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006701 struct mgmt_pending_cmd *cmd;
Johan Hedberg96570ff2013-05-29 09:51:29 +03006702 u8 status;
6703
Johan Hedberg333ae952015-03-17 13:48:47 +02006704 cmd = pending_find(MGMT_OP_SET_POWERED, hdev);
Johan Hedberg96570ff2013-05-29 09:51:29 +03006705 if (!cmd)
Marcel Holtmann3eec7052013-10-06 23:55:46 -07006706 return;
Johan Hedberg96570ff2013-05-29 09:51:29 +03006707
6708 if (err == -ERFKILL)
6709 status = MGMT_STATUS_RFKILLED;
6710 else
6711 status = MGMT_STATUS_FAILED;
6712
Johan Hedberga69e8372015-03-06 21:08:53 +02006713 mgmt_cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_POWERED, status);
Johan Hedberg96570ff2013-05-29 09:51:29 +03006714
6715 mgmt_pending_remove(cmd);
Johan Hedberg96570ff2013-05-29 09:51:29 +03006716}
6717
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07006718void mgmt_new_link_key(struct hci_dev *hdev, struct link_key *key,
6719 bool persistent)
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006720{
Johan Hedberg86742e12011-11-07 23:13:38 +02006721 struct mgmt_ev_new_link_key ev;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006722
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006723 memset(&ev, 0, sizeof(ev));
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006724
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006725 ev.store_hint = persistent;
Johan Hedbergd753fdc2012-02-17 14:06:34 +02006726 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03006727 ev.key.addr.type = BDADDR_BREDR;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006728 ev.key.type = key->type;
Andrei Emeltchenko9b3b4462012-05-23 11:31:20 +03006729 memcpy(ev.key.val, key->val, HCI_LINK_KEY_SIZE);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006730 ev.key.pin_len = key->pin_len;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006731
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07006732 mgmt_event(MGMT_EV_NEW_LINK_KEY, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006733}
Johan Hedbergf7520542011-01-20 12:34:39 +02006734
Johan Hedbergd7b25452014-05-23 13:19:53 +03006735static u8 mgmt_ltk_type(struct smp_ltk *ltk)
6736{
Johan Hedberg23fb8de2014-05-23 13:15:37 +03006737 switch (ltk->type) {
6738 case SMP_LTK:
6739 case SMP_LTK_SLAVE:
6740 if (ltk->authenticated)
6741 return MGMT_LTK_AUTHENTICATED;
6742 return MGMT_LTK_UNAUTHENTICATED;
6743 case SMP_LTK_P256:
6744 if (ltk->authenticated)
6745 return MGMT_LTK_P256_AUTH;
6746 return MGMT_LTK_P256_UNAUTH;
6747 case SMP_LTK_P256_DEBUG:
6748 return MGMT_LTK_P256_DEBUG;
6749 }
Johan Hedbergd7b25452014-05-23 13:19:53 +03006750
6751 return MGMT_LTK_UNAUTHENTICATED;
6752}
6753
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006754void mgmt_new_ltk(struct hci_dev *hdev, struct smp_ltk *key, bool persistent)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006755{
6756 struct mgmt_ev_new_long_term_key ev;
6757
6758 memset(&ev, 0, sizeof(ev));
6759
Marcel Holtmann5192d302014-02-19 17:11:58 -08006760 /* Devices using resolvable or non-resolvable random addresses
Florian Grandelf72186d2015-05-26 03:31:09 +02006761 * without providing an identity resolving key don't require
Marcel Holtmann5192d302014-02-19 17:11:58 -08006762 * to store long term keys. Their addresses will change the
6763 * next time around.
6764 *
6765 * Only when a remote device provides an identity address
6766 * make sure the long term key is stored. If the remote
6767 * identity is known, the long term keys are internally
6768 * mapped to the identity address. So allow static random
6769 * and public addresses here.
6770 */
Johan Hedbergba74b662014-02-19 14:57:45 +02006771 if (key->bdaddr_type == ADDR_LE_DEV_RANDOM &&
6772 (key->bdaddr.b[5] & 0xc0) != 0xc0)
6773 ev.store_hint = 0x00;
6774 else
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006775 ev.store_hint = persistent;
Johan Hedbergba74b662014-02-19 14:57:45 +02006776
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006777 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03006778 ev.key.addr.type = link_to_bdaddr(LE_LINK, key->bdaddr_type);
Johan Hedbergd7b25452014-05-23 13:19:53 +03006779 ev.key.type = mgmt_ltk_type(key);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006780 ev.key.enc_size = key->enc_size;
6781 ev.key.ediv = key->ediv;
Marcel Holtmannfe39c7b2014-02-27 16:00:28 -08006782 ev.key.rand = key->rand;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006783
Johan Hedberg2ceba532014-06-16 19:25:16 +03006784 if (key->type == SMP_LTK)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006785 ev.key.master = 1;
6786
Johan Hedberg1fc62c52015-06-10 11:11:20 +03006787 /* Make sure we copy only the significant bytes based on the
6788 * encryption key size, and set the rest of the value to zeroes.
6789 */
Jakub Pawlowskicb922052015-08-05 23:16:29 +02006790 memcpy(ev.key.val, key->val, key->enc_size);
Johan Hedberg1fc62c52015-06-10 11:11:20 +03006791 memset(ev.key.val + key->enc_size, 0,
6792 sizeof(ev.key.val) - key->enc_size);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006793
Marcel Holtmann083368f2013-10-15 14:26:29 -07006794 mgmt_event(MGMT_EV_NEW_LONG_TERM_KEY, hdev, &ev, sizeof(ev), NULL);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006795}
6796
Johan Hedbergcad20c22015-10-12 13:36:19 +02006797void mgmt_new_irk(struct hci_dev *hdev, struct smp_irk *irk, bool persistent)
Johan Hedberg95fbac82014-02-19 15:18:31 +02006798{
6799 struct mgmt_ev_new_irk ev;
6800
6801 memset(&ev, 0, sizeof(ev));
6802
Johan Hedbergcad20c22015-10-12 13:36:19 +02006803 ev.store_hint = persistent;
Marcel Holtmannbab6d1e2014-02-19 11:51:54 -08006804
Johan Hedberg95fbac82014-02-19 15:18:31 +02006805 bacpy(&ev.rpa, &irk->rpa);
6806 bacpy(&ev.irk.addr.bdaddr, &irk->bdaddr);
6807 ev.irk.addr.type = link_to_bdaddr(LE_LINK, irk->addr_type);
6808 memcpy(ev.irk.val, irk->val, sizeof(irk->val));
6809
6810 mgmt_event(MGMT_EV_NEW_IRK, hdev, &ev, sizeof(ev), NULL);
6811}
6812
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006813void mgmt_new_csrk(struct hci_dev *hdev, struct smp_csrk *csrk,
6814 bool persistent)
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006815{
6816 struct mgmt_ev_new_csrk ev;
6817
6818 memset(&ev, 0, sizeof(ev));
6819
6820 /* Devices using resolvable or non-resolvable random addresses
Florian Grandelf72186d2015-05-26 03:31:09 +02006821 * without providing an identity resolving key don't require
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006822 * to store signature resolving keys. Their addresses will change
6823 * the next time around.
6824 *
6825 * Only when a remote device provides an identity address
6826 * make sure the signature resolving key is stored. So allow
6827 * static random and public addresses here.
6828 */
6829 if (csrk->bdaddr_type == ADDR_LE_DEV_RANDOM &&
6830 (csrk->bdaddr.b[5] & 0xc0) != 0xc0)
6831 ev.store_hint = 0x00;
6832 else
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006833 ev.store_hint = persistent;
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006834
6835 bacpy(&ev.key.addr.bdaddr, &csrk->bdaddr);
6836 ev.key.addr.type = link_to_bdaddr(LE_LINK, csrk->bdaddr_type);
Johan Hedberg4cd39282015-02-27 10:11:13 +02006837 ev.key.type = csrk->type;
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006838 memcpy(ev.key.val, csrk->val, sizeof(csrk->val));
6839
6840 mgmt_event(MGMT_EV_NEW_CSRK, hdev, &ev, sizeof(ev), NULL);
6841}
6842
Andre Guedesffb5a8272014-07-01 18:10:11 -03006843void mgmt_new_conn_param(struct hci_dev *hdev, bdaddr_t *bdaddr,
Johan Hedbergf4869e22014-07-02 17:37:32 +03006844 u8 bdaddr_type, u8 store_hint, u16 min_interval,
6845 u16 max_interval, u16 latency, u16 timeout)
Andre Guedesffb5a8272014-07-01 18:10:11 -03006846{
6847 struct mgmt_ev_new_conn_param ev;
6848
Johan Hedbergc103aea2014-07-02 17:37:34 +03006849 if (!hci_is_identity_address(bdaddr, bdaddr_type))
6850 return;
6851
Andre Guedesffb5a8272014-07-01 18:10:11 -03006852 memset(&ev, 0, sizeof(ev));
6853 bacpy(&ev.addr.bdaddr, bdaddr);
6854 ev.addr.type = link_to_bdaddr(LE_LINK, bdaddr_type);
Johan Hedbergf4869e22014-07-02 17:37:32 +03006855 ev.store_hint = store_hint;
Andre Guedesffb5a8272014-07-01 18:10:11 -03006856 ev.min_interval = cpu_to_le16(min_interval);
6857 ev.max_interval = cpu_to_le16(max_interval);
6858 ev.latency = cpu_to_le16(latency);
6859 ev.timeout = cpu_to_le16(timeout);
6860
6861 mgmt_event(MGMT_EV_NEW_CONN_PARAM, hdev, &ev, sizeof(ev), NULL);
6862}
6863
Alfonso Acosta48ec92f2014-10-07 08:44:10 +00006864void mgmt_device_connected(struct hci_dev *hdev, struct hci_conn *conn,
6865 u32 flags, u8 *name, u8 name_len)
Johan Hedbergf7520542011-01-20 12:34:39 +02006866{
Johan Hedbergb644ba32012-01-17 21:48:47 +02006867 char buf[512];
6868 struct mgmt_ev_device_connected *ev = (void *) buf;
6869 u16 eir_len = 0;
Johan Hedbergf7520542011-01-20 12:34:39 +02006870
Alfonso Acosta48ec92f2014-10-07 08:44:10 +00006871 bacpy(&ev->addr.bdaddr, &conn->dst);
6872 ev->addr.type = link_to_bdaddr(conn->type, conn->dst_type);
Johan Hedbergf7520542011-01-20 12:34:39 +02006873
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02006874 ev->flags = __cpu_to_le32(flags);
Johan Hedberg08c79b62012-02-23 22:31:51 +02006875
Alfonso Acostafd45ada2014-10-07 08:44:11 +00006876 /* We must ensure that the EIR Data fields are ordered and
6877 * unique. Keep it simple for now and avoid the problem by not
6878 * adding any BR/EDR data to the LE adv.
6879 */
6880 if (conn->le_adv_data_len > 0) {
6881 memcpy(&ev->eir[eir_len],
6882 conn->le_adv_data, conn->le_adv_data_len);
6883 eir_len = conn->le_adv_data_len;
6884 } else {
6885 if (name_len > 0)
6886 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE,
6887 name, name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02006888
Alfonso Acostaddbea5c2014-10-07 08:44:12 +00006889 if (memcmp(conn->dev_class, "\0\0\0", 3) != 0)
Alfonso Acostafd45ada2014-10-07 08:44:11 +00006890 eir_len = eir_append_data(ev->eir, eir_len,
6891 EIR_CLASS_OF_DEV,
6892 conn->dev_class, 3);
6893 }
Johan Hedbergb644ba32012-01-17 21:48:47 +02006894
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02006895 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02006896
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07006897 mgmt_event(MGMT_EV_DEVICE_CONNECTED, hdev, buf,
6898 sizeof(*ev) + eir_len, NULL);
Johan Hedbergf7520542011-01-20 12:34:39 +02006899}
6900
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006901static void disconnect_rsp(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedberg8962ee72011-01-20 12:40:27 +02006902{
Johan Hedberg8962ee72011-01-20 12:40:27 +02006903 struct sock **sk = data;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006904
Johan Hedbergf5818c22014-12-05 13:36:02 +02006905 cmd->cmd_complete(cmd, 0);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006906
6907 *sk = cmd->sk;
6908 sock_hold(*sk);
6909
Johan Hedberga664b5b2011-02-19 12:06:02 -03006910 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006911}
6912
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006913static void unpair_device_rsp(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedberga8a1d192011-11-10 15:54:38 +02006914{
Johan Hedbergb1078ad2012-02-09 17:21:16 +02006915 struct hci_dev *hdev = data;
Johan Hedberg124f6e32012-02-09 13:50:12 +02006916 struct mgmt_cp_unpair_device *cp = cmd->param;
Johan Hedberga8a1d192011-11-10 15:54:38 +02006917
Johan Hedbergb1078ad2012-02-09 17:21:16 +02006918 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, cmd->sk);
6919
Johan Hedbergd8b7b1e2014-12-05 13:36:05 +02006920 cmd->cmd_complete(cmd, 0);
Johan Hedberga8a1d192011-11-10 15:54:38 +02006921 mgmt_pending_remove(cmd);
6922}
6923
Johan Hedberg84c61d92014-08-01 11:13:30 +03006924bool mgmt_powering_down(struct hci_dev *hdev)
6925{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006926 struct mgmt_pending_cmd *cmd;
Johan Hedberg84c61d92014-08-01 11:13:30 +03006927 struct mgmt_mode *cp;
6928
Johan Hedberg333ae952015-03-17 13:48:47 +02006929 cmd = pending_find(MGMT_OP_SET_POWERED, hdev);
Johan Hedberg84c61d92014-08-01 11:13:30 +03006930 if (!cmd)
6931 return false;
6932
6933 cp = cmd->param;
6934 if (!cp->val)
6935 return true;
6936
6937 return false;
6938}
6939
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07006940void mgmt_device_disconnected(struct hci_dev *hdev, bdaddr_t *bdaddr,
Johan Hedberg12d4a3b2014-02-24 14:52:18 +02006941 u8 link_type, u8 addr_type, u8 reason,
6942 bool mgmt_connected)
Johan Hedbergf7520542011-01-20 12:34:39 +02006943{
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02006944 struct mgmt_ev_device_disconnected ev;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006945 struct sock *sk = NULL;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006946
Johan Hedberg84c61d92014-08-01 11:13:30 +03006947 /* The connection is still in hci_conn_hash so test for 1
6948 * instead of 0 to know if this is the last one.
6949 */
6950 if (mgmt_powering_down(hdev) && hci_conn_count(hdev) == 1) {
6951 cancel_delayed_work(&hdev->power_off);
6952 queue_work(hdev->req_workqueue, &hdev->power_off.work);
Johan Hedberg8b064a32014-02-24 14:52:22 +02006953 }
6954
Johan Hedberg12d4a3b2014-02-24 14:52:18 +02006955 if (!mgmt_connected)
6956 return;
6957
Andre Guedes57eb7762013-10-30 19:01:41 -03006958 if (link_type != ACL_LINK && link_type != LE_LINK)
6959 return;
6960
Johan Hedberg744cf192011-11-08 20:40:14 +02006961 mgmt_pending_foreach(MGMT_OP_DISCONNECT, hdev, disconnect_rsp, &sk);
Johan Hedbergf7520542011-01-20 12:34:39 +02006962
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02006963 bacpy(&ev.addr.bdaddr, bdaddr);
6964 ev.addr.type = link_to_bdaddr(link_type, addr_type);
6965 ev.reason = reason;
Johan Hedbergf7520542011-01-20 12:34:39 +02006966
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07006967 mgmt_event(MGMT_EV_DEVICE_DISCONNECTED, hdev, &ev, sizeof(ev), sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006968
6969 if (sk)
Szymon Jancd97dcb62012-03-16 16:02:56 +01006970 sock_put(sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006971
Johan Hedberg124f6e32012-02-09 13:50:12 +02006972 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03006973 hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006974}
6975
Marcel Holtmann78929242013-10-06 23:55:47 -07006976void mgmt_disconnect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr,
6977 u8 link_type, u8 addr_type, u8 status)
Johan Hedberg8962ee72011-01-20 12:40:27 +02006978{
Andre Guedes3655bba2013-10-30 19:01:40 -03006979 u8 bdaddr_type = link_to_bdaddr(link_type, addr_type);
6980 struct mgmt_cp_disconnect *cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006981 struct mgmt_pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006982
Jefferson Delfes36a75f12012-09-18 13:36:54 -04006983 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
6984 hdev);
6985
Johan Hedberg333ae952015-03-17 13:48:47 +02006986 cmd = pending_find(MGMT_OP_DISCONNECT, hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006987 if (!cmd)
Marcel Holtmann78929242013-10-06 23:55:47 -07006988 return;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006989
Andre Guedes3655bba2013-10-30 19:01:40 -03006990 cp = cmd->param;
6991
6992 if (bacmp(bdaddr, &cp->addr.bdaddr))
6993 return;
6994
6995 if (cp->addr.type != bdaddr_type)
6996 return;
6997
Johan Hedbergf5818c22014-12-05 13:36:02 +02006998 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03006999 mgmt_pending_remove(cmd);
Johan Hedbergf7520542011-01-20 12:34:39 +02007000}
Johan Hedberg17d5c042011-01-22 06:09:08 +02007001
Marcel Holtmann445608d2013-10-06 23:55:48 -07007002void mgmt_connect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
7003 u8 addr_type, u8 status)
Johan Hedberg17d5c042011-01-22 06:09:08 +02007004{
7005 struct mgmt_ev_connect_failed ev;
Johan Hedbergc9910d02014-02-27 14:35:12 +02007006
Johan Hedberg84c61d92014-08-01 11:13:30 +03007007 /* The connection is still in hci_conn_hash so test for 1
7008 * instead of 0 to know if this is the last one.
7009 */
7010 if (mgmt_powering_down(hdev) && hci_conn_count(hdev) == 1) {
7011 cancel_delayed_work(&hdev->power_off);
7012 queue_work(hdev->req_workqueue, &hdev->power_off.work);
Johan Hedbergc9910d02014-02-27 14:35:12 +02007013 }
Johan Hedberg17d5c042011-01-22 06:09:08 +02007014
Johan Hedberg4c659c32011-11-07 23:13:39 +02007015 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007016 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02007017 ev.status = mgmt_status(status);
Johan Hedberg17d5c042011-01-22 06:09:08 +02007018
Marcel Holtmann445608d2013-10-06 23:55:48 -07007019 mgmt_event(MGMT_EV_CONNECT_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg17d5c042011-01-22 06:09:08 +02007020}
Johan Hedberg980e1a52011-01-22 06:10:07 +02007021
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07007022void mgmt_pin_code_request(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 secure)
Johan Hedberg980e1a52011-01-22 06:10:07 +02007023{
7024 struct mgmt_ev_pin_code_request ev;
7025
Johan Hedbergd8457692012-02-17 14:24:57 +02007026 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03007027 ev.addr.type = BDADDR_BREDR;
Waldemar Rymarkiewicza770bb52011-04-28 12:07:59 +02007028 ev.secure = secure;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007029
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07007030 mgmt_event(MGMT_EV_PIN_CODE_REQUEST, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007031}
7032
Marcel Holtmanne669cf82013-10-15 14:26:21 -07007033void mgmt_pin_code_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
7034 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02007035{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007036 struct mgmt_pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007037
Johan Hedberg333ae952015-03-17 13:48:47 +02007038 cmd = pending_find(MGMT_OP_PIN_CODE_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007039 if (!cmd)
Marcel Holtmanne669cf82013-10-15 14:26:21 -07007040 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007041
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007042 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03007043 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007044}
7045
Marcel Holtmann3eb38522013-10-15 14:26:22 -07007046void mgmt_pin_code_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
7047 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02007048{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007049 struct mgmt_pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007050
Johan Hedberg333ae952015-03-17 13:48:47 +02007051 cmd = pending_find(MGMT_OP_PIN_CODE_NEG_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007052 if (!cmd)
Marcel Holtmann3eb38522013-10-15 14:26:22 -07007053 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007054
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007055 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03007056 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007057}
Johan Hedberga5c29682011-02-19 12:05:57 -03007058
Johan Hedberg744cf192011-11-08 20:40:14 +02007059int mgmt_user_confirm_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Johan Hedberg39adbff2014-03-20 08:18:14 +02007060 u8 link_type, u8 addr_type, u32 value,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007061 u8 confirm_hint)
Johan Hedberga5c29682011-02-19 12:05:57 -03007062{
7063 struct mgmt_ev_user_confirm_request ev;
7064
Johan Hedberg744cf192011-11-08 20:40:14 +02007065 BT_DBG("%s", hdev->name);
Johan Hedberga5c29682011-02-19 12:05:57 -03007066
Johan Hedberg272d90d2012-02-09 15:26:12 +02007067 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007068 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberg55bc1a32011-04-28 11:28:56 -07007069 ev.confirm_hint = confirm_hint;
Johan Hedberg39adbff2014-03-20 08:18:14 +02007070 ev.value = cpu_to_le32(value);
Johan Hedberga5c29682011-02-19 12:05:57 -03007071
Johan Hedberg744cf192011-11-08 20:40:14 +02007072 return mgmt_event(MGMT_EV_USER_CONFIRM_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007073 NULL);
Johan Hedberga5c29682011-02-19 12:05:57 -03007074}
7075
Johan Hedberg272d90d2012-02-09 15:26:12 +02007076int mgmt_user_passkey_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03007077 u8 link_type, u8 addr_type)
Brian Gix604086b2011-11-23 08:28:33 -08007078{
7079 struct mgmt_ev_user_passkey_request ev;
7080
7081 BT_DBG("%s", hdev->name);
7082
Johan Hedberg272d90d2012-02-09 15:26:12 +02007083 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007084 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Brian Gix604086b2011-11-23 08:28:33 -08007085
7086 return mgmt_event(MGMT_EV_USER_PASSKEY_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007087 NULL);
Brian Gix604086b2011-11-23 08:28:33 -08007088}
7089
Brian Gix0df4c182011-11-16 13:53:13 -08007090static int user_pairing_resp_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03007091 u8 link_type, u8 addr_type, u8 status,
7092 u8 opcode)
Johan Hedberga5c29682011-02-19 12:05:57 -03007093{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007094 struct mgmt_pending_cmd *cmd;
Johan Hedberga5c29682011-02-19 12:05:57 -03007095
Johan Hedberg333ae952015-03-17 13:48:47 +02007096 cmd = pending_find(opcode, hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03007097 if (!cmd)
7098 return -ENOENT;
7099
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007100 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03007101 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03007102
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007103 return 0;
Johan Hedberga5c29682011-02-19 12:05:57 -03007104}
7105
Johan Hedberg744cf192011-11-08 20:40:14 +02007106int mgmt_user_confirm_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007107 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03007108{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007109 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007110 status, MGMT_OP_USER_CONFIRM_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03007111}
7112
Johan Hedberg272d90d2012-02-09 15:26:12 +02007113int mgmt_user_confirm_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007114 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03007115{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007116 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03007117 status,
7118 MGMT_OP_USER_CONFIRM_NEG_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03007119}
Johan Hedberg2a611692011-02-19 12:06:00 -03007120
Brian Gix604086b2011-11-23 08:28:33 -08007121int mgmt_user_passkey_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007122 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08007123{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007124 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007125 status, MGMT_OP_USER_PASSKEY_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08007126}
7127
Johan Hedberg272d90d2012-02-09 15:26:12 +02007128int mgmt_user_passkey_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007129 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08007130{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007131 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03007132 status,
7133 MGMT_OP_USER_PASSKEY_NEG_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08007134}
7135
Johan Hedberg92a25252012-09-06 18:39:26 +03007136int mgmt_user_passkey_notify(struct hci_dev *hdev, bdaddr_t *bdaddr,
7137 u8 link_type, u8 addr_type, u32 passkey,
7138 u8 entered)
7139{
7140 struct mgmt_ev_passkey_notify ev;
7141
7142 BT_DBG("%s", hdev->name);
7143
7144 bacpy(&ev.addr.bdaddr, bdaddr);
7145 ev.addr.type = link_to_bdaddr(link_type, addr_type);
7146 ev.passkey = __cpu_to_le32(passkey);
7147 ev.entered = entered;
7148
7149 return mgmt_event(MGMT_EV_PASSKEY_NOTIFY, hdev, &ev, sizeof(ev), NULL);
7150}
7151
Johan Hedberge1e930f2014-09-08 17:09:49 -07007152void mgmt_auth_failed(struct hci_conn *conn, u8 hci_status)
Johan Hedberg2a611692011-02-19 12:06:00 -03007153{
7154 struct mgmt_ev_auth_failed ev;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007155 struct mgmt_pending_cmd *cmd;
Johan Hedberge1e930f2014-09-08 17:09:49 -07007156 u8 status = mgmt_status(hci_status);
Johan Hedberg2a611692011-02-19 12:06:00 -03007157
Johan Hedberge1e930f2014-09-08 17:09:49 -07007158 bacpy(&ev.addr.bdaddr, &conn->dst);
7159 ev.addr.type = link_to_bdaddr(conn->type, conn->dst_type);
7160 ev.status = status;
Johan Hedberg2a611692011-02-19 12:06:00 -03007161
Johan Hedberge1e930f2014-09-08 17:09:49 -07007162 cmd = find_pairing(conn);
7163
7164 mgmt_event(MGMT_EV_AUTH_FAILED, conn->hdev, &ev, sizeof(ev),
7165 cmd ? cmd->sk : NULL);
7166
Johan Hedberga511b352014-12-11 21:45:45 +02007167 if (cmd) {
7168 cmd->cmd_complete(cmd, status);
7169 mgmt_pending_remove(cmd);
7170 }
Johan Hedberg2a611692011-02-19 12:06:00 -03007171}
Johan Hedbergb312b1612011-03-16 14:29:37 +02007172
Marcel Holtmann464996a2013-10-15 14:26:24 -07007173void mgmt_auth_enable_complete(struct hci_dev *hdev, u8 status)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007174{
7175 struct cmd_lookup match = { NULL, hdev };
Marcel Holtmann464996a2013-10-15 14:26:24 -07007176 bool changed;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007177
7178 if (status) {
7179 u8 mgmt_err = mgmt_status(status);
7180 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007181 cmd_status_rsp, &mgmt_err);
Marcel Holtmann464996a2013-10-15 14:26:24 -07007182 return;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007183 }
7184
Marcel Holtmann464996a2013-10-15 14:26:24 -07007185 if (test_bit(HCI_AUTH, &hdev->flags))
Marcel Holtmann238be782015-03-13 02:11:06 -07007186 changed = !hci_dev_test_and_set_flag(hdev, HCI_LINK_SECURITY);
Marcel Holtmann464996a2013-10-15 14:26:24 -07007187 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007188 changed = hci_dev_test_and_clear_flag(hdev, HCI_LINK_SECURITY);
Johan Hedberg47990ea2012-02-22 11:58:37 +02007189
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007190 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev, settings_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007191 &match);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007192
Johan Hedberg47990ea2012-02-22 11:58:37 +02007193 if (changed)
Marcel Holtmann464996a2013-10-15 14:26:24 -07007194 new_settings(hdev, match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007195
7196 if (match.sk)
7197 sock_put(match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007198}
7199
Johan Hedberg890ea892013-03-15 17:06:52 -05007200static void clear_eir(struct hci_request *req)
Johan Hedbergcacaf522012-02-21 00:52:42 +02007201{
Johan Hedberg890ea892013-03-15 17:06:52 -05007202 struct hci_dev *hdev = req->hdev;
Johan Hedbergcacaf522012-02-21 00:52:42 +02007203 struct hci_cp_write_eir cp;
7204
Johan Hedberg976eb202012-10-24 21:12:01 +03007205 if (!lmp_ext_inq_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05007206 return;
Johan Hedbergcacaf522012-02-21 00:52:42 +02007207
Johan Hedbergc80da272012-02-22 15:38:48 +02007208 memset(hdev->eir, 0, sizeof(hdev->eir));
7209
Johan Hedbergcacaf522012-02-21 00:52:42 +02007210 memset(&cp, 0, sizeof(cp));
7211
Johan Hedberg890ea892013-03-15 17:06:52 -05007212 hci_req_add(req, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
Johan Hedbergcacaf522012-02-21 00:52:42 +02007213}
7214
Marcel Holtmann3e248562013-10-15 14:26:25 -07007215void mgmt_ssp_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007216{
7217 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg890ea892013-03-15 17:06:52 -05007218 struct hci_request req;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007219 bool changed = false;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007220
7221 if (status) {
7222 u8 mgmt_err = mgmt_status(status);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007223
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007224 if (enable && hci_dev_test_and_clear_flag(hdev,
7225 HCI_SSP_ENABLED)) {
Marcel Holtmanna358dc12015-03-13 02:11:02 -07007226 hci_dev_clear_flag(hdev, HCI_HS_ENABLED);
Marcel Holtmann3e248562013-10-15 14:26:25 -07007227 new_settings(hdev, NULL);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07007228 }
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007229
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007230 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, cmd_status_rsp,
7231 &mgmt_err);
Marcel Holtmann3e248562013-10-15 14:26:25 -07007232 return;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007233 }
7234
7235 if (enable) {
Marcel Holtmann238be782015-03-13 02:11:06 -07007236 changed = !hci_dev_test_and_set_flag(hdev, HCI_SSP_ENABLED);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007237 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007238 changed = hci_dev_test_and_clear_flag(hdev, HCI_SSP_ENABLED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07007239 if (!changed)
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007240 changed = hci_dev_test_and_clear_flag(hdev,
7241 HCI_HS_ENABLED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07007242 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07007243 hci_dev_clear_flag(hdev, HCI_HS_ENABLED);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007244 }
7245
7246 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, settings_rsp, &match);
7247
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007248 if (changed)
Marcel Holtmann3e248562013-10-15 14:26:25 -07007249 new_settings(hdev, match.sk);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007250
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02007251 if (match.sk)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007252 sock_put(match.sk);
7253
Johan Hedberg890ea892013-03-15 17:06:52 -05007254 hci_req_init(&req, hdev);
7255
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07007256 if (hci_dev_test_flag(hdev, HCI_SSP_ENABLED)) {
7257 if (hci_dev_test_flag(hdev, HCI_USE_DEBUG_KEYS))
Johan Hedberg37699722014-06-24 14:00:27 +03007258 hci_req_add(&req, HCI_OP_WRITE_SSP_DEBUG_MODE,
7259 sizeof(enable), &enable);
Johan Hedbergb1a89172015-11-25 16:15:42 +02007260 __hci_req_update_eir(&req);
Johan Hedberg37699722014-06-24 14:00:27 +03007261 } else {
Johan Hedberg890ea892013-03-15 17:06:52 -05007262 clear_eir(&req);
Johan Hedberg37699722014-06-24 14:00:27 +03007263 }
Johan Hedberg890ea892013-03-15 17:06:52 -05007264
7265 hci_req_run(&req, NULL);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007266}
7267
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007268static void sk_lookup(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedberg90e70452012-02-23 23:09:40 +02007269{
7270 struct cmd_lookup *match = data;
7271
Johan Hedberg90e70452012-02-23 23:09:40 +02007272 if (match->sk == NULL) {
7273 match->sk = cmd->sk;
7274 sock_hold(match->sk);
7275 }
Johan Hedberg90e70452012-02-23 23:09:40 +02007276}
7277
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07007278void mgmt_set_class_of_dev_complete(struct hci_dev *hdev, u8 *dev_class,
7279 u8 status)
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01007280{
Johan Hedberg90e70452012-02-23 23:09:40 +02007281 struct cmd_lookup match = { NULL, hdev, mgmt_status(status) };
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01007282
Johan Hedberg92da6092013-03-15 17:06:55 -05007283 mgmt_pending_foreach(MGMT_OP_SET_DEV_CLASS, hdev, sk_lookup, &match);
7284 mgmt_pending_foreach(MGMT_OP_ADD_UUID, hdev, sk_lookup, &match);
7285 mgmt_pending_foreach(MGMT_OP_REMOVE_UUID, hdev, sk_lookup, &match);
Johan Hedberg90e70452012-02-23 23:09:40 +02007286
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02007287 if (!status) {
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02007288 mgmt_limited_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev, dev_class,
7289 3, HCI_MGMT_DEV_CLASS_EVENTS, NULL);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02007290 ext_info_changed(hdev, NULL);
7291 }
Johan Hedberg90e70452012-02-23 23:09:40 +02007292
7293 if (match.sk)
7294 sock_put(match.sk);
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01007295}
7296
Marcel Holtmann7667da32013-10-15 14:26:27 -07007297void mgmt_set_local_name_complete(struct hci_dev *hdev, u8 *name, u8 status)
Johan Hedbergb312b1612011-03-16 14:29:37 +02007298{
Johan Hedbergb312b1612011-03-16 14:29:37 +02007299 struct mgmt_cp_set_local_name ev;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007300 struct mgmt_pending_cmd *cmd;
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02007301
Johan Hedberg13928972013-03-15 17:07:00 -05007302 if (status)
Marcel Holtmann7667da32013-10-15 14:26:27 -07007303 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02007304
7305 memset(&ev, 0, sizeof(ev));
7306 memcpy(ev.name, name, HCI_MAX_NAME_LENGTH);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02007307 memcpy(ev.short_name, hdev->short_name, HCI_MAX_SHORT_NAME_LENGTH);
Johan Hedbergb312b1612011-03-16 14:29:37 +02007308
Johan Hedberg333ae952015-03-17 13:48:47 +02007309 cmd = pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
Johan Hedberg13928972013-03-15 17:07:00 -05007310 if (!cmd) {
7311 memcpy(hdev->dev_name, name, sizeof(hdev->dev_name));
Johan Hedbergb312b1612011-03-16 14:29:37 +02007312
Johan Hedberg13928972013-03-15 17:07:00 -05007313 /* If this is a HCI command related to powering on the
7314 * HCI dev don't send any mgmt signals.
7315 */
Johan Hedberg333ae952015-03-17 13:48:47 +02007316 if (pending_find(MGMT_OP_SET_POWERED, hdev))
Marcel Holtmann7667da32013-10-15 14:26:27 -07007317 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02007318 }
7319
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02007320 mgmt_limited_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, &ev, sizeof(ev),
7321 HCI_MGMT_LOCAL_NAME_EVENTS, cmd ? cmd->sk : NULL);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02007322 ext_info_changed(hdev, cmd ? cmd->sk : NULL);
Johan Hedbergb312b1612011-03-16 14:29:37 +02007323}
Szymon Jancc35938b2011-03-22 13:12:21 +01007324
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007325static inline bool has_uuid(u8 *uuid, u16 uuid_count, u8 (*uuids)[16])
7326{
7327 int i;
7328
7329 for (i = 0; i < uuid_count; i++) {
7330 if (!memcmp(uuid, uuids[i], 16))
7331 return true;
7332 }
7333
7334 return false;
7335}
7336
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007337static bool eir_has_uuids(u8 *eir, u16 eir_len, u16 uuid_count, u8 (*uuids)[16])
7338{
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007339 u16 parsed = 0;
7340
7341 while (parsed < eir_len) {
7342 u8 field_len = eir[0];
7343 u8 uuid[16];
7344 int i;
7345
7346 if (field_len == 0)
7347 break;
7348
7349 if (eir_len - parsed < field_len + 1)
7350 break;
7351
7352 switch (eir[1]) {
7353 case EIR_UUID16_ALL:
7354 case EIR_UUID16_SOME:
7355 for (i = 0; i + 3 <= field_len; i += 2) {
Johan Hedberg189f6ad2014-12-05 13:40:01 +02007356 memcpy(uuid, bluetooth_base_uuid, 16);
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007357 uuid[13] = eir[i + 3];
7358 uuid[12] = eir[i + 2];
7359 if (has_uuid(uuid, uuid_count, uuids))
7360 return true;
7361 }
7362 break;
7363 case EIR_UUID32_ALL:
7364 case EIR_UUID32_SOME:
7365 for (i = 0; i + 5 <= field_len; i += 4) {
Johan Hedberg189f6ad2014-12-05 13:40:01 +02007366 memcpy(uuid, bluetooth_base_uuid, 16);
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007367 uuid[15] = eir[i + 5];
7368 uuid[14] = eir[i + 4];
7369 uuid[13] = eir[i + 3];
7370 uuid[12] = eir[i + 2];
7371 if (has_uuid(uuid, uuid_count, uuids))
7372 return true;
7373 }
7374 break;
7375 case EIR_UUID128_ALL:
7376 case EIR_UUID128_SOME:
7377 for (i = 0; i + 17 <= field_len; i += 16) {
7378 memcpy(uuid, eir + i + 2, 16);
7379 if (has_uuid(uuid, uuid_count, uuids))
7380 return true;
7381 }
7382 break;
7383 }
7384
7385 parsed += field_len + 1;
7386 eir += field_len + 1;
7387 }
7388
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007389 return false;
7390}
7391
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007392static void restart_le_scan(struct hci_dev *hdev)
7393{
7394 /* If controller is not scanning we are done. */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07007395 if (!hci_dev_test_flag(hdev, HCI_LE_SCAN))
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007396 return;
7397
7398 if (time_after(jiffies + DISCOV_LE_RESTART_DELAY,
7399 hdev->discovery.scan_start +
7400 hdev->discovery.scan_duration))
7401 return;
7402
Johan Hedberg7c1fbed2015-11-11 08:11:23 +02007403 queue_delayed_work(hdev->req_workqueue, &hdev->le_scan_restart,
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007404 DISCOV_LE_RESTART_DELAY);
7405}
7406
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007407static bool is_filter_match(struct hci_dev *hdev, s8 rssi, u8 *eir,
7408 u16 eir_len, u8 *scan_rsp, u8 scan_rsp_len)
7409{
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007410 /* If a RSSI threshold has been specified, and
7411 * HCI_QUIRK_STRICT_DUPLICATE_FILTER is not set, then all results with
7412 * a RSSI smaller than the RSSI threshold will be dropped. If the quirk
7413 * is set, let it through for further processing, as we might need to
7414 * restart the scan.
7415 *
7416 * For BR/EDR devices (pre 1.2) providing no RSSI during inquiry,
7417 * the results are also dropped.
7418 */
7419 if (hdev->discovery.rssi != HCI_RSSI_INVALID &&
7420 (rssi == HCI_RSSI_INVALID ||
7421 (rssi < hdev->discovery.rssi &&
7422 !test_bit(HCI_QUIRK_STRICT_DUPLICATE_FILTER, &hdev->quirks))))
7423 return false;
7424
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007425 if (hdev->discovery.uuid_count != 0) {
7426 /* If a list of UUIDs is provided in filter, results with no
7427 * matching UUID should be dropped.
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007428 */
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007429 if (!eir_has_uuids(eir, eir_len, hdev->discovery.uuid_count,
7430 hdev->discovery.uuids) &&
7431 !eir_has_uuids(scan_rsp, scan_rsp_len,
7432 hdev->discovery.uuid_count,
7433 hdev->discovery.uuids))
7434 return false;
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007435 }
7436
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007437 /* If duplicate filtering does not report RSSI changes, then restart
7438 * scanning to ensure updated result with updated RSSI values.
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007439 */
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007440 if (test_bit(HCI_QUIRK_STRICT_DUPLICATE_FILTER, &hdev->quirks)) {
7441 restart_le_scan(hdev);
7442
7443 /* Validate RSSI value against the RSSI threshold once more. */
7444 if (hdev->discovery.rssi != HCI_RSSI_INVALID &&
7445 rssi < hdev->discovery.rssi)
7446 return false;
7447 }
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007448
7449 return true;
7450}
7451
Marcel Holtmann901801b2013-10-06 23:55:51 -07007452void mgmt_device_found(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
Marcel Holtmannaf589252014-07-01 14:11:20 +02007453 u8 addr_type, u8 *dev_class, s8 rssi, u32 flags,
7454 u8 *eir, u16 eir_len, u8 *scan_rsp, u8 scan_rsp_len)
Johan Hedberge17acd42011-03-30 23:57:16 +03007455{
Johan Hedberge319d2e2012-01-15 19:51:59 +02007456 char buf[512];
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007457 struct mgmt_ev_device_found *ev = (void *)buf;
Johan Hedberg1dc06092012-01-15 21:01:23 +02007458 size_t ev_size;
Johan Hedberge17acd42011-03-30 23:57:16 +03007459
Johan Hedberg75ce2082014-07-02 22:42:01 +03007460 /* Don't send events for a non-kernel initiated discovery. With
7461 * LE one exception is if we have pend_le_reports > 0 in which
7462 * case we're doing passive scanning and want these events.
7463 */
7464 if (!hci_discovery_active(hdev)) {
7465 if (link_type == ACL_LINK)
7466 return;
Johan Hedberg66f84552014-07-04 12:37:18 +03007467 if (link_type == LE_LINK && list_empty(&hdev->pend_le_reports))
Johan Hedberg75ce2082014-07-02 22:42:01 +03007468 return;
7469 }
Andre Guedes12602d02013-04-30 15:29:40 -03007470
Jakub Pawlowski82f8b652015-03-04 16:24:26 -08007471 if (hdev->discovery.result_filtering) {
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007472 /* We are using service discovery */
7473 if (!is_filter_match(hdev, rssi, eir, eir_len, scan_rsp,
7474 scan_rsp_len))
7475 return;
7476 }
Marcel Holtmannbda157a2014-12-05 10:55:56 +01007477
Johan Hedberg78b781c2016-01-05 13:19:32 +02007478 if (hdev->discovery.limited) {
7479 /* Check for limited discoverable bit */
7480 if (dev_class) {
7481 if (!(dev_class[1] & 0x20))
7482 return;
7483 } else {
7484 u8 *flags = eir_get_data(eir, eir_len, EIR_FLAGS, NULL);
7485 if (!flags || !(flags[0] & LE_AD_LIMITED))
7486 return;
7487 }
7488 }
7489
Johan Hedberg5d2e9fa2014-03-25 10:30:47 +02007490 /* Make sure that the buffer is big enough. The 5 extra bytes
7491 * are for the potential CoD field.
7492 */
7493 if (sizeof(*ev) + eir_len + scan_rsp_len + 5 > sizeof(buf))
Marcel Holtmann901801b2013-10-06 23:55:51 -07007494 return;
Andre Guedes7d262f82012-01-10 18:20:49 -03007495
Johan Hedberg1dc06092012-01-15 21:01:23 +02007496 memset(buf, 0, sizeof(buf));
7497
Marcel Holtmannda25cf62014-12-05 13:03:35 +01007498 /* In case of device discovery with BR/EDR devices (pre 1.2), the
7499 * RSSI value was reported as 0 when not available. This behavior
7500 * is kept when using device discovery. This is required for full
7501 * backwards compatibility with the API.
7502 *
7503 * However when using service discovery, the value 127 will be
7504 * returned when the RSSI is not available.
7505 */
Szymon Janc91200e92015-01-22 16:57:05 +01007506 if (rssi == HCI_RSSI_INVALID && !hdev->discovery.report_invalid_rssi &&
7507 link_type == ACL_LINK)
Marcel Holtmannefb25132014-12-05 13:03:34 +01007508 rssi = 0;
7509
Johan Hedberg841c5642014-07-07 12:45:54 +03007510 bacpy(&ev->addr.bdaddr, bdaddr);
7511 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberge319d2e2012-01-15 19:51:59 +02007512 ev->rssi = rssi;
Marcel Holtmannaf589252014-07-01 14:11:20 +02007513 ev->flags = cpu_to_le32(flags);
Johan Hedberge17acd42011-03-30 23:57:16 +03007514
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007515 if (eir_len > 0)
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007516 /* Copy EIR or advertising data into event */
Johan Hedberge319d2e2012-01-15 19:51:59 +02007517 memcpy(ev->eir, eir, eir_len);
Johan Hedberge17acd42011-03-30 23:57:16 +03007518
Johan Hedberg0d3b7f62016-01-05 13:19:31 +02007519 if (dev_class && !eir_get_data(ev->eir, eir_len, EIR_CLASS_OF_DEV,
7520 NULL))
Johan Hedberg1dc06092012-01-15 21:01:23 +02007521 eir_len = eir_append_data(ev->eir, eir_len, EIR_CLASS_OF_DEV,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007522 dev_class, 3);
Johan Hedberg1dc06092012-01-15 21:01:23 +02007523
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007524 if (scan_rsp_len > 0)
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007525 /* Append scan response data to event */
Johan Hedberg5d2e9fa2014-03-25 10:30:47 +02007526 memcpy(ev->eir + eir_len, scan_rsp, scan_rsp_len);
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007527
Johan Hedberg5d2e9fa2014-03-25 10:30:47 +02007528 ev->eir_len = cpu_to_le16(eir_len + scan_rsp_len);
7529 ev_size = sizeof(*ev) + eir_len + scan_rsp_len;
Andre Guedesf8523592011-09-09 18:56:26 -03007530
Marcel Holtmann901801b2013-10-06 23:55:51 -07007531 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, ev_size, NULL);
Johan Hedberge17acd42011-03-30 23:57:16 +03007532}
Johan Hedberga88a9652011-03-30 13:18:12 +03007533
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07007534void mgmt_remote_name(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
7535 u8 addr_type, s8 rssi, u8 *name, u8 name_len)
Johan Hedberga88a9652011-03-30 13:18:12 +03007536{
Johan Hedbergb644ba32012-01-17 21:48:47 +02007537 struct mgmt_ev_device_found *ev;
7538 char buf[sizeof(*ev) + HCI_MAX_NAME_LENGTH + 2];
7539 u16 eir_len;
Johan Hedberga88a9652011-03-30 13:18:12 +03007540
Johan Hedbergb644ba32012-01-17 21:48:47 +02007541 ev = (struct mgmt_ev_device_found *) buf;
Johan Hedberga88a9652011-03-30 13:18:12 +03007542
Johan Hedbergb644ba32012-01-17 21:48:47 +02007543 memset(buf, 0, sizeof(buf));
Johan Hedberga88a9652011-03-30 13:18:12 +03007544
Johan Hedbergb644ba32012-01-17 21:48:47 +02007545 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007546 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergb644ba32012-01-17 21:48:47 +02007547 ev->rssi = rssi;
7548
7549 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE, name,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007550 name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02007551
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02007552 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02007553
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07007554 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, sizeof(*ev) + eir_len, NULL);
Johan Hedberga88a9652011-03-30 13:18:12 +03007555}
Johan Hedberg314b2382011-04-27 10:29:57 -04007556
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07007557void mgmt_discovering(struct hci_dev *hdev, u8 discovering)
Johan Hedberg314b2382011-04-27 10:29:57 -04007558{
Johan Hedbergf963e8e2012-02-20 23:30:44 +02007559 struct mgmt_ev_discovering ev;
Johan Hedberg164a6e72011-11-01 17:06:44 +02007560
Andre Guedes343fb142011-11-22 17:14:19 -03007561 BT_DBG("%s discovering %u", hdev->name, discovering);
7562
Johan Hedbergf963e8e2012-02-20 23:30:44 +02007563 memset(&ev, 0, sizeof(ev));
7564 ev.type = hdev->discovery.type;
7565 ev.discovering = discovering;
7566
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07007567 mgmt_event(MGMT_EV_DISCOVERING, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg314b2382011-04-27 10:29:57 -04007568}
Antti Julku5e762442011-08-25 16:48:02 +03007569
Johan Hedberg6d785aa32015-03-06 21:08:51 +02007570static struct hci_mgmt_chan chan = {
7571 .channel = HCI_CHANNEL_CONTROL,
7572 .handler_count = ARRAY_SIZE(mgmt_handlers),
7573 .handlers = mgmt_handlers,
Johan Hedberg88b94ce2015-03-17 13:48:49 +02007574 .hdev_init = mgmt_init_hdev,
Johan Hedberg6d785aa32015-03-06 21:08:51 +02007575};
7576
7577int mgmt_init(void)
7578{
7579 return hci_mgmt_chan_register(&chan);
7580}
7581
7582void mgmt_exit(void)
7583{
7584 hci_mgmt_chan_unregister(&chan);
7585}