blob: 7b2bac492fb1cab0a80cf74306a98c6d1230f00e [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 Holtmannedd3896b2014-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Holtmann08dc0e982015-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 Hedbergf7b64e692010-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 Hedbergf7b64e692010-12-13 21:07:06 +0200863
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -0300864 hci_dev_unlock(hdev);
Johan Hedbergf7b64e692010-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 Hedbergf7b64e692010-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 Hedberge8ba3a1f2013-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 Hedberge8ba3a1f2013-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 Hedberge8ba3a1f2013-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 Hedberge8ba3a1f2013-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 Hedberge8ba3a1f2013-10-19 23:38:18 +03001404 return new_settings(hdev, sk);
Johan Hedberg562064e2014-07-08 16:35:34 +03001405 }
Johan Hedberge8ba3a1f2013-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 Hedberge8ba3a1f2013-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 Hedberg4ae143012013-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 Hedberg4ae143012013-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
MichaƂ Narajowskiaf4168c2016-09-19 14:33:33 +02003181 if (!lmp_le_capable(hdev))
3182 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_APPEARANCE,
3183 MGMT_STATUS_NOT_SUPPORTED);
3184
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02003185 apperance = le16_to_cpu(cp->appearance);
3186
3187 hci_dev_lock(hdev);
3188
3189 if (hdev->appearance != apperance) {
3190 hdev->appearance = apperance;
3191
3192 if (hci_dev_test_flag(hdev, HCI_LE_ADV))
3193 adv_expire(hdev, MGMT_ADV_FLAG_APPEARANCE);
MichaƂ Narajowskie74317f2016-09-19 20:25:56 +02003194
3195 ext_info_changed(hdev, sk);
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02003196 }
3197
3198 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_APPEARANCE, 0, NULL,
3199 0);
3200
3201 hci_dev_unlock(hdev);
3202
3203 return err;
3204}
3205
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003206static void read_local_oob_data_complete(struct hci_dev *hdev, u8 status,
3207 u16 opcode, struct sk_buff *skb)
3208{
3209 struct mgmt_rp_read_local_oob_data mgmt_rp;
3210 size_t rp_size = sizeof(mgmt_rp);
3211 struct mgmt_pending_cmd *cmd;
3212
3213 BT_DBG("%s status %u", hdev->name, status);
3214
3215 cmd = pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev);
3216 if (!cmd)
3217 return;
3218
3219 if (status || !skb) {
3220 mgmt_cmd_status(cmd->sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3221 status ? mgmt_status(status) : MGMT_STATUS_FAILED);
3222 goto remove;
3223 }
3224
3225 memset(&mgmt_rp, 0, sizeof(mgmt_rp));
3226
3227 if (opcode == HCI_OP_READ_LOCAL_OOB_DATA) {
3228 struct hci_rp_read_local_oob_data *rp = (void *) skb->data;
3229
3230 if (skb->len < sizeof(*rp)) {
3231 mgmt_cmd_status(cmd->sk, hdev->id,
3232 MGMT_OP_READ_LOCAL_OOB_DATA,
3233 MGMT_STATUS_FAILED);
3234 goto remove;
3235 }
3236
3237 memcpy(mgmt_rp.hash192, rp->hash, sizeof(rp->hash));
3238 memcpy(mgmt_rp.rand192, rp->rand, sizeof(rp->rand));
3239
3240 rp_size -= sizeof(mgmt_rp.hash256) + sizeof(mgmt_rp.rand256);
3241 } else {
3242 struct hci_rp_read_local_oob_ext_data *rp = (void *) skb->data;
3243
3244 if (skb->len < sizeof(*rp)) {
3245 mgmt_cmd_status(cmd->sk, hdev->id,
3246 MGMT_OP_READ_LOCAL_OOB_DATA,
3247 MGMT_STATUS_FAILED);
3248 goto remove;
3249 }
3250
3251 memcpy(mgmt_rp.hash192, rp->hash192, sizeof(rp->hash192));
3252 memcpy(mgmt_rp.rand192, rp->rand192, sizeof(rp->rand192));
3253
3254 memcpy(mgmt_rp.hash256, rp->hash256, sizeof(rp->hash256));
3255 memcpy(mgmt_rp.rand256, rp->rand256, sizeof(rp->rand256));
3256 }
3257
3258 mgmt_cmd_complete(cmd->sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3259 MGMT_STATUS_SUCCESS, &mgmt_rp, rp_size);
3260
3261remove:
3262 mgmt_pending_remove(cmd);
3263}
3264
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02003265static int read_local_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003266 void *data, u16 data_len)
Szymon Jancc35938b2011-03-22 13:12:21 +01003267{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003268 struct mgmt_pending_cmd *cmd;
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003269 struct hci_request req;
Szymon Jancc35938b2011-03-22 13:12:21 +01003270 int err;
3271
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003272 BT_DBG("%s", hdev->name);
Szymon Jancc35938b2011-03-22 13:12:21 +01003273
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003274 hci_dev_lock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003275
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003276 if (!hdev_is_powered(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02003277 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3278 MGMT_STATUS_NOT_POWERED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003279 goto unlock;
3280 }
3281
Andre Guedes9a1a1992012-07-24 15:03:48 -03003282 if (!lmp_ssp_capable(hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02003283 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3284 MGMT_STATUS_NOT_SUPPORTED);
Szymon Jancc35938b2011-03-22 13:12:21 +01003285 goto unlock;
3286 }
3287
Johan Hedberg333ae952015-03-17 13:48:47 +02003288 if (pending_find(MGMT_OP_READ_LOCAL_OOB_DATA, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02003289 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_DATA,
3290 MGMT_STATUS_BUSY);
Szymon Jancc35938b2011-03-22 13:12:21 +01003291 goto unlock;
3292 }
3293
Johan Hedberg2e58ef32011-11-08 20:40:15 +02003294 cmd = mgmt_pending_add(sk, MGMT_OP_READ_LOCAL_OOB_DATA, hdev, NULL, 0);
Szymon Jancc35938b2011-03-22 13:12:21 +01003295 if (!cmd) {
3296 err = -ENOMEM;
3297 goto unlock;
3298 }
3299
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003300 hci_req_init(&req, hdev);
Marcel Holtmann4d2d2792014-01-10 02:07:26 -08003301
Johan Hedberg1b9441f2015-04-02 13:41:13 +03003302 if (bredr_sc_enabled(hdev))
3303 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_EXT_DATA, 0, NULL);
3304 else
3305 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_DATA, 0, NULL);
3306
3307 err = hci_req_run_skb(&req, read_local_oob_data_complete);
Szymon Jancc35938b2011-03-22 13:12:21 +01003308 if (err < 0)
3309 mgmt_pending_remove(cmd);
3310
3311unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003312 hci_dev_unlock(hdev);
Szymon Jancc35938b2011-03-22 13:12:21 +01003313 return err;
3314}
3315
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003316static int add_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003317 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003318{
Johan Hedberg5d57e792015-01-23 10:10:38 +02003319 struct mgmt_addr_info *addr = data;
Szymon Janc2763eda2011-03-22 13:12:22 +01003320 int err;
3321
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003322 BT_DBG("%s ", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003323
Johan Hedberg5d57e792015-01-23 10:10:38 +02003324 if (!bdaddr_type_is_valid(addr->type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003325 return mgmt_cmd_complete(sk, hdev->id,
3326 MGMT_OP_ADD_REMOTE_OOB_DATA,
3327 MGMT_STATUS_INVALID_PARAMS,
3328 addr, sizeof(*addr));
Johan Hedberg5d57e792015-01-23 10:10:38 +02003329
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003330 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003331
Marcel Holtmannec109112014-01-10 02:07:30 -08003332 if (len == MGMT_ADD_REMOTE_OOB_DATA_SIZE) {
3333 struct mgmt_cp_add_remote_oob_data *cp = data;
3334 u8 status;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003335
Johan Hedbergc19a4952014-11-17 20:52:19 +02003336 if (cp->addr.type != BDADDR_BREDR) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003337 err = mgmt_cmd_complete(sk, hdev->id,
3338 MGMT_OP_ADD_REMOTE_OOB_DATA,
3339 MGMT_STATUS_INVALID_PARAMS,
3340 &cp->addr, sizeof(cp->addr));
Johan Hedbergc19a4952014-11-17 20:52:19 +02003341 goto unlock;
3342 }
3343
Marcel Holtmannec109112014-01-10 02:07:30 -08003344 err = hci_add_remote_oob_data(hdev, &cp->addr.bdaddr,
Johan Hedberg6928a922014-10-26 20:46:09 +01003345 cp->addr.type, cp->hash,
3346 cp->rand, NULL, NULL);
Marcel Holtmannec109112014-01-10 02:07:30 -08003347 if (err < 0)
3348 status = MGMT_STATUS_FAILED;
3349 else
3350 status = MGMT_STATUS_SUCCESS;
3351
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003352 err = mgmt_cmd_complete(sk, hdev->id,
3353 MGMT_OP_ADD_REMOTE_OOB_DATA, status,
3354 &cp->addr, sizeof(cp->addr));
Marcel Holtmannec109112014-01-10 02:07:30 -08003355 } else if (len == MGMT_ADD_REMOTE_OOB_EXT_DATA_SIZE) {
3356 struct mgmt_cp_add_remote_oob_ext_data *cp = data;
Marcel Holtmann41bcfd52015-01-31 00:37:02 -08003357 u8 *rand192, *hash192, *rand256, *hash256;
Marcel Holtmannec109112014-01-10 02:07:30 -08003358 u8 status;
3359
Johan Hedberg86df9202014-10-26 20:52:27 +01003360 if (bdaddr_type_is_le(cp->addr.type)) {
Johan Hedbergd25b78e2015-01-27 12:55:52 +02003361 /* Enforce zero-valued 192-bit parameters as
3362 * long as legacy SMP OOB isn't implemented.
3363 */
3364 if (memcmp(cp->rand192, ZERO_KEY, 16) ||
3365 memcmp(cp->hash192, ZERO_KEY, 16)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003366 err = mgmt_cmd_complete(sk, hdev->id,
3367 MGMT_OP_ADD_REMOTE_OOB_DATA,
3368 MGMT_STATUS_INVALID_PARAMS,
3369 addr, sizeof(*addr));
Johan Hedbergd25b78e2015-01-27 12:55:52 +02003370 goto unlock;
3371 }
3372
Johan Hedberg86df9202014-10-26 20:52:27 +01003373 rand192 = NULL;
3374 hash192 = NULL;
3375 } else {
Marcel Holtmann41bcfd52015-01-31 00:37:02 -08003376 /* In case one of the P-192 values is set to zero,
3377 * then just disable OOB data for P-192.
3378 */
3379 if (!memcmp(cp->rand192, ZERO_KEY, 16) ||
3380 !memcmp(cp->hash192, ZERO_KEY, 16)) {
3381 rand192 = NULL;
3382 hash192 = NULL;
3383 } else {
3384 rand192 = cp->rand192;
3385 hash192 = cp->hash192;
3386 }
3387 }
3388
3389 /* In case one of the P-256 values is set to zero, then just
3390 * disable OOB data for P-256.
3391 */
3392 if (!memcmp(cp->rand256, ZERO_KEY, 16) ||
3393 !memcmp(cp->hash256, ZERO_KEY, 16)) {
3394 rand256 = NULL;
3395 hash256 = NULL;
3396 } else {
3397 rand256 = cp->rand256;
3398 hash256 = cp->hash256;
Johan Hedberg86df9202014-10-26 20:52:27 +01003399 }
3400
Johan Hedberg81328d5c2014-10-26 20:33:47 +01003401 err = hci_add_remote_oob_data(hdev, &cp->addr.bdaddr,
Johan Hedberg86df9202014-10-26 20:52:27 +01003402 cp->addr.type, hash192, rand192,
Marcel Holtmann41bcfd52015-01-31 00:37:02 -08003403 hash256, rand256);
Marcel Holtmannec109112014-01-10 02:07:30 -08003404 if (err < 0)
3405 status = MGMT_STATUS_FAILED;
3406 else
3407 status = MGMT_STATUS_SUCCESS;
3408
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003409 err = mgmt_cmd_complete(sk, hdev->id,
3410 MGMT_OP_ADD_REMOTE_OOB_DATA,
3411 status, &cp->addr, sizeof(cp->addr));
Marcel Holtmannec109112014-01-10 02:07:30 -08003412 } else {
3413 BT_ERR("add_remote_oob_data: invalid length of %u bytes", len);
Johan Hedberga69e8372015-03-06 21:08:53 +02003414 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_REMOTE_OOB_DATA,
3415 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmannec109112014-01-10 02:07:30 -08003416 }
Szymon Janc2763eda2011-03-22 13:12:22 +01003417
Johan Hedbergc19a4952014-11-17 20:52:19 +02003418unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003419 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003420 return err;
3421}
3422
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003423static int remove_remote_oob_data(struct sock *sk, struct hci_dev *hdev,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03003424 void *data, u16 len)
Szymon Janc2763eda2011-03-22 13:12:22 +01003425{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003426 struct mgmt_cp_remove_remote_oob_data *cp = data;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003427 u8 status;
Szymon Janc2763eda2011-03-22 13:12:22 +01003428 int err;
3429
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003430 BT_DBG("%s", hdev->name);
Szymon Janc2763eda2011-03-22 13:12:22 +01003431
Johan Hedbergc19a4952014-11-17 20:52:19 +02003432 if (cp->addr.type != BDADDR_BREDR)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003433 return mgmt_cmd_complete(sk, hdev->id,
3434 MGMT_OP_REMOVE_REMOTE_OOB_DATA,
3435 MGMT_STATUS_INVALID_PARAMS,
3436 &cp->addr, sizeof(cp->addr));
Johan Hedbergc19a4952014-11-17 20:52:19 +02003437
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003438 hci_dev_lock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003439
Johan Hedbergeedbd582014-11-15 09:34:23 +02003440 if (!bacmp(&cp->addr.bdaddr, BDADDR_ANY)) {
3441 hci_remote_oob_data_clear(hdev);
3442 status = MGMT_STATUS_SUCCESS;
3443 goto done;
3444 }
3445
Johan Hedberg6928a922014-10-26 20:46:09 +01003446 err = hci_remove_remote_oob_data(hdev, &cp->addr.bdaddr, cp->addr.type);
Szymon Janc2763eda2011-03-22 13:12:22 +01003447 if (err < 0)
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003448 status = MGMT_STATUS_INVALID_PARAMS;
Szymon Janc2763eda2011-03-22 13:12:22 +01003449 else
Szymon Janca6785be2012-12-13 15:11:21 +01003450 status = MGMT_STATUS_SUCCESS;
Johan Hedbergbf1e3542012-02-19 13:16:14 +02003451
Johan Hedbergeedbd582014-11-15 09:34:23 +02003452done:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003453 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_REMOTE_OOB_DATA,
3454 status, &cp->addr, sizeof(cp->addr));
Szymon Janc2763eda2011-03-22 13:12:22 +01003455
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003456 hci_dev_unlock(hdev);
Szymon Janc2763eda2011-03-22 13:12:22 +01003457 return err;
3458}
3459
Johan Hedberge68f0722015-11-11 08:30:30 +02003460void mgmt_start_discovery_complete(struct hci_dev *hdev, u8 status)
Andre Guedes7c307722013-04-30 15:29:28 -03003461{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003462 struct mgmt_pending_cmd *cmd;
Lukasz Rymanowskiae55f592014-03-27 20:55:19 +01003463
Andre Guedes7c307722013-04-30 15:29:28 -03003464 BT_DBG("status %d", status);
3465
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003466 hci_dev_lock(hdev);
3467
Johan Hedberg333ae952015-03-17 13:48:47 +02003468 cmd = pending_find(MGMT_OP_START_DISCOVERY, hdev);
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003469 if (!cmd)
Johan Hedberg333ae952015-03-17 13:48:47 +02003470 cmd = pending_find(MGMT_OP_START_SERVICE_DISCOVERY, hdev);
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003471
Johan Hedberg78b781c2016-01-05 13:19:32 +02003472 if (!cmd)
3473 cmd = pending_find(MGMT_OP_START_LIMITED_DISCOVERY, hdev);
3474
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003475 if (cmd) {
Johan Hedberg2922a942014-12-05 13:36:06 +02003476 cmd->cmd_complete(cmd, mgmt_status(status));
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003477 mgmt_pending_remove(cmd);
Andre Guedes7c307722013-04-30 15:29:28 -03003478 }
3479
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003480 hci_dev_unlock(hdev);
Andre Guedes7c307722013-04-30 15:29:28 -03003481}
3482
Johan Hedberg591752a2015-11-11 08:11:24 +02003483static bool discovery_type_is_valid(struct hci_dev *hdev, uint8_t type,
3484 uint8_t *mgmt_status)
3485{
3486 switch (type) {
3487 case DISCOV_TYPE_LE:
3488 *mgmt_status = mgmt_le_support(hdev);
3489 if (*mgmt_status)
3490 return false;
3491 break;
3492 case DISCOV_TYPE_INTERLEAVED:
3493 *mgmt_status = mgmt_le_support(hdev);
3494 if (*mgmt_status)
3495 return false;
3496 /* Intentional fall-through */
3497 case DISCOV_TYPE_BREDR:
3498 *mgmt_status = mgmt_bredr_support(hdev);
3499 if (*mgmt_status)
3500 return false;
3501 break;
3502 default:
3503 *mgmt_status = MGMT_STATUS_INVALID_PARAMS;
3504 return false;
3505 }
3506
3507 return true;
3508}
3509
Johan Hedberg78b781c2016-01-05 13:19:32 +02003510static int start_discovery_internal(struct sock *sk, struct hci_dev *hdev,
3511 u16 op, void *data, u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003512{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003513 struct mgmt_cp_start_discovery *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003514 struct mgmt_pending_cmd *cmd;
Marcel Holtmann80190442014-12-04 11:36:36 +01003515 u8 status;
Johan Hedberg14a53662011-04-27 10:29:56 -04003516 int err;
3517
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003518 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003519
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003520 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003521
Johan Hedberg4b34ee782012-02-21 14:13:02 +02003522 if (!hdev_is_powered(hdev)) {
Johan Hedberg78b781c2016-01-05 13:19:32 +02003523 err = mgmt_cmd_complete(sk, hdev->id, op,
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003524 MGMT_STATUS_NOT_POWERED,
3525 &cp->type, sizeof(cp->type));
Johan Hedbergbd2d1332011-11-07 23:13:37 +02003526 goto failed;
3527 }
3528
Marcel Holtmannf5a969f2014-12-04 11:36:34 +01003529 if (hdev->discovery.state != DISCOVERY_STOPPED ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07003530 hci_dev_test_flag(hdev, HCI_PERIODIC_INQ)) {
Johan Hedberg78b781c2016-01-05 13:19:32 +02003531 err = mgmt_cmd_complete(sk, hdev->id, op, MGMT_STATUS_BUSY,
3532 &cp->type, sizeof(cp->type));
Andre Guedes642be6c2012-03-21 00:03:37 -03003533 goto failed;
3534 }
3535
Johan Hedberg591752a2015-11-11 08:11:24 +02003536 if (!discovery_type_is_valid(hdev, cp->type, &status)) {
Johan Hedberg78b781c2016-01-05 13:19:32 +02003537 err = mgmt_cmd_complete(sk, hdev->id, op, status,
3538 &cp->type, sizeof(cp->type));
Johan Hedberg591752a2015-11-11 08:11:24 +02003539 goto failed;
3540 }
3541
Marcel Holtmann22078802014-12-05 11:45:22 +01003542 /* Clear the discovery filter first to free any previously
3543 * allocated memory for the UUID list.
3544 */
3545 hci_discovery_filter_clear(hdev);
3546
Andre Guedes4aab14e2012-02-17 20:39:36 -03003547 hdev->discovery.type = cp->type;
Marcel Holtmannda25cf62014-12-05 13:03:35 +01003548 hdev->discovery.report_invalid_rssi = false;
Johan Hedberg78b781c2016-01-05 13:19:32 +02003549 if (op == MGMT_OP_START_LIMITED_DISCOVERY)
3550 hdev->discovery.limited = true;
3551 else
3552 hdev->discovery.limited = false;
Andre Guedes4aab14e2012-02-17 20:39:36 -03003553
Johan Hedberg78b781c2016-01-05 13:19:32 +02003554 cmd = mgmt_pending_add(sk, op, hdev, data, len);
Johan Hedberge68f0722015-11-11 08:30:30 +02003555 if (!cmd) {
3556 err = -ENOMEM;
Johan Hedberg04106752013-01-10 14:54:09 +02003557 goto failed;
Andre Guedesf39799f2012-02-17 20:39:35 -03003558 }
Andre Guedes3fd24152012-02-03 17:48:01 -03003559
Johan Hedberge68f0722015-11-11 08:30:30 +02003560 cmd->cmd_complete = generic_cmd_complete;
Marcel Holtmannf5a969f2014-12-04 11:36:34 +01003561
3562 hci_discovery_set_state(hdev, DISCOVERY_STARTING);
Johan Hedberge68f0722015-11-11 08:30:30 +02003563 queue_work(hdev->req_workqueue, &hdev->discov_update);
3564 err = 0;
Johan Hedberg14a53662011-04-27 10:29:56 -04003565
3566failed:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003567 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003568 return err;
3569}
3570
Johan Hedberg78b781c2016-01-05 13:19:32 +02003571static int start_discovery(struct sock *sk, struct hci_dev *hdev,
3572 void *data, u16 len)
3573{
3574 return start_discovery_internal(sk, hdev, MGMT_OP_START_DISCOVERY,
3575 data, len);
3576}
3577
3578static int start_limited_discovery(struct sock *sk, struct hci_dev *hdev,
3579 void *data, u16 len)
3580{
3581 return start_discovery_internal(sk, hdev,
3582 MGMT_OP_START_LIMITED_DISCOVERY,
3583 data, len);
3584}
3585
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003586static int service_discovery_cmd_complete(struct mgmt_pending_cmd *cmd,
3587 u8 status)
Andre Guedes1183fdc2013-04-30 15:29:35 -03003588{
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003589 return mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, status,
3590 cmd->param, 1);
Johan Hedberg2922a942014-12-05 13:36:06 +02003591}
3592
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003593static int start_service_discovery(struct sock *sk, struct hci_dev *hdev,
3594 void *data, u16 len)
3595{
3596 struct mgmt_cp_start_service_discovery *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003597 struct mgmt_pending_cmd *cmd;
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003598 const u16 max_uuid_count = ((U16_MAX - sizeof(*cp)) / 16);
3599 u16 uuid_count, expected_len;
3600 u8 status;
Andre Guedes1183fdc2013-04-30 15:29:35 -03003601 int err;
3602
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003603 BT_DBG("%s", hdev->name);
Andre Guedes1183fdc2013-04-30 15:29:35 -03003604
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003605 hci_dev_lock(hdev);
Andre Guedes1183fdc2013-04-30 15:29:35 -03003606
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003607 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003608 err = mgmt_cmd_complete(sk, hdev->id,
3609 MGMT_OP_START_SERVICE_DISCOVERY,
3610 MGMT_STATUS_NOT_POWERED,
3611 &cp->type, sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003612 goto failed;
3613 }
3614
3615 if (hdev->discovery.state != DISCOVERY_STOPPED ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07003616 hci_dev_test_flag(hdev, HCI_PERIODIC_INQ)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003617 err = mgmt_cmd_complete(sk, hdev->id,
3618 MGMT_OP_START_SERVICE_DISCOVERY,
3619 MGMT_STATUS_BUSY, &cp->type,
3620 sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003621 goto failed;
3622 }
3623
3624 uuid_count = __le16_to_cpu(cp->uuid_count);
3625 if (uuid_count > max_uuid_count) {
3626 BT_ERR("service_discovery: too big uuid_count value %u",
3627 uuid_count);
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003628 err = mgmt_cmd_complete(sk, hdev->id,
3629 MGMT_OP_START_SERVICE_DISCOVERY,
3630 MGMT_STATUS_INVALID_PARAMS, &cp->type,
3631 sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003632 goto failed;
3633 }
3634
3635 expected_len = sizeof(*cp) + uuid_count * 16;
3636 if (expected_len != len) {
3637 BT_ERR("service_discovery: expected %u bytes, got %u bytes",
3638 expected_len, len);
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003639 err = mgmt_cmd_complete(sk, hdev->id,
3640 MGMT_OP_START_SERVICE_DISCOVERY,
3641 MGMT_STATUS_INVALID_PARAMS, &cp->type,
3642 sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003643 goto failed;
3644 }
3645
Johan Hedberg591752a2015-11-11 08:11:24 +02003646 if (!discovery_type_is_valid(hdev, cp->type, &status)) {
3647 err = mgmt_cmd_complete(sk, hdev->id,
3648 MGMT_OP_START_SERVICE_DISCOVERY,
3649 status, &cp->type, sizeof(cp->type));
3650 goto failed;
3651 }
3652
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003653 cmd = mgmt_pending_add(sk, MGMT_OP_START_SERVICE_DISCOVERY,
Johan Hedberg2922a942014-12-05 13:36:06 +02003654 hdev, data, len);
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003655 if (!cmd) {
3656 err = -ENOMEM;
3657 goto failed;
3658 }
3659
Johan Hedberg2922a942014-12-05 13:36:06 +02003660 cmd->cmd_complete = service_discovery_cmd_complete;
3661
Marcel Holtmann22078802014-12-05 11:45:22 +01003662 /* Clear the discovery filter first to free any previously
3663 * allocated memory for the UUID list.
3664 */
3665 hci_discovery_filter_clear(hdev);
3666
Jakub Pawlowski82f8b652015-03-04 16:24:26 -08003667 hdev->discovery.result_filtering = true;
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003668 hdev->discovery.type = cp->type;
3669 hdev->discovery.rssi = cp->rssi;
3670 hdev->discovery.uuid_count = uuid_count;
3671
3672 if (uuid_count > 0) {
3673 hdev->discovery.uuids = kmemdup(cp->uuids, uuid_count * 16,
3674 GFP_KERNEL);
3675 if (!hdev->discovery.uuids) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003676 err = mgmt_cmd_complete(sk, hdev->id,
3677 MGMT_OP_START_SERVICE_DISCOVERY,
3678 MGMT_STATUS_FAILED,
3679 &cp->type, sizeof(cp->type));
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003680 mgmt_pending_remove(cmd);
3681 goto failed;
3682 }
3683 }
3684
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003685 hci_discovery_set_state(hdev, DISCOVERY_STARTING);
Johan Hedberge68f0722015-11-11 08:30:30 +02003686 queue_work(hdev->req_workqueue, &hdev->discov_update);
3687 err = 0;
Jakub Pawlowski66ea9422014-12-05 10:55:59 +01003688
3689failed:
3690 hci_dev_unlock(hdev);
Andre Guedes1183fdc2013-04-30 15:29:35 -03003691 return err;
3692}
3693
Johan Hedberg2154d3f2015-11-11 08:30:45 +02003694void mgmt_stop_discovery_complete(struct hci_dev *hdev, u8 status)
Andre Guedes0e05bba2013-04-30 15:29:33 -03003695{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003696 struct mgmt_pending_cmd *cmd;
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003697
Andre Guedes0e05bba2013-04-30 15:29:33 -03003698 BT_DBG("status %d", status);
3699
3700 hci_dev_lock(hdev);
3701
Johan Hedberg333ae952015-03-17 13:48:47 +02003702 cmd = pending_find(MGMT_OP_STOP_DISCOVERY, hdev);
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003703 if (cmd) {
Johan Hedberg2922a942014-12-05 13:36:06 +02003704 cmd->cmd_complete(cmd, mgmt_status(status));
Marcel Holtmann11e6e252014-12-04 11:36:35 +01003705 mgmt_pending_remove(cmd);
Andre Guedes0e05bba2013-04-30 15:29:33 -03003706 }
3707
Andre Guedes0e05bba2013-04-30 15:29:33 -03003708 hci_dev_unlock(hdev);
3709}
3710
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003711static int stop_discovery(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003712 u16 len)
Johan Hedberg14a53662011-04-27 10:29:56 -04003713{
Johan Hedbergd9306502012-02-20 23:25:18 +02003714 struct mgmt_cp_stop_discovery *mgmt_cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003715 struct mgmt_pending_cmd *cmd;
Johan Hedberg14a53662011-04-27 10:29:56 -04003716 int err;
3717
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003718 BT_DBG("%s", hdev->name);
Johan Hedberg14a53662011-04-27 10:29:56 -04003719
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003720 hci_dev_lock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003721
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003722 if (!hci_discovery_active(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003723 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
3724 MGMT_STATUS_REJECTED, &mgmt_cp->type,
3725 sizeof(mgmt_cp->type));
Johan Hedbergd9306502012-02-20 23:25:18 +02003726 goto unlock;
3727 }
3728
3729 if (hdev->discovery.type != mgmt_cp->type) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003730 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_STOP_DISCOVERY,
3731 MGMT_STATUS_INVALID_PARAMS,
3732 &mgmt_cp->type, sizeof(mgmt_cp->type));
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003733 goto unlock;
Johan Hedbergff9ef572012-01-04 14:23:45 +02003734 }
3735
Johan Hedberg2922a942014-12-05 13:36:06 +02003736 cmd = mgmt_pending_add(sk, MGMT_OP_STOP_DISCOVERY, hdev, data, len);
Johan Hedberg14a53662011-04-27 10:29:56 -04003737 if (!cmd) {
3738 err = -ENOMEM;
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003739 goto unlock;
Johan Hedberg14a53662011-04-27 10:29:56 -04003740 }
3741
Johan Hedberg2922a942014-12-05 13:36:06 +02003742 cmd->cmd_complete = generic_cmd_complete;
3743
Johan Hedberg2154d3f2015-11-11 08:30:45 +02003744 hci_discovery_set_state(hdev, DISCOVERY_STOPPING);
3745 queue_work(hdev->req_workqueue, &hdev->discov_update);
3746 err = 0;
Johan Hedberg14a53662011-04-27 10:29:56 -04003747
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003748unlock:
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003749 hci_dev_unlock(hdev);
Johan Hedberg14a53662011-04-27 10:29:56 -04003750 return err;
3751}
3752
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003753static int confirm_name(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003754 u16 len)
Johan Hedberg561aafb2012-01-04 13:31:59 +02003755{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003756 struct mgmt_cp_confirm_name *cp = data;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003757 struct inquiry_entry *e;
Johan Hedberg561aafb2012-01-04 13:31:59 +02003758 int err;
3759
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003760 BT_DBG("%s", hdev->name);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003761
Johan Hedberg561aafb2012-01-04 13:31:59 +02003762 hci_dev_lock(hdev);
3763
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003764 if (!hci_discovery_active(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003765 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
3766 MGMT_STATUS_FAILED, &cp->addr,
3767 sizeof(cp->addr));
Johan Hedberg30dc78e2012-01-04 15:44:20 +02003768 goto failed;
3769 }
3770
Johan Hedberga198e7b2012-02-17 14:27:06 +02003771 e = hci_inquiry_cache_lookup_unknown(hdev, &cp->addr.bdaddr);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003772 if (!e) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003773 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME,
3774 MGMT_STATUS_INVALID_PARAMS, &cp->addr,
3775 sizeof(cp->addr));
Johan Hedberg561aafb2012-01-04 13:31:59 +02003776 goto failed;
3777 }
3778
3779 if (cp->name_known) {
3780 e->name_state = NAME_KNOWN;
3781 list_del(&e->list);
3782 } else {
3783 e->name_state = NAME_NEEDED;
Johan Hedberga3d4e20a2012-01-09 00:53:02 +02003784 hci_inquiry_cache_update_resolve(hdev, e);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003785 }
3786
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003787 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_CONFIRM_NAME, 0,
3788 &cp->addr, sizeof(cp->addr));
Johan Hedberg561aafb2012-01-04 13:31:59 +02003789
3790failed:
3791 hci_dev_unlock(hdev);
Johan Hedberg561aafb2012-01-04 13:31:59 +02003792 return err;
3793}
3794
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003795static int block_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003796 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003797{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003798 struct mgmt_cp_block_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003799 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003800 int err;
3801
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003802 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003803
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003804 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003805 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE,
3806 MGMT_STATUS_INVALID_PARAMS,
3807 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003808
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003809 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003810
Johan Hedbergdcc36c12014-07-09 12:59:13 +03003811 err = hci_bdaddr_list_add(&hdev->blacklist, &cp->addr.bdaddr,
3812 cp->addr.type);
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003813 if (err < 0) {
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003814 status = MGMT_STATUS_FAILED;
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003815 goto done;
3816 }
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003817
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003818 mgmt_event(MGMT_EV_DEVICE_BLOCKED, hdev, &cp->addr, sizeof(cp->addr),
3819 sk);
3820 status = MGMT_STATUS_SUCCESS;
3821
3822done:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003823 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_BLOCK_DEVICE, status,
3824 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003825
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003826 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003827
3828 return err;
3829}
3830
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003831static int unblock_device(struct sock *sk, struct hci_dev *hdev, void *data,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03003832 u16 len)
Antti Julku7fbec222011-06-15 12:01:15 +03003833{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03003834 struct mgmt_cp_unblock_device *cp = data;
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003835 u8 status;
Antti Julku7fbec222011-06-15 12:01:15 +03003836 int err;
3837
Johan Hedbergbdb6d972012-02-28 06:13:32 +02003838 BT_DBG("%s", hdev->name);
Antti Julku7fbec222011-06-15 12:01:15 +03003839
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003840 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003841 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE,
3842 MGMT_STATUS_INVALID_PARAMS,
3843 &cp->addr, sizeof(cp->addr));
Johan Hedberg4ee71b22013-01-20 14:27:19 +02003844
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003845 hci_dev_lock(hdev);
Antti Julku5e762442011-08-25 16:48:02 +03003846
Johan Hedbergdcc36c12014-07-09 12:59:13 +03003847 err = hci_bdaddr_list_del(&hdev->blacklist, &cp->addr.bdaddr,
3848 cp->addr.type);
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003849 if (err < 0) {
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003850 status = MGMT_STATUS_INVALID_PARAMS;
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003851 goto done;
3852 }
Johan Hedbergf0eeea82012-02-19 12:58:54 +02003853
Johan Hedberg2a8357f2014-07-01 22:09:47 +03003854 mgmt_event(MGMT_EV_DEVICE_UNBLOCKED, hdev, &cp->addr, sizeof(cp->addr),
3855 sk);
3856 status = MGMT_STATUS_SUCCESS;
3857
3858done:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003859 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_UNBLOCK_DEVICE, status,
3860 &cp->addr, sizeof(cp->addr));
Antti Julku5e762442011-08-25 16:48:02 +03003861
Gustavo F. Padovan09fd0de2011-06-17 13:03:21 -03003862 hci_dev_unlock(hdev);
Antti Julku7fbec222011-06-15 12:01:15 +03003863
3864 return err;
3865}
3866
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003867static int set_device_id(struct sock *sk, struct hci_dev *hdev, void *data,
3868 u16 len)
3869{
3870 struct mgmt_cp_set_device_id *cp = data;
Johan Hedberg890ea892013-03-15 17:06:52 -05003871 struct hci_request req;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003872 int err;
Szymon Jancc72d4b82012-03-16 16:02:57 +01003873 __u16 source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003874
3875 BT_DBG("%s", hdev->name);
3876
Szymon Jancc72d4b82012-03-16 16:02:57 +01003877 source = __le16_to_cpu(cp->source);
3878
3879 if (source > 0x0002)
Johan Hedberga69e8372015-03-06 21:08:53 +02003880 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DEVICE_ID,
3881 MGMT_STATUS_INVALID_PARAMS);
Szymon Jancc72d4b82012-03-16 16:02:57 +01003882
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003883 hci_dev_lock(hdev);
3884
Szymon Jancc72d4b82012-03-16 16:02:57 +01003885 hdev->devid_source = source;
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003886 hdev->devid_vendor = __le16_to_cpu(cp->vendor);
3887 hdev->devid_product = __le16_to_cpu(cp->product);
3888 hdev->devid_version = __le16_to_cpu(cp->version);
3889
Johan Hedberg2a1afb52015-03-06 21:08:54 +02003890 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_DEVICE_ID, 0,
3891 NULL, 0);
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003892
Johan Hedberg890ea892013-03-15 17:06:52 -05003893 hci_req_init(&req, hdev);
Johan Hedbergb1a89172015-11-25 16:15:42 +02003894 __hci_req_update_eir(&req);
Johan Hedberg890ea892013-03-15 17:06:52 -05003895 hci_req_run(&req, NULL);
Marcel Holtmanncdbaccc2012-03-11 20:00:29 -07003896
3897 hci_dev_unlock(hdev);
3898
3899 return err;
3900}
3901
Arman Uguray24b4f382015-03-23 15:57:12 -07003902static void enable_advertising_instance(struct hci_dev *hdev, u8 status,
3903 u16 opcode)
3904{
3905 BT_DBG("status %d", status);
3906}
3907
Marcel Holtmann1904a852015-01-11 13:50:44 -08003908static void set_advertising_complete(struct hci_dev *hdev, u8 status,
3909 u16 opcode)
Johan Hedberg4375f102013-09-25 13:26:10 +03003910{
3911 struct cmd_lookup match = { NULL, hdev };
Arman Uguray24b4f382015-03-23 15:57:12 -07003912 struct hci_request req;
Florian Grandel7816b822015-06-18 03:16:45 +02003913 u8 instance;
3914 struct adv_info *adv_instance;
3915 int err;
Johan Hedberg4375f102013-09-25 13:26:10 +03003916
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303917 hci_dev_lock(hdev);
3918
Johan Hedberg4375f102013-09-25 13:26:10 +03003919 if (status) {
3920 u8 mgmt_err = mgmt_status(status);
3921
3922 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev,
3923 cmd_status_rsp, &mgmt_err);
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303924 goto unlock;
Johan Hedberg4375f102013-09-25 13:26:10 +03003925 }
3926
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07003927 if (hci_dev_test_flag(hdev, HCI_LE_ADV))
Marcel Holtmanna1536da2015-03-13 02:11:01 -07003928 hci_dev_set_flag(hdev, HCI_ADVERTISING);
Johan Hedbergc93bd152014-07-08 15:07:48 +03003929 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07003930 hci_dev_clear_flag(hdev, HCI_ADVERTISING);
Johan Hedbergc93bd152014-07-08 15:07:48 +03003931
Johan Hedberg4375f102013-09-25 13:26:10 +03003932 mgmt_pending_foreach(MGMT_OP_SET_ADVERTISING, hdev, settings_rsp,
3933 &match);
3934
3935 new_settings(hdev, match.sk);
3936
3937 if (match.sk)
3938 sock_put(match.sk);
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303939
Arman Uguray24b4f382015-03-23 15:57:12 -07003940 /* If "Set Advertising" was just disabled and instance advertising was
Florian Grandel7816b822015-06-18 03:16:45 +02003941 * set up earlier, then re-enable multi-instance advertising.
Arman Uguray24b4f382015-03-23 15:57:12 -07003942 */
3943 if (hci_dev_test_flag(hdev, HCI_ADVERTISING) ||
Florian Grandel7816b822015-06-18 03:16:45 +02003944 list_empty(&hdev->adv_instances))
Arman Uguray24b4f382015-03-23 15:57:12 -07003945 goto unlock;
3946
Florian Grandel7816b822015-06-18 03:16:45 +02003947 instance = hdev->cur_adv_instance;
3948 if (!instance) {
3949 adv_instance = list_first_entry_or_null(&hdev->adv_instances,
3950 struct adv_info, list);
3951 if (!adv_instance)
3952 goto unlock;
3953
3954 instance = adv_instance->instance;
3955 }
3956
Arman Uguray24b4f382015-03-23 15:57:12 -07003957 hci_req_init(&req, hdev);
3958
Johan Hedbergf2252572015-11-18 12:49:20 +02003959 err = __hci_req_schedule_adv_instance(&req, instance, true);
Arman Uguray24b4f382015-03-23 15:57:12 -07003960
Florian Grandel7816b822015-06-18 03:16:45 +02003961 if (!err)
3962 err = hci_req_run(&req, enable_advertising_instance);
3963
3964 if (err)
Arman Uguray24b4f382015-03-23 15:57:12 -07003965 BT_ERR("Failed to re-configure advertising");
3966
Jaganath Kanakkassery3ad67582014-12-11 11:43:12 +05303967unlock:
3968 hci_dev_unlock(hdev);
Johan Hedberg4375f102013-09-25 13:26:10 +03003969}
3970
Marcel Holtmann21b51872013-10-10 09:47:53 -07003971static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
3972 u16 len)
Johan Hedberg4375f102013-09-25 13:26:10 +03003973{
3974 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02003975 struct mgmt_pending_cmd *cmd;
Johan Hedberg4375f102013-09-25 13:26:10 +03003976 struct hci_request req;
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07003977 u8 val, status;
Johan Hedberg4375f102013-09-25 13:26:10 +03003978 int err;
3979
3980 BT_DBG("request for %s", hdev->name);
3981
Johan Hedberge6fe7982013-10-02 15:45:22 +03003982 status = mgmt_le_support(hdev);
3983 if (status)
Johan Hedberga69e8372015-03-06 21:08:53 +02003984 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3985 status);
Johan Hedberg4375f102013-09-25 13:26:10 +03003986
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07003987 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02003988 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
3989 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg4375f102013-09-25 13:26:10 +03003990
3991 hci_dev_lock(hdev);
3992
3993 val = !!cp->val;
Johan Hedberg4375f102013-09-25 13:26:10 +03003994
Johan Hedbergf74ca9b2013-10-08 15:52:18 +02003995 /* The following conditions are ones which mean that we should
3996 * not do any HCI communication but directly send a mgmt
3997 * response to user space (after toggling the flag if
3998 * necessary).
3999 */
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004000 if (!hdev_is_powered(hdev) ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004001 (val == hci_dev_test_flag(hdev, HCI_ADVERTISING) &&
4002 (cp->val == 0x02) == hci_dev_test_flag(hdev, HCI_ADVERTISING_CONNECTABLE)) ||
Johan Hedberge8bb6b92014-07-08 15:07:53 +03004003 hci_conn_num(hdev, LE_LINK) > 0 ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004004 (hci_dev_test_flag(hdev, HCI_LE_SCAN) &&
Johan Hedberge8bb6b92014-07-08 15:07:53 +03004005 hdev->le_scan_type == LE_SCAN_ACTIVE)) {
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004006 bool changed;
Johan Hedberg4375f102013-09-25 13:26:10 +03004007
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004008 if (cp->val) {
Johan Hedbergcab054a2015-11-30 11:21:45 +02004009 hdev->cur_adv_instance = 0x00;
Marcel Holtmann238be782015-03-13 02:11:06 -07004010 changed = !hci_dev_test_and_set_flag(hdev, HCI_ADVERTISING);
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004011 if (cp->val == 0x02)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004012 hci_dev_set_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004013 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004014 hci_dev_clear_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004015 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004016 changed = hci_dev_test_and_clear_flag(hdev, HCI_ADVERTISING);
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004017 hci_dev_clear_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Johan Hedberg4375f102013-09-25 13:26:10 +03004018 }
4019
4020 err = send_settings_rsp(sk, MGMT_OP_SET_ADVERTISING, hdev);
4021 if (err < 0)
4022 goto unlock;
4023
4024 if (changed)
4025 err = new_settings(hdev, sk);
4026
4027 goto unlock;
4028 }
4029
Johan Hedberg333ae952015-03-17 13:48:47 +02004030 if (pending_find(MGMT_OP_SET_ADVERTISING, hdev) ||
4031 pending_find(MGMT_OP_SET_LE, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004032 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
4033 MGMT_STATUS_BUSY);
Johan Hedberg4375f102013-09-25 13:26:10 +03004034 goto unlock;
4035 }
4036
4037 cmd = mgmt_pending_add(sk, MGMT_OP_SET_ADVERTISING, hdev, data, len);
4038 if (!cmd) {
4039 err = -ENOMEM;
4040 goto unlock;
4041 }
4042
4043 hci_req_init(&req, hdev);
4044
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004045 if (cp->val == 0x02)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004046 hci_dev_set_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004047 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004048 hci_dev_clear_flag(hdev, HCI_ADVERTISING_CONNECTABLE);
Marcel Holtmanncc91cb042015-03-12 22:30:58 -07004049
Florian Grandel7816b822015-06-18 03:16:45 +02004050 cancel_adv_timeout(hdev);
4051
Arman Uguray24b4f382015-03-23 15:57:12 -07004052 if (val) {
Florian Grandel7816b822015-06-18 03:16:45 +02004053 /* Switch to instance "0" for the Set Advertising setting.
4054 * We cannot use update_[adv|scan_rsp]_data() here as the
4055 * HCI_ADVERTISING flag is not yet set.
4056 */
Johan Hedbergcab054a2015-11-30 11:21:45 +02004057 hdev->cur_adv_instance = 0x00;
Johan Hedbergf2252572015-11-18 12:49:20 +02004058 __hci_req_update_adv_data(&req, 0x00);
4059 __hci_req_update_scan_rsp_data(&req, 0x00);
4060 __hci_req_enable_advertising(&req);
Arman Uguray24b4f382015-03-23 15:57:12 -07004061 } else {
Johan Hedbergf2252572015-11-18 12:49:20 +02004062 __hci_req_disable_advertising(&req);
Arman Uguray24b4f382015-03-23 15:57:12 -07004063 }
Johan Hedberg4375f102013-09-25 13:26:10 +03004064
4065 err = hci_req_run(&req, set_advertising_complete);
4066 if (err < 0)
4067 mgmt_pending_remove(cmd);
4068
4069unlock:
4070 hci_dev_unlock(hdev);
4071 return err;
4072}
4073
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004074static int set_static_address(struct sock *sk, struct hci_dev *hdev,
4075 void *data, u16 len)
4076{
4077 struct mgmt_cp_set_static_address *cp = data;
4078 int err;
4079
4080 BT_DBG("%s", hdev->name);
4081
Marcel Holtmann62af4442013-10-02 22:10:32 -07004082 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004083 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
4084 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004085
4086 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004087 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_STATIC_ADDRESS,
4088 MGMT_STATUS_REJECTED);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004089
4090 if (bacmp(&cp->bdaddr, BDADDR_ANY)) {
4091 if (!bacmp(&cp->bdaddr, BDADDR_NONE))
Johan Hedberga69e8372015-03-06 21:08:53 +02004092 return mgmt_cmd_status(sk, hdev->id,
4093 MGMT_OP_SET_STATIC_ADDRESS,
4094 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004095
4096 /* Two most significant bits shall be set */
4097 if ((cp->bdaddr.b[5] & 0xc0) != 0xc0)
Johan Hedberga69e8372015-03-06 21:08:53 +02004098 return mgmt_cmd_status(sk, hdev->id,
4099 MGMT_OP_SET_STATIC_ADDRESS,
4100 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004101 }
4102
4103 hci_dev_lock(hdev);
4104
4105 bacpy(&hdev->static_addr, &cp->bdaddr);
4106
Marcel Holtmann93690c22015-03-06 10:11:21 -08004107 err = send_settings_rsp(sk, MGMT_OP_SET_STATIC_ADDRESS, hdev);
4108 if (err < 0)
4109 goto unlock;
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004110
Marcel Holtmann93690c22015-03-06 10:11:21 -08004111 err = new_settings(hdev, sk);
4112
4113unlock:
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004114 hci_dev_unlock(hdev);
Marcel Holtmannd13eafc2013-10-02 04:41:30 -07004115 return err;
4116}
4117
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004118static int set_scan_params(struct sock *sk, struct hci_dev *hdev,
4119 void *data, u16 len)
4120{
4121 struct mgmt_cp_set_scan_params *cp = data;
4122 __u16 interval, window;
4123 int err;
4124
4125 BT_DBG("%s", hdev->name);
4126
4127 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004128 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
4129 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004130
4131 interval = __le16_to_cpu(cp->interval);
4132
4133 if (interval < 0x0004 || interval > 0x4000)
Johan Hedberga69e8372015-03-06 21:08:53 +02004134 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
4135 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004136
4137 window = __le16_to_cpu(cp->window);
4138
4139 if (window < 0x0004 || window > 0x4000)
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 Holtmann14b49b92013-10-11 08:23:20 -07004142
Marcel Holtmann899e1072013-10-14 09:55:32 -07004143 if (window > interval)
Johan Hedberga69e8372015-03-06 21:08:53 +02004144 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS,
4145 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann899e1072013-10-14 09:55:32 -07004146
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004147 hci_dev_lock(hdev);
4148
4149 hdev->le_scan_interval = interval;
4150 hdev->le_scan_window = window;
4151
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004152 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_SET_SCAN_PARAMS, 0,
4153 NULL, 0);
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004154
Andre Guedesdd2ef8e2014-02-26 20:21:56 -03004155 /* If background scan is running, restart it so new parameters are
4156 * loaded.
4157 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004158 if (hci_dev_test_flag(hdev, HCI_LE_SCAN) &&
Andre Guedesdd2ef8e2014-02-26 20:21:56 -03004159 hdev->discovery.state == DISCOVERY_STOPPED) {
4160 struct hci_request req;
4161
4162 hci_req_init(&req, hdev);
4163
4164 hci_req_add_le_scan_disable(&req);
4165 hci_req_add_le_passive_scan(&req);
4166
4167 hci_req_run(&req, NULL);
4168 }
4169
Marcel Holtmann14b49b92013-10-11 08:23:20 -07004170 hci_dev_unlock(hdev);
4171
4172 return err;
4173}
4174
Marcel Holtmann1904a852015-01-11 13:50:44 -08004175static void fast_connectable_complete(struct hci_dev *hdev, u8 status,
4176 u16 opcode)
Johan Hedberg33e38b32013-03-15 17:07:05 -05004177{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004178 struct mgmt_pending_cmd *cmd;
Johan Hedberg33e38b32013-03-15 17:07:05 -05004179
4180 BT_DBG("status 0x%02x", status);
4181
4182 hci_dev_lock(hdev);
4183
Johan Hedberg333ae952015-03-17 13:48:47 +02004184 cmd = pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004185 if (!cmd)
4186 goto unlock;
4187
4188 if (status) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004189 mgmt_cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4190 mgmt_status(status));
Johan Hedberg33e38b32013-03-15 17:07:05 -05004191 } else {
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004192 struct mgmt_mode *cp = cmd->param;
4193
4194 if (cp->val)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004195 hci_dev_set_flag(hdev, HCI_FAST_CONNECTABLE);
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004196 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004197 hci_dev_clear_flag(hdev, HCI_FAST_CONNECTABLE);
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004198
Johan Hedberg33e38b32013-03-15 17:07:05 -05004199 send_settings_rsp(cmd->sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev);
4200 new_settings(hdev, cmd->sk);
4201 }
4202
4203 mgmt_pending_remove(cmd);
4204
4205unlock:
4206 hci_dev_unlock(hdev);
4207}
4208
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004209static int set_fast_connectable(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004210 void *data, u16 len)
Antti Julkuf6422ec2011-06-22 13:11:56 +03004211{
Vinicius Costa Gomes650f7262012-02-02 21:07:59 -03004212 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004213 struct mgmt_pending_cmd *cmd;
Johan Hedberg33e38b32013-03-15 17:07:05 -05004214 struct hci_request req;
Antti Julkuf6422ec2011-06-22 13:11:56 +03004215 int err;
4216
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004217 BT_DBG("%s", hdev->name);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004218
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004219 if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) ||
Johan Hedberg56f87902013-10-02 13:43:13 +03004220 hdev->hci_ver < BLUETOOTH_VER_1_2)
Johan Hedberga69e8372015-03-06 21:08:53 +02004221 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4222 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg33c525c2012-10-24 21:11:58 +03004223
Johan Hedberga7e80f22013-01-09 16:05:19 +02004224 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02004225 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4226 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga7e80f22013-01-09 16:05:19 +02004227
Antti Julkuf6422ec2011-06-22 13:11:56 +03004228 hci_dev_lock(hdev);
4229
Johan Hedberg333ae952015-03-17 13:48:47 +02004230 if (pending_find(MGMT_OP_SET_FAST_CONNECTABLE, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004231 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4232 MGMT_STATUS_BUSY);
Johan Hedberg05cbf292013-03-15 17:07:07 -05004233 goto unlock;
4234 }
4235
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004236 if (!!cp->val == hci_dev_test_flag(hdev, HCI_FAST_CONNECTABLE)) {
Johan Hedberg1a4d3c42013-03-15 17:07:08 -05004237 err = send_settings_rsp(sk, MGMT_OP_SET_FAST_CONNECTABLE,
4238 hdev);
4239 goto unlock;
4240 }
4241
Johan Hedberg406ef2a2015-03-10 20:14:27 +02004242 if (!hdev_is_powered(hdev)) {
Marcel Holtmannce05d602015-03-13 02:11:03 -07004243 hci_dev_change_flag(hdev, HCI_FAST_CONNECTABLE);
Johan Hedberg406ef2a2015-03-10 20:14:27 +02004244 err = send_settings_rsp(sk, MGMT_OP_SET_FAST_CONNECTABLE,
4245 hdev);
4246 new_settings(hdev, sk);
4247 goto unlock;
4248 }
4249
Johan Hedberg33e38b32013-03-15 17:07:05 -05004250 cmd = mgmt_pending_add(sk, MGMT_OP_SET_FAST_CONNECTABLE, hdev,
4251 data, len);
4252 if (!cmd) {
4253 err = -ENOMEM;
4254 goto unlock;
4255 }
4256
4257 hci_req_init(&req, hdev);
4258
Johan Hedbergbf943cb2015-11-25 16:15:43 +02004259 __hci_req_write_fast_connectable(&req, cp->val);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004260
4261 err = hci_req_run(&req, fast_connectable_complete);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004262 if (err < 0) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004263 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_FAST_CONNECTABLE,
4264 MGMT_STATUS_FAILED);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004265 mgmt_pending_remove(cmd);
Antti Julkuf6422ec2011-06-22 13:11:56 +03004266 }
4267
Johan Hedberg33e38b32013-03-15 17:07:05 -05004268unlock:
Antti Julkuf6422ec2011-06-22 13:11:56 +03004269 hci_dev_unlock(hdev);
Johan Hedberg33e38b32013-03-15 17:07:05 -05004270
Antti Julkuf6422ec2011-06-22 13:11:56 +03004271 return err;
4272}
4273
Marcel Holtmann1904a852015-01-11 13:50:44 -08004274static void set_bredr_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg0663ca22013-10-02 13:43:14 +03004275{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004276 struct mgmt_pending_cmd *cmd;
Johan Hedberg0663ca22013-10-02 13:43:14 +03004277
4278 BT_DBG("status 0x%02x", status);
4279
4280 hci_dev_lock(hdev);
4281
Johan Hedberg333ae952015-03-17 13:48:47 +02004282 cmd = pending_find(MGMT_OP_SET_BREDR, hdev);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004283 if (!cmd)
4284 goto unlock;
4285
4286 if (status) {
4287 u8 mgmt_err = mgmt_status(status);
4288
4289 /* We need to restore the flag if related HCI commands
4290 * failed.
4291 */
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004292 hci_dev_clear_flag(hdev, HCI_BREDR_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004293
Johan Hedberga69e8372015-03-06 21:08:53 +02004294 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode, mgmt_err);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004295 } else {
4296 send_settings_rsp(cmd->sk, MGMT_OP_SET_BREDR, hdev);
4297 new_settings(hdev, cmd->sk);
4298 }
4299
4300 mgmt_pending_remove(cmd);
4301
4302unlock:
4303 hci_dev_unlock(hdev);
4304}
4305
4306static int set_bredr(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)
4307{
4308 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004309 struct mgmt_pending_cmd *cmd;
Johan Hedberg0663ca22013-10-02 13:43:14 +03004310 struct hci_request req;
4311 int err;
4312
4313 BT_DBG("request for %s", hdev->name);
4314
4315 if (!lmp_bredr_capable(hdev) || !lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004316 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4317 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004318
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004319 if (!hci_dev_test_flag(hdev, HCI_LE_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02004320 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4321 MGMT_STATUS_REJECTED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004322
4323 if (cp->val != 0x00 && cp->val != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02004324 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4325 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004326
4327 hci_dev_lock(hdev);
4328
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004329 if (cp->val == hci_dev_test_flag(hdev, HCI_BREDR_ENABLED)) {
Johan Hedberg0663ca22013-10-02 13:43:14 +03004330 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4331 goto unlock;
4332 }
4333
4334 if (!hdev_is_powered(hdev)) {
4335 if (!cp->val) {
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004336 hci_dev_clear_flag(hdev, HCI_DISCOVERABLE);
4337 hci_dev_clear_flag(hdev, HCI_SSP_ENABLED);
4338 hci_dev_clear_flag(hdev, HCI_LINK_SECURITY);
4339 hci_dev_clear_flag(hdev, HCI_FAST_CONNECTABLE);
4340 hci_dev_clear_flag(hdev, HCI_HS_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004341 }
4342
Marcel Holtmannce05d602015-03-13 02:11:03 -07004343 hci_dev_change_flag(hdev, HCI_BREDR_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004344
4345 err = send_settings_rsp(sk, MGMT_OP_SET_BREDR, hdev);
4346 if (err < 0)
4347 goto unlock;
4348
4349 err = new_settings(hdev, sk);
4350 goto unlock;
4351 }
4352
4353 /* Reject disabling when powered on */
4354 if (!cp->val) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004355 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4356 MGMT_STATUS_REJECTED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004357 goto unlock;
Marcel Holtmann111e4bc2015-01-14 14:40:42 -08004358 } else {
4359 /* When configuring a dual-mode controller to operate
4360 * with LE only and using a static address, then switching
4361 * BR/EDR back on is not allowed.
4362 *
4363 * Dual-mode controllers shall operate with the public
4364 * address as its identity address for BR/EDR and LE. So
4365 * reject the attempt to create an invalid configuration.
Marcel Holtmann3a5486e2015-01-22 11:15:21 -08004366 *
4367 * The same restrictions applies when secure connections
4368 * has been enabled. For BR/EDR this is a controller feature
4369 * while for LE it is a host stack feature. This means that
4370 * switching BR/EDR back on when secure connections has been
4371 * enabled is not a supported transaction.
Marcel Holtmann111e4bc2015-01-14 14:40:42 -08004372 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004373 if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) &&
Marcel Holtmann3a5486e2015-01-22 11:15:21 -08004374 (bacmp(&hdev->static_addr, BDADDR_ANY) ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004375 hci_dev_test_flag(hdev, HCI_SC_ENABLED))) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004376 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4377 MGMT_STATUS_REJECTED);
Marcel Holtmann111e4bc2015-01-14 14:40:42 -08004378 goto unlock;
4379 }
Johan Hedberg0663ca22013-10-02 13:43:14 +03004380 }
4381
Johan Hedberg333ae952015-03-17 13:48:47 +02004382 if (pending_find(MGMT_OP_SET_BREDR, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004383 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_BREDR,
4384 MGMT_STATUS_BUSY);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004385 goto unlock;
4386 }
4387
4388 cmd = mgmt_pending_add(sk, MGMT_OP_SET_BREDR, hdev, data, len);
4389 if (!cmd) {
4390 err = -ENOMEM;
4391 goto unlock;
4392 }
4393
Johan Hedbergf2252572015-11-18 12:49:20 +02004394 /* We need to flip the bit already here so that
4395 * hci_req_update_adv_data generates the correct flags.
Johan Hedberg0663ca22013-10-02 13:43:14 +03004396 */
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004397 hci_dev_set_flag(hdev, HCI_BREDR_ENABLED);
Johan Hedberg0663ca22013-10-02 13:43:14 +03004398
4399 hci_req_init(&req, hdev);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004400
Johan Hedbergbf943cb2015-11-25 16:15:43 +02004401 __hci_req_write_fast_connectable(&req, false);
Johan Hedberg01b1cb82015-11-16 12:52:21 +02004402 __hci_req_update_scan(&req);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004403
Marcel Holtmannf14d8f62013-10-16 00:16:48 -07004404 /* Since only the advertising data flags will change, there
4405 * is no need to update the scan response data.
4406 */
Johan Hedbergcab054a2015-11-30 11:21:45 +02004407 __hci_req_update_adv_data(&req, hdev->cur_adv_instance);
Johan Hedbergaa8af462013-10-14 21:15:26 +03004408
Johan Hedberg0663ca22013-10-02 13:43:14 +03004409 err = hci_req_run(&req, set_bredr_complete);
4410 if (err < 0)
4411 mgmt_pending_remove(cmd);
4412
4413unlock:
4414 hci_dev_unlock(hdev);
4415 return err;
4416}
4417
Johan Hedberga1443f52015-01-23 15:42:46 +02004418static void sc_enable_complete(struct hci_dev *hdev, u8 status, u16 opcode)
4419{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004420 struct mgmt_pending_cmd *cmd;
Johan Hedberga1443f52015-01-23 15:42:46 +02004421 struct mgmt_mode *cp;
4422
4423 BT_DBG("%s status %u", hdev->name, status);
4424
4425 hci_dev_lock(hdev);
4426
Johan Hedberg333ae952015-03-17 13:48:47 +02004427 cmd = pending_find(MGMT_OP_SET_SECURE_CONN, hdev);
Johan Hedberga1443f52015-01-23 15:42:46 +02004428 if (!cmd)
4429 goto unlock;
4430
4431 if (status) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004432 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode,
4433 mgmt_status(status));
Johan Hedberga1443f52015-01-23 15:42:46 +02004434 goto remove;
4435 }
4436
4437 cp = cmd->param;
4438
4439 switch (cp->val) {
4440 case 0x00:
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004441 hci_dev_clear_flag(hdev, HCI_SC_ENABLED);
4442 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Johan Hedberga1443f52015-01-23 15:42:46 +02004443 break;
4444 case 0x01:
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004445 hci_dev_set_flag(hdev, HCI_SC_ENABLED);
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004446 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Johan Hedberga1443f52015-01-23 15:42:46 +02004447 break;
4448 case 0x02:
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004449 hci_dev_set_flag(hdev, HCI_SC_ENABLED);
4450 hci_dev_set_flag(hdev, HCI_SC_ONLY);
Johan Hedberga1443f52015-01-23 15:42:46 +02004451 break;
4452 }
4453
4454 send_settings_rsp(cmd->sk, MGMT_OP_SET_SECURE_CONN, hdev);
4455 new_settings(hdev, cmd->sk);
4456
4457remove:
4458 mgmt_pending_remove(cmd);
4459unlock:
4460 hci_dev_unlock(hdev);
4461}
4462
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004463static int set_secure_conn(struct sock *sk, struct hci_dev *hdev,
4464 void *data, u16 len)
4465{
4466 struct mgmt_mode *cp = data;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004467 struct mgmt_pending_cmd *cmd;
Johan Hedberga1443f52015-01-23 15:42:46 +02004468 struct hci_request req;
Johan Hedberga3209692014-05-26 11:23:35 +03004469 u8 val;
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004470 int err;
4471
4472 BT_DBG("request for %s", hdev->name);
4473
Marcel Holtmann05b3c3e2014-12-31 14:43:18 -08004474 if (!lmp_sc_capable(hdev) &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004475 !hci_dev_test_flag(hdev, HCI_LE_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02004476 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4477 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004478
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004479 if (hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) &&
Johan Hedberg59200282015-01-28 19:56:00 +02004480 lmp_sc_capable(hdev) &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004481 !hci_dev_test_flag(hdev, HCI_SSP_ENABLED))
Johan Hedberga69e8372015-03-06 21:08:53 +02004482 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4483 MGMT_STATUS_REJECTED);
Marcel Holtmanned93ec62015-01-22 11:15:22 -08004484
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004485 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02004486 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004487 MGMT_STATUS_INVALID_PARAMS);
4488
4489 hci_dev_lock(hdev);
4490
Marcel Holtmann05b3c3e2014-12-31 14:43:18 -08004491 if (!hdev_is_powered(hdev) || !lmp_sc_capable(hdev) ||
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004492 !hci_dev_test_flag(hdev, HCI_BREDR_ENABLED)) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004493 bool changed;
4494
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004495 if (cp->val) {
Marcel Holtmann238be782015-03-13 02:11:06 -07004496 changed = !hci_dev_test_and_set_flag(hdev,
4497 HCI_SC_ENABLED);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004498 if (cp->val == 0x02)
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004499 hci_dev_set_flag(hdev, HCI_SC_ONLY);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004500 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004501 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004502 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004503 changed = hci_dev_test_and_clear_flag(hdev,
4504 HCI_SC_ENABLED);
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004505 hci_dev_clear_flag(hdev, HCI_SC_ONLY);
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004506 }
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004507
4508 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4509 if (err < 0)
4510 goto failed;
4511
4512 if (changed)
4513 err = new_settings(hdev, sk);
4514
4515 goto failed;
4516 }
4517
Johan Hedberg333ae952015-03-17 13:48:47 +02004518 if (pending_find(MGMT_OP_SET_SECURE_CONN, hdev)) {
Johan Hedberga69e8372015-03-06 21:08:53 +02004519 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_SECURE_CONN,
4520 MGMT_STATUS_BUSY);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004521 goto failed;
4522 }
4523
Marcel Holtmann0ab04a92014-02-01 09:19:57 -08004524 val = !!cp->val;
4525
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004526 if (val == hci_dev_test_flag(hdev, HCI_SC_ENABLED) &&
4527 (cp->val == 0x02) == hci_dev_test_flag(hdev, HCI_SC_ONLY)) {
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004528 err = send_settings_rsp(sk, MGMT_OP_SET_SECURE_CONN, hdev);
4529 goto failed;
4530 }
4531
4532 cmd = mgmt_pending_add(sk, MGMT_OP_SET_SECURE_CONN, hdev, data, len);
4533 if (!cmd) {
4534 err = -ENOMEM;
4535 goto failed;
4536 }
4537
Johan Hedberga1443f52015-01-23 15:42:46 +02004538 hci_req_init(&req, hdev);
4539 hci_req_add(&req, HCI_OP_WRITE_SC_SUPPORT, 1, &val);
4540 err = hci_req_run(&req, sc_enable_complete);
Marcel Holtmanneac83dc2014-01-10 02:07:23 -08004541 if (err < 0) {
4542 mgmt_pending_remove(cmd);
4543 goto failed;
4544 }
4545
4546failed:
4547 hci_dev_unlock(hdev);
4548 return err;
4549}
4550
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004551static int set_debug_keys(struct sock *sk, struct hci_dev *hdev,
4552 void *data, u16 len)
4553{
4554 struct mgmt_mode *cp = data;
Johan Hedbergb97109792014-06-24 14:00:28 +03004555 bool changed, use_changed;
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004556 int err;
4557
4558 BT_DBG("request for %s", hdev->name);
4559
Johan Hedbergb97109792014-06-24 14:00:28 +03004560 if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02004561 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_DEBUG_KEYS,
4562 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004563
4564 hci_dev_lock(hdev);
4565
4566 if (cp->val)
Marcel Holtmann238be782015-03-13 02:11:06 -07004567 changed = !hci_dev_test_and_set_flag(hdev, HCI_KEEP_DEBUG_KEYS);
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004568 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004569 changed = hci_dev_test_and_clear_flag(hdev,
4570 HCI_KEEP_DEBUG_KEYS);
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004571
Johan Hedbergb97109792014-06-24 14:00:28 +03004572 if (cp->val == 0x02)
Marcel Holtmann238be782015-03-13 02:11:06 -07004573 use_changed = !hci_dev_test_and_set_flag(hdev,
4574 HCI_USE_DEBUG_KEYS);
Johan Hedbergb97109792014-06-24 14:00:28 +03004575 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004576 use_changed = hci_dev_test_and_clear_flag(hdev,
4577 HCI_USE_DEBUG_KEYS);
Johan Hedbergb97109792014-06-24 14:00:28 +03004578
4579 if (hdev_is_powered(hdev) && use_changed &&
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07004580 hci_dev_test_flag(hdev, HCI_SSP_ENABLED)) {
Johan Hedbergb97109792014-06-24 14:00:28 +03004581 u8 mode = (cp->val == 0x02) ? 0x01 : 0x00;
4582 hci_send_cmd(hdev, HCI_OP_WRITE_SSP_DEBUG_MODE,
4583 sizeof(mode), &mode);
4584 }
4585
Marcel Holtmann4e39ac82014-01-31 11:55:22 -08004586 err = send_settings_rsp(sk, MGMT_OP_SET_DEBUG_KEYS, hdev);
4587 if (err < 0)
4588 goto unlock;
4589
4590 if (changed)
4591 err = new_settings(hdev, sk);
4592
4593unlock:
4594 hci_dev_unlock(hdev);
4595 return err;
4596}
4597
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004598static int set_privacy(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4599 u16 len)
4600{
4601 struct mgmt_cp_set_privacy *cp = cp_data;
4602 bool changed;
4603 int err;
4604
4605 BT_DBG("request for %s", hdev->name);
4606
4607 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004608 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4609 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004610
Johan Hedberg82a37ad2016-03-09 17:30:34 +02004611 if (cp->privacy != 0x00 && cp->privacy != 0x01 && cp->privacy != 0x02)
Johan Hedberga69e8372015-03-06 21:08:53 +02004612 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4613 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004614
4615 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004616 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PRIVACY,
4617 MGMT_STATUS_REJECTED);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004618
4619 hci_dev_lock(hdev);
4620
Johan Hedbergc21c0ea2014-02-24 11:10:30 +02004621 /* If user space supports this command it is also expected to
4622 * handle IRKs. Therefore, set the HCI_RPA_RESOLVING flag.
4623 */
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004624 hci_dev_set_flag(hdev, HCI_RPA_RESOLVING);
Johan Hedbergc21c0ea2014-02-24 11:10:30 +02004625
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004626 if (cp->privacy) {
Marcel Holtmann238be782015-03-13 02:11:06 -07004627 changed = !hci_dev_test_and_set_flag(hdev, HCI_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004628 memcpy(hdev->irk, cp->irk, sizeof(hdev->irk));
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004629 hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
Johan Hedberg82a37ad2016-03-09 17:30:34 +02004630 if (cp->privacy == 0x02)
4631 hci_dev_set_flag(hdev, HCI_LIMITED_PRIVACY);
4632 else
4633 hci_dev_clear_flag(hdev, HCI_LIMITED_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004634 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07004635 changed = hci_dev_test_and_clear_flag(hdev, HCI_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004636 memset(hdev->irk, 0, sizeof(hdev->irk));
Marcel Holtmanna358dc12015-03-13 02:11:02 -07004637 hci_dev_clear_flag(hdev, HCI_RPA_EXPIRED);
Johan Hedberg82a37ad2016-03-09 17:30:34 +02004638 hci_dev_clear_flag(hdev, HCI_LIMITED_PRIVACY);
Johan Hedberg62b04cd2014-02-23 19:42:27 +02004639 }
4640
4641 err = send_settings_rsp(sk, MGMT_OP_SET_PRIVACY, hdev);
4642 if (err < 0)
4643 goto unlock;
4644
4645 if (changed)
4646 err = new_settings(hdev, sk);
4647
4648unlock:
4649 hci_dev_unlock(hdev);
4650 return err;
4651}
4652
Johan Hedberg41edf162014-02-18 10:19:35 +02004653static bool irk_is_valid(struct mgmt_irk_info *irk)
4654{
4655 switch (irk->addr.type) {
4656 case BDADDR_LE_PUBLIC:
4657 return true;
4658
4659 case BDADDR_LE_RANDOM:
4660 /* Two most significant bits shall be set */
4661 if ((irk->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4662 return false;
4663 return true;
4664 }
4665
4666 return false;
4667}
4668
4669static int load_irks(struct sock *sk, struct hci_dev *hdev, void *cp_data,
4670 u16 len)
4671{
4672 struct mgmt_cp_load_irks *cp = cp_data;
Johan Hedbergba1d6932014-07-03 13:52:27 +03004673 const u16 max_irk_count = ((U16_MAX - sizeof(*cp)) /
4674 sizeof(struct mgmt_irk_info));
Johan Hedberg41edf162014-02-18 10:19:35 +02004675 u16 irk_count, expected_len;
4676 int i, err;
4677
4678 BT_DBG("request for %s", hdev->name);
4679
4680 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004681 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4682 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberg41edf162014-02-18 10:19:35 +02004683
4684 irk_count = __le16_to_cpu(cp->irk_count);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004685 if (irk_count > max_irk_count) {
4686 BT_ERR("load_irks: too big irk_count value %u", irk_count);
Johan Hedberga69e8372015-03-06 21:08:53 +02004687 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4688 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004689 }
Johan Hedberg41edf162014-02-18 10:19:35 +02004690
4691 expected_len = sizeof(*cp) + irk_count * sizeof(struct mgmt_irk_info);
4692 if (expected_len != len) {
4693 BT_ERR("load_irks: expected %u bytes, got %u bytes",
Johan Hedberg2606ecb2014-03-07 15:04:13 +02004694 expected_len, len);
Johan Hedberga69e8372015-03-06 21:08:53 +02004695 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_IRKS,
4696 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg41edf162014-02-18 10:19:35 +02004697 }
4698
4699 BT_DBG("%s irk_count %u", hdev->name, irk_count);
4700
4701 for (i = 0; i < irk_count; i++) {
4702 struct mgmt_irk_info *key = &cp->irks[i];
4703
4704 if (!irk_is_valid(key))
Johan Hedberga69e8372015-03-06 21:08:53 +02004705 return mgmt_cmd_status(sk, hdev->id,
4706 MGMT_OP_LOAD_IRKS,
4707 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg41edf162014-02-18 10:19:35 +02004708 }
4709
4710 hci_dev_lock(hdev);
4711
4712 hci_smp_irks_clear(hdev);
4713
4714 for (i = 0; i < irk_count; i++) {
4715 struct mgmt_irk_info *irk = &cp->irks[i];
Johan Hedberg41edf162014-02-18 10:19:35 +02004716
Johan Hedberg85813a72015-10-21 18:02:59 +03004717 hci_add_irk(hdev, &irk->addr.bdaddr,
4718 le_addr_type(irk->addr.type), irk->val,
Johan Hedberg41edf162014-02-18 10:19:35 +02004719 BDADDR_ANY);
4720 }
4721
Marcel Holtmanna1536da2015-03-13 02:11:01 -07004722 hci_dev_set_flag(hdev, HCI_RPA_RESOLVING);
Johan Hedberg41edf162014-02-18 10:19:35 +02004723
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004724 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_LOAD_IRKS, 0, NULL, 0);
Johan Hedberg41edf162014-02-18 10:19:35 +02004725
4726 hci_dev_unlock(hdev);
4727
4728 return err;
4729}
4730
Johan Hedberg3f706b72013-01-20 14:27:16 +02004731static bool ltk_is_valid(struct mgmt_ltk_info *key)
4732{
4733 if (key->master != 0x00 && key->master != 0x01)
4734 return false;
Marcel Holtmann490cb0b2014-02-16 12:59:05 -08004735
4736 switch (key->addr.type) {
4737 case BDADDR_LE_PUBLIC:
4738 return true;
4739
4740 case BDADDR_LE_RANDOM:
4741 /* Two most significant bits shall be set */
4742 if ((key->addr.bdaddr.b[5] & 0xc0) != 0xc0)
4743 return false;
4744 return true;
4745 }
4746
4747 return false;
Johan Hedberg3f706b72013-01-20 14:27:16 +02004748}
4749
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004750static int load_long_term_keys(struct sock *sk, struct hci_dev *hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03004751 void *cp_data, u16 len)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004752{
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004753 struct mgmt_cp_load_long_term_keys *cp = cp_data;
Johan Hedbergba1d6932014-07-03 13:52:27 +03004754 const u16 max_key_count = ((U16_MAX - sizeof(*cp)) /
4755 sizeof(struct mgmt_ltk_info));
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004756 u16 key_count, expected_len;
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004757 int i, err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004758
Marcel Holtmanncf99ba12013-10-02 21:16:08 -07004759 BT_DBG("request for %s", hdev->name);
4760
4761 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02004762 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4763 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmanncf99ba12013-10-02 21:16:08 -07004764
Marcel Holtmann1f350c82012-03-12 20:31:08 -07004765 key_count = __le16_to_cpu(cp->key_count);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004766 if (key_count > max_key_count) {
4767 BT_ERR("load_ltks: too big key_count value %u", key_count);
Johan Hedberga69e8372015-03-06 21:08:53 +02004768 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4769 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergba1d6932014-07-03 13:52:27 +03004770 }
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004771
4772 expected_len = sizeof(*cp) + key_count *
4773 sizeof(struct mgmt_ltk_info);
4774 if (expected_len != len) {
4775 BT_ERR("load_keys: expected %u bytes, got %u bytes",
Johan Hedberg2606ecb2014-03-07 15:04:13 +02004776 expected_len, len);
Johan Hedberga69e8372015-03-06 21:08:53 +02004777 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS,
4778 MGMT_STATUS_INVALID_PARAMS);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004779 }
4780
Johan Hedbergbdb6d972012-02-28 06:13:32 +02004781 BT_DBG("%s key_count %u", hdev->name, key_count);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004782
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004783 for (i = 0; i < key_count; i++) {
4784 struct mgmt_ltk_info *key = &cp->keys[i];
4785
Johan Hedberg3f706b72013-01-20 14:27:16 +02004786 if (!ltk_is_valid(key))
Johan Hedberga69e8372015-03-06 21:08:53 +02004787 return mgmt_cmd_status(sk, hdev->id,
4788 MGMT_OP_LOAD_LONG_TERM_KEYS,
4789 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberg54ad6d82013-01-20 14:27:15 +02004790 }
4791
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004792 hci_dev_lock(hdev);
4793
4794 hci_smp_ltks_clear(hdev);
4795
4796 for (i = 0; i < key_count; i++) {
4797 struct mgmt_ltk_info *key = &cp->keys[i];
Johan Hedberg85813a72015-10-21 18:02:59 +03004798 u8 type, authenticated;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004799
Johan Hedberg61b43352014-05-29 19:36:53 +03004800 switch (key->type) {
4801 case MGMT_LTK_UNAUTHENTICATED:
Johan Hedbergd7b25452014-05-23 13:19:53 +03004802 authenticated = 0x00;
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;
4805 case MGMT_LTK_AUTHENTICATED:
Johan Hedbergd7b25452014-05-23 13:19:53 +03004806 authenticated = 0x01;
Johan Hedberg23fb8de2014-05-23 13:15:37 +03004807 type = key->master ? SMP_LTK : SMP_LTK_SLAVE;
Johan Hedberg61b43352014-05-29 19:36:53 +03004808 break;
Johan Hedberg23fb8de2014-05-23 13:15:37 +03004809 case MGMT_LTK_P256_UNAUTH:
4810 authenticated = 0x00;
4811 type = SMP_LTK_P256;
4812 break;
4813 case MGMT_LTK_P256_AUTH:
4814 authenticated = 0x01;
4815 type = SMP_LTK_P256;
4816 break;
4817 case MGMT_LTK_P256_DEBUG:
4818 authenticated = 0x00;
4819 type = SMP_LTK_P256_DEBUG;
Johan Hedberg61b43352014-05-29 19:36:53 +03004820 default:
4821 continue;
4822 }
Johan Hedbergd7b25452014-05-23 13:19:53 +03004823
Johan Hedberg85813a72015-10-21 18:02:59 +03004824 hci_add_ltk(hdev, &key->addr.bdaddr,
4825 le_addr_type(key->addr.type), type, authenticated,
4826 key->val, key->enc_size, key->ediv, key->rand);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004827 }
4828
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004829 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_LOAD_LONG_TERM_KEYS, 0,
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004830 NULL, 0);
4831
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004832 hci_dev_unlock(hdev);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004833
Johan Hedberg715a5bf2013-01-09 15:29:34 +02004834 return err;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03004835}
4836
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004837static int conn_info_cmd_complete(struct mgmt_pending_cmd *cmd, u8 status)
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004838{
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004839 struct hci_conn *conn = cmd->user_data;
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004840 struct mgmt_rp_get_conn_info rp;
Johan Hedberg9df74652014-12-19 22:26:03 +02004841 int err;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004842
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004843 memcpy(&rp.addr, cmd->param, sizeof(rp.addr));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004844
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004845 if (status == MGMT_STATUS_SUCCESS) {
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004846 rp.rssi = conn->rssi;
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004847 rp.tx_power = conn->tx_power;
4848 rp.max_tx_power = conn->max_tx_power;
4849 } else {
4850 rp.rssi = HCI_RSSI_INVALID;
4851 rp.tx_power = HCI_TX_POWER_INVALID;
4852 rp.max_tx_power = HCI_TX_POWER_INVALID;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004853 }
4854
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004855 err = mgmt_cmd_complete(cmd->sk, cmd->index, MGMT_OP_GET_CONN_INFO,
4856 status, &rp, sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004857
4858 hci_conn_drop(conn);
Johan Hedbergf8aaf9b2014-08-17 23:28:57 +03004859 hci_conn_put(conn);
Johan Hedberg9df74652014-12-19 22:26:03 +02004860
4861 return err;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004862}
4863
Marcel Holtmann1904a852015-01-11 13:50:44 -08004864static void conn_info_refresh_complete(struct hci_dev *hdev, u8 hci_status,
4865 u16 opcode)
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004866{
4867 struct hci_cp_read_rssi *cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004868 struct mgmt_pending_cmd *cmd;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004869 struct hci_conn *conn;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004870 u16 handle;
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004871 u8 status;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004872
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004873 BT_DBG("status 0x%02x", hci_status);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004874
4875 hci_dev_lock(hdev);
4876
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004877 /* Commands sent in request are either Read RSSI or Read Transmit Power
4878 * Level so we check which one was last sent to retrieve connection
4879 * handle. Both commands have handle as first parameter so it's safe to
4880 * cast data on the same command struct.
4881 *
4882 * First command sent is always Read RSSI and we fail only if it fails.
4883 * In other case we simply override error to indicate success as we
4884 * already remembered if TX power value is actually valid.
4885 */
4886 cp = hci_sent_cmd_data(hdev, HCI_OP_READ_RSSI);
4887 if (!cp) {
4888 cp = hci_sent_cmd_data(hdev, HCI_OP_READ_TX_POWER);
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004889 status = MGMT_STATUS_SUCCESS;
4890 } else {
4891 status = mgmt_status(hci_status);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004892 }
4893
4894 if (!cp) {
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004895 BT_ERR("invalid sent_cmd in conn_info response");
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004896 goto unlock;
4897 }
4898
4899 handle = __le16_to_cpu(cp->handle);
4900 conn = hci_conn_hash_lookup_handle(hdev, handle);
4901 if (!conn) {
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004902 BT_ERR("unknown handle (%d) in conn_info response", handle);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004903 goto unlock;
4904 }
4905
Johan Hedberg333ae952015-03-17 13:48:47 +02004906 cmd = pending_find_data(MGMT_OP_GET_CONN_INFO, hdev, conn);
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004907 if (!cmd)
4908 goto unlock;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004909
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004910 cmd->cmd_complete(cmd, status);
4911 mgmt_pending_remove(cmd);
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004912
4913unlock:
4914 hci_dev_unlock(hdev);
4915}
4916
4917static int get_conn_info(struct sock *sk, struct hci_dev *hdev, void *data,
4918 u16 len)
4919{
4920 struct mgmt_cp_get_conn_info *cp = data;
4921 struct mgmt_rp_get_conn_info rp;
4922 struct hci_conn *conn;
4923 unsigned long conn_info_age;
4924 int err = 0;
4925
4926 BT_DBG("%s", hdev->name);
4927
4928 memset(&rp, 0, sizeof(rp));
4929 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
4930 rp.addr.type = cp->addr.type;
4931
4932 if (!bdaddr_type_is_valid(cp->addr.type))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004933 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4934 MGMT_STATUS_INVALID_PARAMS,
4935 &rp, sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004936
4937 hci_dev_lock(hdev);
4938
4939 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004940 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4941 MGMT_STATUS_NOT_POWERED, &rp,
4942 sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004943 goto unlock;
4944 }
4945
4946 if (cp->addr.type == BDADDR_BREDR)
4947 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
4948 &cp->addr.bdaddr);
4949 else
4950 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, &cp->addr.bdaddr);
4951
4952 if (!conn || conn->state != BT_CONNECTED) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004953 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4954 MGMT_STATUS_NOT_CONNECTED, &rp,
4955 sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004956 goto unlock;
4957 }
4958
Johan Hedberg333ae952015-03-17 13:48:47 +02004959 if (pending_find_data(MGMT_OP_GET_CONN_INFO, hdev, conn)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02004960 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
4961 MGMT_STATUS_BUSY, &rp, sizeof(rp));
Johan Hedberg9981bdb2014-12-05 13:42:57 +02004962 goto unlock;
4963 }
4964
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004965 /* To avoid client trying to guess when to poll again for information we
4966 * calculate conn info age as random value between min/max set in hdev.
4967 */
4968 conn_info_age = hdev->conn_info_min_age +
4969 prandom_u32_max(hdev->conn_info_max_age -
4970 hdev->conn_info_min_age);
4971
4972 /* Query controller to refresh cached values if they are too old or were
4973 * never read.
4974 */
Andrzej Kaczmarekf4e2dd52014-05-16 16:48:57 +02004975 if (time_after(jiffies, conn->conn_info_timestamp +
4976 msecs_to_jiffies(conn_info_age)) ||
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004977 !conn->conn_info_timestamp) {
4978 struct hci_request req;
4979 struct hci_cp_read_tx_power req_txp_cp;
4980 struct hci_cp_read_rssi req_rssi_cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02004981 struct mgmt_pending_cmd *cmd;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004982
4983 hci_req_init(&req, hdev);
4984 req_rssi_cp.handle = cpu_to_le16(conn->handle);
4985 hci_req_add(&req, HCI_OP_READ_RSSI, sizeof(req_rssi_cp),
4986 &req_rssi_cp);
4987
Andrzej Kaczmarekf7faab02014-05-14 13:43:04 +02004988 /* For LE links TX power does not change thus we don't need to
4989 * query for it once value is known.
4990 */
4991 if (!bdaddr_type_is_le(cp->addr.type) ||
4992 conn->tx_power == HCI_TX_POWER_INVALID) {
4993 req_txp_cp.handle = cpu_to_le16(conn->handle);
4994 req_txp_cp.type = 0x00;
4995 hci_req_add(&req, HCI_OP_READ_TX_POWER,
4996 sizeof(req_txp_cp), &req_txp_cp);
4997 }
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02004998
Andrzej Kaczmarekeed5daf2014-05-14 13:43:06 +02004999 /* Max TX power needs to be read only once per connection */
5000 if (conn->max_tx_power == HCI_TX_POWER_INVALID) {
5001 req_txp_cp.handle = cpu_to_le16(conn->handle);
5002 req_txp_cp.type = 0x01;
5003 hci_req_add(&req, HCI_OP_READ_TX_POWER,
5004 sizeof(req_txp_cp), &req_txp_cp);
5005 }
5006
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005007 err = hci_req_run(&req, conn_info_refresh_complete);
5008 if (err < 0)
5009 goto unlock;
5010
5011 cmd = mgmt_pending_add(sk, MGMT_OP_GET_CONN_INFO, hdev,
5012 data, len);
5013 if (!cmd) {
5014 err = -ENOMEM;
5015 goto unlock;
5016 }
5017
5018 hci_conn_hold(conn);
Johan Hedbergf8aaf9b2014-08-17 23:28:57 +03005019 cmd->user_data = hci_conn_get(conn);
Johan Hedberg9981bdb2014-12-05 13:42:57 +02005020 cmd->cmd_complete = conn_info_cmd_complete;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005021
5022 conn->conn_info_timestamp = jiffies;
5023 } else {
5024 /* Cache is valid, just reply with values cached in hci_conn */
5025 rp.rssi = conn->rssi;
5026 rp.tx_power = conn->tx_power;
Andrzej Kaczmarekeed5daf2014-05-14 13:43:06 +02005027 rp.max_tx_power = conn->max_tx_power;
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005028
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005029 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CONN_INFO,
5030 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
Andrzej Kaczmarekdd983802014-05-14 13:43:03 +02005031 }
5032
5033unlock:
5034 hci_dev_unlock(hdev);
5035 return err;
5036}
5037
Johan Hedberg3b0602c2015-03-06 21:08:55 +02005038static int clock_info_cmd_complete(struct mgmt_pending_cmd *cmd, u8 status)
Johan Hedberg69487372014-12-05 13:36:07 +02005039{
5040 struct hci_conn *conn = cmd->user_data;
5041 struct mgmt_rp_get_clock_info rp;
5042 struct hci_dev *hdev;
Johan Hedberg9df74652014-12-19 22:26:03 +02005043 int err;
Johan Hedberg69487372014-12-05 13:36:07 +02005044
5045 memset(&rp, 0, sizeof(rp));
Marcel Holtmann56f787c2016-08-29 06:19:47 +02005046 memcpy(&rp.addr, cmd->param, sizeof(rp.addr));
Johan Hedberg69487372014-12-05 13:36:07 +02005047
5048 if (status)
5049 goto complete;
5050
5051 hdev = hci_dev_get(cmd->index);
5052 if (hdev) {
5053 rp.local_clock = cpu_to_le32(hdev->clock);
5054 hci_dev_put(hdev);
5055 }
5056
5057 if (conn) {
5058 rp.piconet_clock = cpu_to_le32(conn->clock);
5059 rp.accuracy = cpu_to_le16(conn->clock_accuracy);
5060 }
5061
5062complete:
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005063 err = mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, status, &rp,
5064 sizeof(rp));
Johan Hedberg69487372014-12-05 13:36:07 +02005065
5066 if (conn) {
5067 hci_conn_drop(conn);
5068 hci_conn_put(conn);
5069 }
Johan Hedberg9df74652014-12-19 22:26:03 +02005070
5071 return err;
Johan Hedberg69487372014-12-05 13:36:07 +02005072}
5073
Marcel Holtmann1904a852015-01-11 13:50:44 -08005074static void get_clock_info_complete(struct hci_dev *hdev, u8 status, u16 opcode)
Johan Hedberg95868422014-06-28 17:54:07 +03005075{
Johan Hedberg95868422014-06-28 17:54:07 +03005076 struct hci_cp_read_clock *hci_cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02005077 struct mgmt_pending_cmd *cmd;
Johan Hedberg95868422014-06-28 17:54:07 +03005078 struct hci_conn *conn;
5079
5080 BT_DBG("%s status %u", hdev->name, status);
5081
5082 hci_dev_lock(hdev);
5083
5084 hci_cp = hci_sent_cmd_data(hdev, HCI_OP_READ_CLOCK);
5085 if (!hci_cp)
5086 goto unlock;
5087
5088 if (hci_cp->which) {
5089 u16 handle = __le16_to_cpu(hci_cp->handle);
5090 conn = hci_conn_hash_lookup_handle(hdev, handle);
5091 } else {
5092 conn = NULL;
5093 }
5094
Johan Hedberg333ae952015-03-17 13:48:47 +02005095 cmd = pending_find_data(MGMT_OP_GET_CLOCK_INFO, hdev, conn);
Johan Hedberg95868422014-06-28 17:54:07 +03005096 if (!cmd)
5097 goto unlock;
5098
Johan Hedberg69487372014-12-05 13:36:07 +02005099 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberg95868422014-06-28 17:54:07 +03005100 mgmt_pending_remove(cmd);
Johan Hedberg95868422014-06-28 17:54:07 +03005101
5102unlock:
5103 hci_dev_unlock(hdev);
5104}
5105
5106static int get_clock_info(struct sock *sk, struct hci_dev *hdev, void *data,
5107 u16 len)
5108{
5109 struct mgmt_cp_get_clock_info *cp = data;
5110 struct mgmt_rp_get_clock_info rp;
5111 struct hci_cp_read_clock hci_cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02005112 struct mgmt_pending_cmd *cmd;
Johan Hedberg95868422014-06-28 17:54:07 +03005113 struct hci_request req;
5114 struct hci_conn *conn;
5115 int err;
5116
5117 BT_DBG("%s", hdev->name);
5118
5119 memset(&rp, 0, sizeof(rp));
5120 bacpy(&rp.addr.bdaddr, &cp->addr.bdaddr);
5121 rp.addr.type = cp->addr.type;
5122
5123 if (cp->addr.type != BDADDR_BREDR)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005124 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CLOCK_INFO,
5125 MGMT_STATUS_INVALID_PARAMS,
5126 &rp, sizeof(rp));
Johan Hedberg95868422014-06-28 17:54:07 +03005127
5128 hci_dev_lock(hdev);
5129
5130 if (!hdev_is_powered(hdev)) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005131 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_CLOCK_INFO,
5132 MGMT_STATUS_NOT_POWERED, &rp,
5133 sizeof(rp));
Johan Hedberg95868422014-06-28 17:54:07 +03005134 goto unlock;
5135 }
5136
5137 if (bacmp(&cp->addr.bdaddr, BDADDR_ANY)) {
5138 conn = hci_conn_hash_lookup_ba(hdev, ACL_LINK,
5139 &cp->addr.bdaddr);
5140 if (!conn || conn->state != BT_CONNECTED) {
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005141 err = mgmt_cmd_complete(sk, hdev->id,
5142 MGMT_OP_GET_CLOCK_INFO,
5143 MGMT_STATUS_NOT_CONNECTED,
5144 &rp, sizeof(rp));
Johan Hedberg95868422014-06-28 17:54:07 +03005145 goto unlock;
5146 }
5147 } else {
5148 conn = NULL;
5149 }
5150
5151 cmd = mgmt_pending_add(sk, MGMT_OP_GET_CLOCK_INFO, hdev, data, len);
5152 if (!cmd) {
5153 err = -ENOMEM;
5154 goto unlock;
5155 }
5156
Johan Hedberg69487372014-12-05 13:36:07 +02005157 cmd->cmd_complete = clock_info_cmd_complete;
5158
Johan Hedberg95868422014-06-28 17:54:07 +03005159 hci_req_init(&req, hdev);
5160
5161 memset(&hci_cp, 0, sizeof(hci_cp));
5162 hci_req_add(&req, HCI_OP_READ_CLOCK, sizeof(hci_cp), &hci_cp);
5163
5164 if (conn) {
5165 hci_conn_hold(conn);
Johan Hedbergf8aaf9b2014-08-17 23:28:57 +03005166 cmd->user_data = hci_conn_get(conn);
Johan Hedberg95868422014-06-28 17:54:07 +03005167
5168 hci_cp.handle = cpu_to_le16(conn->handle);
5169 hci_cp.which = 0x01; /* Piconet clock */
5170 hci_req_add(&req, HCI_OP_READ_CLOCK, sizeof(hci_cp), &hci_cp);
5171 }
5172
5173 err = hci_req_run(&req, get_clock_info_complete);
5174 if (err < 0)
5175 mgmt_pending_remove(cmd);
5176
5177unlock:
5178 hci_dev_unlock(hdev);
5179 return err;
5180}
5181
Johan Hedberg5a154e62014-12-19 22:26:02 +02005182static bool is_connected(struct hci_dev *hdev, bdaddr_t *addr, u8 type)
5183{
5184 struct hci_conn *conn;
5185
5186 conn = hci_conn_hash_lookup_ba(hdev, LE_LINK, addr);
5187 if (!conn)
5188 return false;
5189
5190 if (conn->dst_type != type)
5191 return false;
5192
5193 if (conn->state != BT_CONNECTED)
5194 return false;
5195
5196 return true;
5197}
5198
5199/* This function requires the caller holds hdev->lock */
Johan Hedberg51d7a942015-11-11 08:11:18 +02005200static int hci_conn_params_set(struct hci_dev *hdev, bdaddr_t *addr,
Johan Hedberg5a154e62014-12-19 22:26:02 +02005201 u8 addr_type, u8 auto_connect)
5202{
Johan Hedberg5a154e62014-12-19 22:26:02 +02005203 struct hci_conn_params *params;
5204
5205 params = hci_conn_params_add(hdev, addr, addr_type);
5206 if (!params)
5207 return -EIO;
5208
5209 if (params->auto_connect == auto_connect)
5210 return 0;
5211
5212 list_del_init(&params->action);
5213
5214 switch (auto_connect) {
5215 case HCI_AUTO_CONN_DISABLED:
5216 case HCI_AUTO_CONN_LINK_LOSS:
Jakub Pawlowski28a667c2015-08-07 20:22:54 +02005217 /* If auto connect is being disabled when we're trying to
5218 * connect to device, keep connecting.
5219 */
5220 if (params->explicit_connect)
5221 list_add(&params->action, &hdev->pend_le_conns);
Johan Hedberg5a154e62014-12-19 22:26:02 +02005222 break;
5223 case HCI_AUTO_CONN_REPORT:
Johan Hedberg49c50922015-10-16 10:07:51 +03005224 if (params->explicit_connect)
5225 list_add(&params->action, &hdev->pend_le_conns);
5226 else
5227 list_add(&params->action, &hdev->pend_le_reports);
Johan Hedberg5a154e62014-12-19 22:26:02 +02005228 break;
5229 case HCI_AUTO_CONN_DIRECT:
5230 case HCI_AUTO_CONN_ALWAYS:
Johan Hedberg51d7a942015-11-11 08:11:18 +02005231 if (!is_connected(hdev, addr, addr_type))
Johan Hedberg5a154e62014-12-19 22:26:02 +02005232 list_add(&params->action, &hdev->pend_le_conns);
Johan Hedberg5a154e62014-12-19 22:26:02 +02005233 break;
5234 }
5235
5236 params->auto_connect = auto_connect;
5237
5238 BT_DBG("addr %pMR (type %u) auto_connect %u", addr, addr_type,
5239 auto_connect);
5240
5241 return 0;
5242}
5243
Marcel Holtmann8afef092014-06-29 22:28:34 +02005244static void device_added(struct sock *sk, struct hci_dev *hdev,
5245 bdaddr_t *bdaddr, u8 type, u8 action)
5246{
5247 struct mgmt_ev_device_added ev;
5248
5249 bacpy(&ev.addr.bdaddr, bdaddr);
5250 ev.addr.type = type;
5251 ev.action = action;
5252
5253 mgmt_event(MGMT_EV_DEVICE_ADDED, hdev, &ev, sizeof(ev), sk);
5254}
5255
Marcel Holtmann2faade52014-06-29 19:44:03 +02005256static int add_device(struct sock *sk, struct hci_dev *hdev,
5257 void *data, u16 len)
5258{
5259 struct mgmt_cp_add_device *cp = data;
5260 u8 auto_conn, addr_type;
5261 int err;
5262
5263 BT_DBG("%s", hdev->name);
5264
Johan Hedberg66593582014-07-09 12:59:14 +03005265 if (!bdaddr_type_is_valid(cp->addr.type) ||
Marcel Holtmann2faade52014-06-29 19:44:03 +02005266 !bacmp(&cp->addr.bdaddr, BDADDR_ANY))
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005267 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5268 MGMT_STATUS_INVALID_PARAMS,
5269 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005270
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005271 if (cp->action != 0x00 && cp->action != 0x01 && cp->action != 0x02)
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005272 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5273 MGMT_STATUS_INVALID_PARAMS,
5274 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005275
5276 hci_dev_lock(hdev);
5277
Johan Hedberg66593582014-07-09 12:59:14 +03005278 if (cp->addr.type == BDADDR_BREDR) {
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005279 /* Only incoming connections action is supported for now */
Johan Hedberg66593582014-07-09 12:59:14 +03005280 if (cp->action != 0x01) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005281 err = mgmt_cmd_complete(sk, hdev->id,
5282 MGMT_OP_ADD_DEVICE,
5283 MGMT_STATUS_INVALID_PARAMS,
5284 &cp->addr, sizeof(cp->addr));
Johan Hedberg66593582014-07-09 12:59:14 +03005285 goto unlock;
5286 }
5287
5288 err = hci_bdaddr_list_add(&hdev->whitelist, &cp->addr.bdaddr,
5289 cp->addr.type);
5290 if (err)
5291 goto unlock;
Johan Hedberga3974072014-07-09 12:59:15 +03005292
Johan Hedberg01b1cb82015-11-16 12:52:21 +02005293 hci_req_update_scan(hdev);
Johan Hedberga3974072014-07-09 12:59:15 +03005294
Johan Hedberg66593582014-07-09 12:59:14 +03005295 goto added;
5296 }
5297
Johan Hedberg85813a72015-10-21 18:02:59 +03005298 addr_type = le_addr_type(cp->addr.type);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005299
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005300 if (cp->action == 0x02)
Marcel Holtmann2faade52014-06-29 19:44:03 +02005301 auto_conn = HCI_AUTO_CONN_ALWAYS;
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02005302 else if (cp->action == 0x01)
5303 auto_conn = HCI_AUTO_CONN_DIRECT;
Marcel Holtmann2faade52014-06-29 19:44:03 +02005304 else
Johan Hedberga3451d22014-07-02 17:37:27 +03005305 auto_conn = HCI_AUTO_CONN_REPORT;
Marcel Holtmann2faade52014-06-29 19:44:03 +02005306
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005307 /* Kernel internally uses conn_params with resolvable private
5308 * address, but Add Device allows only identity addresses.
5309 * Make sure it is enforced before calling
5310 * hci_conn_params_lookup.
5311 */
5312 if (!hci_is_identity_address(&cp->addr.bdaddr, addr_type)) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005313 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5314 MGMT_STATUS_INVALID_PARAMS,
5315 &cp->addr, sizeof(cp->addr));
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005316 goto unlock;
5317 }
5318
Marcel Holtmannbf5b3c82014-06-30 12:34:39 +02005319 /* If the connection parameters don't exist for this device,
5320 * they will be created and configured with defaults.
5321 */
Johan Hedberg51d7a942015-11-11 08:11:18 +02005322 if (hci_conn_params_set(hdev, &cp->addr.bdaddr, addr_type,
Marcel Holtmannd06b50c2014-07-01 12:11:06 +02005323 auto_conn) < 0) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005324 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5325 MGMT_STATUS_FAILED, &cp->addr,
5326 sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005327 goto unlock;
5328 }
5329
Johan Hedberg51d7a942015-11-11 08:11:18 +02005330 hci_update_background_scan(hdev);
5331
Johan Hedberg66593582014-07-09 12:59:14 +03005332added:
Marcel Holtmann8afef092014-06-29 22:28:34 +02005333 device_added(sk, hdev, &cp->addr.bdaddr, cp->addr.type, cp->action);
5334
Johan Hedberg51d7a942015-11-11 08:11:18 +02005335 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_DEVICE,
5336 MGMT_STATUS_SUCCESS, &cp->addr,
5337 sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005338
5339unlock:
5340 hci_dev_unlock(hdev);
5341 return err;
5342}
5343
Marcel Holtmann8afef092014-06-29 22:28:34 +02005344static void device_removed(struct sock *sk, struct hci_dev *hdev,
5345 bdaddr_t *bdaddr, u8 type)
5346{
5347 struct mgmt_ev_device_removed ev;
5348
5349 bacpy(&ev.addr.bdaddr, bdaddr);
5350 ev.addr.type = type;
5351
5352 mgmt_event(MGMT_EV_DEVICE_REMOVED, hdev, &ev, sizeof(ev), sk);
5353}
5354
Marcel Holtmann2faade52014-06-29 19:44:03 +02005355static int remove_device(struct sock *sk, struct hci_dev *hdev,
5356 void *data, u16 len)
5357{
5358 struct mgmt_cp_remove_device *cp = data;
5359 int err;
5360
5361 BT_DBG("%s", hdev->name);
5362
5363 hci_dev_lock(hdev);
5364
5365 if (bacmp(&cp->addr.bdaddr, BDADDR_ANY)) {
Johan Hedbergc71593d2014-07-02 17:37:28 +03005366 struct hci_conn_params *params;
Marcel Holtmann2faade52014-06-29 19:44:03 +02005367 u8 addr_type;
5368
Johan Hedberg66593582014-07-09 12:59:14 +03005369 if (!bdaddr_type_is_valid(cp->addr.type)) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005370 err = mgmt_cmd_complete(sk, hdev->id,
5371 MGMT_OP_REMOVE_DEVICE,
5372 MGMT_STATUS_INVALID_PARAMS,
5373 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005374 goto unlock;
5375 }
5376
Johan Hedberg66593582014-07-09 12:59:14 +03005377 if (cp->addr.type == BDADDR_BREDR) {
5378 err = hci_bdaddr_list_del(&hdev->whitelist,
5379 &cp->addr.bdaddr,
5380 cp->addr.type);
5381 if (err) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005382 err = mgmt_cmd_complete(sk, hdev->id,
5383 MGMT_OP_REMOVE_DEVICE,
5384 MGMT_STATUS_INVALID_PARAMS,
5385 &cp->addr,
5386 sizeof(cp->addr));
Johan Hedberg66593582014-07-09 12:59:14 +03005387 goto unlock;
5388 }
5389
Johan Hedberg01b1cb82015-11-16 12:52:21 +02005390 hci_req_update_scan(hdev);
Johan Hedberga3974072014-07-09 12:59:15 +03005391
Johan Hedberg66593582014-07-09 12:59:14 +03005392 device_removed(sk, hdev, &cp->addr.bdaddr,
5393 cp->addr.type);
5394 goto complete;
5395 }
5396
Johan Hedberg85813a72015-10-21 18:02:59 +03005397 addr_type = le_addr_type(cp->addr.type);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005398
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005399 /* Kernel internally uses conn_params with resolvable private
5400 * address, but Remove Device allows only identity addresses.
5401 * Make sure it is enforced before calling
5402 * hci_conn_params_lookup.
5403 */
5404 if (!hci_is_identity_address(&cp->addr.bdaddr, addr_type)) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005405 err = mgmt_cmd_complete(sk, hdev->id,
5406 MGMT_OP_REMOVE_DEVICE,
5407 MGMT_STATUS_INVALID_PARAMS,
5408 &cp->addr, sizeof(cp->addr));
Jakub Pawlowski9a0a8a82015-07-20 13:12:49 +02005409 goto unlock;
5410 }
5411
Johan Hedbergc71593d2014-07-02 17:37:28 +03005412 params = hci_conn_params_lookup(hdev, &cp->addr.bdaddr,
5413 addr_type);
5414 if (!params) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005415 err = mgmt_cmd_complete(sk, hdev->id,
5416 MGMT_OP_REMOVE_DEVICE,
5417 MGMT_STATUS_INVALID_PARAMS,
5418 &cp->addr, sizeof(cp->addr));
Johan Hedbergc71593d2014-07-02 17:37:28 +03005419 goto unlock;
5420 }
5421
Johan Hedberg679d2b62015-10-16 10:07:52 +03005422 if (params->auto_connect == HCI_AUTO_CONN_DISABLED ||
5423 params->auto_connect == HCI_AUTO_CONN_EXPLICIT) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005424 err = mgmt_cmd_complete(sk, hdev->id,
5425 MGMT_OP_REMOVE_DEVICE,
5426 MGMT_STATUS_INVALID_PARAMS,
5427 &cp->addr, sizeof(cp->addr));
Johan Hedbergc71593d2014-07-02 17:37:28 +03005428 goto unlock;
5429 }
5430
Johan Hedbergd1dbf122014-07-04 16:17:23 +03005431 list_del(&params->action);
Johan Hedbergc71593d2014-07-02 17:37:28 +03005432 list_del(&params->list);
5433 kfree(params);
Johan Hedberg51d7a942015-11-11 08:11:18 +02005434 hci_update_background_scan(hdev);
Marcel Holtmann8afef092014-06-29 22:28:34 +02005435
5436 device_removed(sk, hdev, &cp->addr.bdaddr, cp->addr.type);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005437 } else {
Johan Hedberg19de0822014-07-06 13:06:51 +03005438 struct hci_conn_params *p, *tmp;
Johan Hedberg66593582014-07-09 12:59:14 +03005439 struct bdaddr_list *b, *btmp;
Johan Hedberg19de0822014-07-06 13:06:51 +03005440
Marcel Holtmann2faade52014-06-29 19:44:03 +02005441 if (cp->addr.type) {
Johan Hedberg51d7a942015-11-11 08:11:18 +02005442 err = mgmt_cmd_complete(sk, hdev->id,
5443 MGMT_OP_REMOVE_DEVICE,
5444 MGMT_STATUS_INVALID_PARAMS,
5445 &cp->addr, sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005446 goto unlock;
5447 }
5448
Johan Hedberg66593582014-07-09 12:59:14 +03005449 list_for_each_entry_safe(b, btmp, &hdev->whitelist, list) {
5450 device_removed(sk, hdev, &b->bdaddr, b->bdaddr_type);
5451 list_del(&b->list);
5452 kfree(b);
5453 }
5454
Johan Hedberg01b1cb82015-11-16 12:52:21 +02005455 hci_req_update_scan(hdev);
Johan Hedberga3974072014-07-09 12:59:15 +03005456
Johan Hedberg19de0822014-07-06 13:06:51 +03005457 list_for_each_entry_safe(p, tmp, &hdev->le_conn_params, list) {
5458 if (p->auto_connect == HCI_AUTO_CONN_DISABLED)
5459 continue;
5460 device_removed(sk, hdev, &p->addr, p->addr_type);
Johan Hedberg679d2b62015-10-16 10:07:52 +03005461 if (p->explicit_connect) {
5462 p->auto_connect = HCI_AUTO_CONN_EXPLICIT;
5463 continue;
5464 }
Johan Hedberg19de0822014-07-06 13:06:51 +03005465 list_del(&p->action);
5466 list_del(&p->list);
5467 kfree(p);
5468 }
5469
5470 BT_DBG("All LE connection parameters were removed");
5471
Johan Hedberg51d7a942015-11-11 08:11:18 +02005472 hci_update_background_scan(hdev);
Marcel Holtmann2faade52014-06-29 19:44:03 +02005473 }
5474
Johan Hedberg66593582014-07-09 12:59:14 +03005475complete:
Johan Hedberg51d7a942015-11-11 08:11:18 +02005476 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_REMOVE_DEVICE,
5477 MGMT_STATUS_SUCCESS, &cp->addr,
5478 sizeof(cp->addr));
Marcel Holtmann2faade52014-06-29 19:44:03 +02005479unlock:
5480 hci_dev_unlock(hdev);
5481 return err;
5482}
5483
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005484static int load_conn_param(struct sock *sk, struct hci_dev *hdev, void *data,
5485 u16 len)
5486{
5487 struct mgmt_cp_load_conn_param *cp = data;
Johan Hedbergba1d6932014-07-03 13:52:27 +03005488 const u16 max_param_count = ((U16_MAX - sizeof(*cp)) /
5489 sizeof(struct mgmt_conn_param));
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005490 u16 param_count, expected_len;
5491 int i;
5492
5493 if (!lmp_le_capable(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02005494 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM,
5495 MGMT_STATUS_NOT_SUPPORTED);
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005496
5497 param_count = __le16_to_cpu(cp->param_count);
Johan Hedbergba1d6932014-07-03 13:52:27 +03005498 if (param_count > max_param_count) {
5499 BT_ERR("load_conn_param: too big param_count value %u",
5500 param_count);
Johan Hedberga69e8372015-03-06 21:08:53 +02005501 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM,
5502 MGMT_STATUS_INVALID_PARAMS);
Johan Hedbergba1d6932014-07-03 13:52:27 +03005503 }
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005504
5505 expected_len = sizeof(*cp) + param_count *
5506 sizeof(struct mgmt_conn_param);
5507 if (expected_len != len) {
5508 BT_ERR("load_conn_param: expected %u bytes, got %u bytes",
5509 expected_len, len);
Johan Hedberga69e8372015-03-06 21:08:53 +02005510 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM,
5511 MGMT_STATUS_INVALID_PARAMS);
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005512 }
5513
5514 BT_DBG("%s param_count %u", hdev->name, param_count);
5515
5516 hci_dev_lock(hdev);
5517
5518 hci_conn_params_clear_disabled(hdev);
5519
5520 for (i = 0; i < param_count; i++) {
5521 struct mgmt_conn_param *param = &cp->params[i];
5522 struct hci_conn_params *hci_param;
5523 u16 min, max, latency, timeout;
5524 u8 addr_type;
5525
5526 BT_DBG("Adding %pMR (type %u)", &param->addr.bdaddr,
5527 param->addr.type);
5528
5529 if (param->addr.type == BDADDR_LE_PUBLIC) {
5530 addr_type = ADDR_LE_DEV_PUBLIC;
5531 } else if (param->addr.type == BDADDR_LE_RANDOM) {
5532 addr_type = ADDR_LE_DEV_RANDOM;
5533 } else {
5534 BT_ERR("Ignoring invalid connection parameters");
5535 continue;
5536 }
5537
5538 min = le16_to_cpu(param->min_interval);
5539 max = le16_to_cpu(param->max_interval);
5540 latency = le16_to_cpu(param->latency);
5541 timeout = le16_to_cpu(param->timeout);
5542
5543 BT_DBG("min 0x%04x max 0x%04x latency 0x%04x timeout 0x%04x",
5544 min, max, latency, timeout);
5545
5546 if (hci_check_conn_params(min, max, latency, timeout) < 0) {
5547 BT_ERR("Ignoring invalid connection parameters");
5548 continue;
5549 }
5550
5551 hci_param = hci_conn_params_add(hdev, &param->addr.bdaddr,
5552 addr_type);
5553 if (!hci_param) {
5554 BT_ERR("Failed to add connection parameters");
5555 continue;
5556 }
5557
5558 hci_param->conn_min_interval = min;
5559 hci_param->conn_max_interval = max;
5560 hci_param->conn_latency = latency;
5561 hci_param->supervision_timeout = timeout;
5562 }
5563
5564 hci_dev_unlock(hdev);
5565
Johan Hedberg2a1afb52015-03-06 21:08:54 +02005566 return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_LOAD_CONN_PARAM, 0,
5567 NULL, 0);
Johan Hedberga26f3dc2014-07-02 17:37:29 +03005568}
5569
Marcel Holtmanndbece372014-07-04 18:11:55 +02005570static int set_external_config(struct sock *sk, struct hci_dev *hdev,
5571 void *data, u16 len)
5572{
5573 struct mgmt_cp_set_external_config *cp = data;
5574 bool changed;
5575 int err;
5576
5577 BT_DBG("%s", hdev->name);
5578
5579 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02005580 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_EXTERNAL_CONFIG,
5581 MGMT_STATUS_REJECTED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005582
5583 if (cp->config != 0x00 && cp->config != 0x01)
Johan Hedberga69e8372015-03-06 21:08:53 +02005584 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_EXTERNAL_CONFIG,
5585 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005586
5587 if (!test_bit(HCI_QUIRK_EXTERNAL_CONFIG, &hdev->quirks))
Johan Hedberga69e8372015-03-06 21:08:53 +02005588 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_EXTERNAL_CONFIG,
5589 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005590
5591 hci_dev_lock(hdev);
5592
5593 if (cp->config)
Marcel Holtmann238be782015-03-13 02:11:06 -07005594 changed = !hci_dev_test_and_set_flag(hdev, HCI_EXT_CONFIGURED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005595 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07005596 changed = hci_dev_test_and_clear_flag(hdev, HCI_EXT_CONFIGURED);
Marcel Holtmanndbece372014-07-04 18:11:55 +02005597
5598 err = send_options_rsp(sk, MGMT_OP_SET_EXTERNAL_CONFIG, hdev);
5599 if (err < 0)
5600 goto unlock;
5601
5602 if (!changed)
5603 goto unlock;
5604
Marcel Holtmannf4537c02014-07-04 19:06:23 +02005605 err = new_options(hdev, sk);
5606
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07005607 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED) == is_configured(hdev)) {
Marcel Holtmanndbece372014-07-04 18:11:55 +02005608 mgmt_index_removed(hdev);
Marcel Holtmannd603b76b2014-07-06 12:11:14 +02005609
Marcel Holtmann516018a2015-03-13 02:11:04 -07005610 if (hci_dev_test_and_change_flag(hdev, HCI_UNCONFIGURED)) {
Marcel Holtmanna1536da2015-03-13 02:11:01 -07005611 hci_dev_set_flag(hdev, HCI_CONFIG);
5612 hci_dev_set_flag(hdev, HCI_AUTO_OFF);
Marcel Holtmannd603b76b2014-07-06 12:11:14 +02005613
5614 queue_work(hdev->req_workqueue, &hdev->power_on);
5615 } else {
Marcel Holtmann5ea234d2014-07-06 12:11:16 +02005616 set_bit(HCI_RAW, &hdev->flags);
Marcel Holtmannd603b76b2014-07-06 12:11:14 +02005617 mgmt_index_added(hdev);
5618 }
Marcel Holtmanndbece372014-07-04 18:11:55 +02005619 }
5620
5621unlock:
5622 hci_dev_unlock(hdev);
5623 return err;
5624}
5625
Marcel Holtmann9713c172014-07-06 12:11:15 +02005626static int set_public_address(struct sock *sk, struct hci_dev *hdev,
5627 void *data, u16 len)
5628{
5629 struct mgmt_cp_set_public_address *cp = data;
5630 bool changed;
5631 int err;
5632
5633 BT_DBG("%s", hdev->name);
5634
5635 if (hdev_is_powered(hdev))
Johan Hedberga69e8372015-03-06 21:08:53 +02005636 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PUBLIC_ADDRESS,
5637 MGMT_STATUS_REJECTED);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005638
5639 if (!bacmp(&cp->bdaddr, BDADDR_ANY))
Johan Hedberga69e8372015-03-06 21:08:53 +02005640 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PUBLIC_ADDRESS,
5641 MGMT_STATUS_INVALID_PARAMS);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005642
5643 if (!hdev->set_bdaddr)
Johan Hedberga69e8372015-03-06 21:08:53 +02005644 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_PUBLIC_ADDRESS,
5645 MGMT_STATUS_NOT_SUPPORTED);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005646
5647 hci_dev_lock(hdev);
5648
5649 changed = !!bacmp(&hdev->public_addr, &cp->bdaddr);
5650 bacpy(&hdev->public_addr, &cp->bdaddr);
5651
5652 err = send_options_rsp(sk, MGMT_OP_SET_PUBLIC_ADDRESS, hdev);
5653 if (err < 0)
5654 goto unlock;
5655
5656 if (!changed)
5657 goto unlock;
5658
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07005659 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED))
Marcel Holtmann9713c172014-07-06 12:11:15 +02005660 err = new_options(hdev, sk);
5661
5662 if (is_configured(hdev)) {
5663 mgmt_index_removed(hdev);
5664
Marcel Holtmanna358dc12015-03-13 02:11:02 -07005665 hci_dev_clear_flag(hdev, HCI_UNCONFIGURED);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005666
Marcel Holtmanna1536da2015-03-13 02:11:01 -07005667 hci_dev_set_flag(hdev, HCI_CONFIG);
5668 hci_dev_set_flag(hdev, HCI_AUTO_OFF);
Marcel Holtmann9713c172014-07-06 12:11:15 +02005669
5670 queue_work(hdev->req_workqueue, &hdev->power_on);
5671 }
5672
5673unlock:
5674 hci_dev_unlock(hdev);
5675 return err;
5676}
5677
Johan Hedberg40f66c02015-04-07 21:52:22 +03005678static void read_local_oob_ext_data_complete(struct hci_dev *hdev, u8 status,
5679 u16 opcode, struct sk_buff *skb)
5680{
5681 const struct mgmt_cp_read_local_oob_ext_data *mgmt_cp;
5682 struct mgmt_rp_read_local_oob_ext_data *mgmt_rp;
5683 u8 *h192, *r192, *h256, *r256;
5684 struct mgmt_pending_cmd *cmd;
5685 u16 eir_len;
5686 int err;
5687
5688 BT_DBG("%s status %u", hdev->name, status);
5689
5690 cmd = pending_find(MGMT_OP_READ_LOCAL_OOB_EXT_DATA, hdev);
5691 if (!cmd)
5692 return;
5693
5694 mgmt_cp = cmd->param;
5695
5696 if (status) {
5697 status = mgmt_status(status);
5698 eir_len = 0;
5699
5700 h192 = NULL;
5701 r192 = NULL;
5702 h256 = NULL;
5703 r256 = NULL;
5704 } else if (opcode == HCI_OP_READ_LOCAL_OOB_DATA) {
5705 struct hci_rp_read_local_oob_data *rp;
5706
5707 if (skb->len != sizeof(*rp)) {
5708 status = MGMT_STATUS_FAILED;
5709 eir_len = 0;
5710 } else {
5711 status = MGMT_STATUS_SUCCESS;
5712 rp = (void *)skb->data;
5713
5714 eir_len = 5 + 18 + 18;
5715 h192 = rp->hash;
5716 r192 = rp->rand;
5717 h256 = NULL;
5718 r256 = NULL;
5719 }
5720 } else {
5721 struct hci_rp_read_local_oob_ext_data *rp;
5722
5723 if (skb->len != sizeof(*rp)) {
5724 status = MGMT_STATUS_FAILED;
5725 eir_len = 0;
5726 } else {
5727 status = MGMT_STATUS_SUCCESS;
5728 rp = (void *)skb->data;
5729
5730 if (hci_dev_test_flag(hdev, HCI_SC_ONLY)) {
5731 eir_len = 5 + 18 + 18;
5732 h192 = NULL;
5733 r192 = NULL;
5734 } else {
5735 eir_len = 5 + 18 + 18 + 18 + 18;
5736 h192 = rp->hash192;
5737 r192 = rp->rand192;
5738 }
5739
5740 h256 = rp->hash256;
5741 r256 = rp->rand256;
5742 }
5743 }
5744
5745 mgmt_rp = kmalloc(sizeof(*mgmt_rp) + eir_len, GFP_KERNEL);
5746 if (!mgmt_rp)
5747 goto done;
5748
5749 if (status)
5750 goto send_rsp;
5751
5752 eir_len = eir_append_data(mgmt_rp->eir, 0, EIR_CLASS_OF_DEV,
5753 hdev->dev_class, 3);
5754
5755 if (h192 && r192) {
5756 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5757 EIR_SSP_HASH_C192, h192, 16);
5758 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5759 EIR_SSP_RAND_R192, r192, 16);
5760 }
5761
5762 if (h256 && r256) {
5763 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5764 EIR_SSP_HASH_C256, h256, 16);
5765 eir_len = eir_append_data(mgmt_rp->eir, eir_len,
5766 EIR_SSP_RAND_R256, r256, 16);
5767 }
5768
5769send_rsp:
5770 mgmt_rp->type = mgmt_cp->type;
5771 mgmt_rp->eir_len = cpu_to_le16(eir_len);
5772
5773 err = mgmt_cmd_complete(cmd->sk, hdev->id,
5774 MGMT_OP_READ_LOCAL_OOB_EXT_DATA, status,
5775 mgmt_rp, sizeof(*mgmt_rp) + eir_len);
5776 if (err < 0 || status)
5777 goto done;
5778
5779 hci_sock_set_flag(cmd->sk, HCI_MGMT_OOB_DATA_EVENTS);
5780
5781 err = mgmt_limited_event(MGMT_EV_LOCAL_OOB_DATA_UPDATED, hdev,
5782 mgmt_rp, sizeof(*mgmt_rp) + eir_len,
5783 HCI_MGMT_OOB_DATA_EVENTS, cmd->sk);
5784done:
5785 kfree(mgmt_rp);
5786 mgmt_pending_remove(cmd);
5787}
5788
5789static int read_local_ssp_oob_req(struct hci_dev *hdev, struct sock *sk,
5790 struct mgmt_cp_read_local_oob_ext_data *cp)
5791{
5792 struct mgmt_pending_cmd *cmd;
5793 struct hci_request req;
5794 int err;
5795
5796 cmd = mgmt_pending_add(sk, MGMT_OP_READ_LOCAL_OOB_EXT_DATA, hdev,
5797 cp, sizeof(*cp));
5798 if (!cmd)
5799 return -ENOMEM;
5800
5801 hci_req_init(&req, hdev);
5802
5803 if (bredr_sc_enabled(hdev))
5804 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_EXT_DATA, 0, NULL);
5805 else
5806 hci_req_add(&req, HCI_OP_READ_LOCAL_OOB_DATA, 0, NULL);
5807
5808 err = hci_req_run_skb(&req, read_local_oob_ext_data_complete);
5809 if (err < 0) {
5810 mgmt_pending_remove(cmd);
5811 return err;
5812 }
5813
5814 return 0;
5815}
5816
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005817static int read_local_oob_ext_data(struct sock *sk, struct hci_dev *hdev,
5818 void *data, u16 data_len)
5819{
5820 struct mgmt_cp_read_local_oob_ext_data *cp = data;
5821 struct mgmt_rp_read_local_oob_ext_data *rp;
5822 size_t rp_len;
5823 u16 eir_len;
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005824 u8 status, flags, role, addr[7], hash[16], rand[16];
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005825 int err;
5826
5827 BT_DBG("%s", hdev->name);
5828
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005829 if (hdev_is_powered(hdev)) {
5830 switch (cp->type) {
5831 case BIT(BDADDR_BREDR):
5832 status = mgmt_bredr_support(hdev);
5833 if (status)
5834 eir_len = 0;
5835 else
5836 eir_len = 5;
5837 break;
5838 case (BIT(BDADDR_LE_PUBLIC) | BIT(BDADDR_LE_RANDOM)):
5839 status = mgmt_le_support(hdev);
5840 if (status)
5841 eir_len = 0;
5842 else
5843 eir_len = 9 + 3 + 18 + 18 + 3;
5844 break;
5845 default:
5846 status = MGMT_STATUS_INVALID_PARAMS;
5847 eir_len = 0;
5848 break;
5849 }
5850 } else {
5851 status = MGMT_STATUS_NOT_POWERED;
5852 eir_len = 0;
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005853 }
5854
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005855 rp_len = sizeof(*rp) + eir_len;
5856 rp = kmalloc(rp_len, GFP_ATOMIC);
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005857 if (!rp)
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005858 return -ENOMEM;
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005859
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005860 if (status)
5861 goto complete;
5862
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005863 hci_dev_lock(hdev);
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005864
5865 eir_len = 0;
5866 switch (cp->type) {
5867 case BIT(BDADDR_BREDR):
Johan Hedberg40f66c02015-04-07 21:52:22 +03005868 if (hci_dev_test_flag(hdev, HCI_SSP_ENABLED)) {
5869 err = read_local_ssp_oob_req(hdev, sk, cp);
5870 hci_dev_unlock(hdev);
5871 if (!err)
5872 goto done;
5873
5874 status = MGMT_STATUS_FAILED;
5875 goto complete;
5876 } else {
5877 eir_len = eir_append_data(rp->eir, eir_len,
5878 EIR_CLASS_OF_DEV,
5879 hdev->dev_class, 3);
5880 }
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005881 break;
5882 case (BIT(BDADDR_LE_PUBLIC) | BIT(BDADDR_LE_RANDOM)):
Marcel Holtmann5082a592015-03-16 12:39:00 -07005883 if (hci_dev_test_flag(hdev, HCI_SC_ENABLED) &&
5884 smp_generate_oob(hdev, hash, rand) < 0) {
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005885 hci_dev_unlock(hdev);
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005886 status = MGMT_STATUS_FAILED;
5887 goto complete;
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005888 }
5889
Marcel Holtmanne2135682015-04-02 12:00:58 -07005890 /* This should return the active RPA, but since the RPA
5891 * is only programmed on demand, it is really hard to fill
5892 * this in at the moment. For now disallow retrieving
5893 * local out-of-band data when privacy is in use.
5894 *
5895 * Returning the identity address will not help here since
5896 * pairing happens before the identity resolving key is
5897 * known and thus the connection establishment happens
5898 * based on the RPA and not the identity address.
5899 */
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005900 if (hci_dev_test_flag(hdev, HCI_PRIVACY)) {
Marcel Holtmanne2135682015-04-02 12:00:58 -07005901 hci_dev_unlock(hdev);
5902 status = MGMT_STATUS_REJECTED;
5903 goto complete;
5904 }
5905
5906 if (hci_dev_test_flag(hdev, HCI_FORCE_STATIC_ADDR) ||
5907 !bacmp(&hdev->bdaddr, BDADDR_ANY) ||
5908 (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED) &&
5909 bacmp(&hdev->static_addr, BDADDR_ANY))) {
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005910 memcpy(addr, &hdev->static_addr, 6);
5911 addr[6] = 0x01;
5912 } else {
5913 memcpy(addr, &hdev->bdaddr, 6);
5914 addr[6] = 0x00;
5915 }
5916
5917 eir_len = eir_append_data(rp->eir, eir_len, EIR_LE_BDADDR,
5918 addr, sizeof(addr));
5919
5920 if (hci_dev_test_flag(hdev, HCI_ADVERTISING))
5921 role = 0x02;
5922 else
5923 role = 0x01;
5924
5925 eir_len = eir_append_data(rp->eir, eir_len, EIR_LE_ROLE,
5926 &role, sizeof(role));
5927
Marcel Holtmann5082a592015-03-16 12:39:00 -07005928 if (hci_dev_test_flag(hdev, HCI_SC_ENABLED)) {
5929 eir_len = eir_append_data(rp->eir, eir_len,
5930 EIR_LE_SC_CONFIRM,
5931 hash, sizeof(hash));
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005932
Marcel Holtmann5082a592015-03-16 12:39:00 -07005933 eir_len = eir_append_data(rp->eir, eir_len,
5934 EIR_LE_SC_RANDOM,
5935 rand, sizeof(rand));
5936 }
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005937
Johan Hedbergf2252572015-11-18 12:49:20 +02005938 flags = mgmt_get_adv_discov_flags(hdev);
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005939
5940 if (!hci_dev_test_flag(hdev, HCI_BREDR_ENABLED))
5941 flags |= LE_AD_NO_BREDR;
5942
5943 eir_len = eir_append_data(rp->eir, eir_len, EIR_FLAGS,
5944 &flags, sizeof(flags));
5945 break;
5946 }
5947
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005948 hci_dev_unlock(hdev);
5949
Marcel Holtmann72000df2015-03-16 16:11:21 -07005950 hci_sock_set_flag(sk, HCI_MGMT_OOB_DATA_EVENTS);
5951
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005952 status = MGMT_STATUS_SUCCESS;
5953
5954complete:
Marcel Holtmannefcd8c92015-03-28 15:18:58 -07005955 rp->type = cp->type;
5956 rp->eir_len = cpu_to_le16(eir_len);
5957
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005958 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_READ_LOCAL_OOB_EXT_DATA,
Marcel Holtmann57b0d3e2015-03-28 15:18:59 -07005959 status, rp, sizeof(*rp) + eir_len);
5960 if (err < 0 || status)
Marcel Holtmann72000df2015-03-16 16:11:21 -07005961 goto done;
5962
5963 err = mgmt_limited_event(MGMT_EV_LOCAL_OOB_DATA_UPDATED, hdev,
5964 rp, sizeof(*rp) + eir_len,
5965 HCI_MGMT_OOB_DATA_EVENTS, sk);
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005966
Marcel Holtmann0821a2c2015-03-16 01:10:23 -07005967done:
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07005968 kfree(rp);
5969
5970 return err;
5971}
5972
Arman Uguray089fa8c2015-03-25 18:53:45 -07005973static u32 get_supported_adv_flags(struct hci_dev *hdev)
5974{
5975 u32 flags = 0;
5976
5977 flags |= MGMT_ADV_FLAG_CONNECTABLE;
5978 flags |= MGMT_ADV_FLAG_DISCOV;
5979 flags |= MGMT_ADV_FLAG_LIMITED_DISCOV;
5980 flags |= MGMT_ADV_FLAG_MANAGED_FLAGS;
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02005981 flags |= MGMT_ADV_FLAG_APPEARANCE;
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02005982 flags |= MGMT_ADV_FLAG_LOCAL_NAME;
Arman Uguray089fa8c2015-03-25 18:53:45 -07005983
5984 if (hdev->adv_tx_power != HCI_TX_POWER_INVALID)
5985 flags |= MGMT_ADV_FLAG_TX_POWER;
5986
5987 return flags;
5988}
5989
Marcel Holtmannd3d53052015-03-14 20:53:25 -07005990static int read_adv_features(struct sock *sk, struct hci_dev *hdev,
5991 void *data, u16 data_len)
5992{
5993 struct mgmt_rp_read_adv_features *rp;
5994 size_t rp_len;
Johan Hedberg02c04afe2015-11-26 12:15:58 +02005995 int err;
Florian Grandel286e0c82015-06-18 03:16:38 +02005996 struct adv_info *adv_instance;
Arman Uguray089fa8c2015-03-25 18:53:45 -07005997 u32 supported_flags;
Johan Hedberg02c04afe2015-11-26 12:15:58 +02005998 u8 *instance;
Marcel Holtmannd3d53052015-03-14 20:53:25 -07005999
6000 BT_DBG("%s", hdev->name);
6001
Arman Uguray089fa8c2015-03-25 18:53:45 -07006002 if (!lmp_le_capable(hdev))
6003 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_READ_ADV_FEATURES,
6004 MGMT_STATUS_REJECTED);
6005
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006006 hci_dev_lock(hdev);
6007
Johan Hedberg02c04afe2015-11-26 12:15:58 +02006008 rp_len = sizeof(*rp) + hdev->adv_instance_cnt;
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006009 rp = kmalloc(rp_len, GFP_ATOMIC);
6010 if (!rp) {
6011 hci_dev_unlock(hdev);
6012 return -ENOMEM;
6013 }
6014
Arman Uguray089fa8c2015-03-25 18:53:45 -07006015 supported_flags = get_supported_adv_flags(hdev);
6016
6017 rp->supported_flags = cpu_to_le32(supported_flags);
Marcel Holtmanndc5d82a2015-03-19 17:22:25 -07006018 rp->max_adv_data_len = HCI_MAX_AD_LENGTH;
6019 rp->max_scan_rsp_len = HCI_MAX_AD_LENGTH;
Florian Grandeld2609b32015-06-18 03:16:34 +02006020 rp->max_instances = HCI_MAX_ADV_INSTANCES;
Johan Hedberg02c04afe2015-11-26 12:15:58 +02006021 rp->num_instances = hdev->adv_instance_cnt;
Arman Uguray24b4f382015-03-23 15:57:12 -07006022
Johan Hedberg02c04afe2015-11-26 12:15:58 +02006023 instance = rp->instance;
6024 list_for_each_entry(adv_instance, &hdev->adv_instances, list) {
6025 *instance = adv_instance->instance;
6026 instance++;
Arman Uguray24b4f382015-03-23 15:57:12 -07006027 }
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006028
6029 hci_dev_unlock(hdev);
6030
6031 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_READ_ADV_FEATURES,
6032 MGMT_STATUS_SUCCESS, rp, rp_len);
6033
6034 kfree(rp);
6035
6036 return err;
6037}
6038
Szymon Janc2bb368702016-09-18 12:50:05 +02006039static u8 tlv_data_max_len(u32 adv_flags, bool is_adv_data)
Arman Uguray24b4f382015-03-23 15:57:12 -07006040{
Arman Uguray4117ed72015-03-23 15:57:14 -07006041 u8 max_len = HCI_MAX_AD_LENGTH;
Arman Uguray24b4f382015-03-23 15:57:12 -07006042
Marcel Holtmann31a32482015-11-19 16:16:42 +01006043 if (is_adv_data) {
6044 if (adv_flags & (MGMT_ADV_FLAG_DISCOV |
6045 MGMT_ADV_FLAG_LIMITED_DISCOV |
Szymon Janc2bb368702016-09-18 12:50:05 +02006046 MGMT_ADV_FLAG_MANAGED_FLAGS))
Marcel Holtmann31a32482015-11-19 16:16:42 +01006047 max_len -= 3;
Arman Uguray24b4f382015-03-23 15:57:12 -07006048
Szymon Janc2bb368702016-09-18 12:50:05 +02006049 if (adv_flags & MGMT_ADV_FLAG_TX_POWER)
Marcel Holtmann31a32482015-11-19 16:16:42 +01006050 max_len -= 3;
MichaƂ Narajowski7c295c42016-09-18 12:50:02 +02006051 } else {
6052 /* at least 1 byte of name should fit in */
6053 if (adv_flags & MGMT_ADV_FLAG_LOCAL_NAME)
6054 max_len -= 3;
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02006055
Szymon Janc2bb368702016-09-18 12:50:05 +02006056 if (adv_flags & (MGMT_ADV_FLAG_APPEARANCE))
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02006057 max_len -= 4;
Arman Uguray5507e352015-03-25 18:53:44 -07006058 }
6059
Szymon Janc2bb368702016-09-18 12:50:05 +02006060 return max_len;
6061}
6062
6063static bool flags_managed(u32 adv_flags)
6064{
6065 return adv_flags & (MGMT_ADV_FLAG_DISCOV |
6066 MGMT_ADV_FLAG_LIMITED_DISCOV |
6067 MGMT_ADV_FLAG_MANAGED_FLAGS);
6068}
6069
6070static bool tx_power_managed(u32 adv_flags)
6071{
6072 return adv_flags & MGMT_ADV_FLAG_TX_POWER;
6073}
6074
6075static bool name_managed(u32 adv_flags)
6076{
6077 return adv_flags & MGMT_ADV_FLAG_LOCAL_NAME;
6078}
6079
6080static bool appearance_managed(u32 adv_flags)
6081{
6082 return adv_flags & MGMT_ADV_FLAG_APPEARANCE;
6083}
6084
6085static bool tlv_data_is_valid(u32 adv_flags, u8 *data, u8 len, bool is_adv_data)
6086{
6087 int i, cur_len;
6088 u8 max_len;
6089
6090 max_len = tlv_data_max_len(adv_flags, is_adv_data);
6091
Arman Uguray4117ed72015-03-23 15:57:14 -07006092 if (len > max_len)
Arman Uguray24b4f382015-03-23 15:57:12 -07006093 return false;
6094
Arman Uguray4117ed72015-03-23 15:57:14 -07006095 /* Make sure that the data is correctly formatted. */
6096 for (i = 0, cur_len = 0; i < len; i += (cur_len + 1)) {
6097 cur_len = data[i];
Arman Uguray24b4f382015-03-23 15:57:12 -07006098
Szymon Janc9c9db782016-09-18 12:50:06 +02006099 if (data[i + 1] == EIR_FLAGS &&
6100 (!is_adv_data || flags_managed(adv_flags)))
Arman Ugurayb44133f2015-03-25 18:53:41 -07006101 return false;
6102
Szymon Janc2bb368702016-09-18 12:50:05 +02006103 if (data[i + 1] == EIR_TX_POWER && tx_power_managed(adv_flags))
6104 return false;
6105
6106 if (data[i + 1] == EIR_NAME_COMPLETE && name_managed(adv_flags))
6107 return false;
6108
6109 if (data[i + 1] == EIR_NAME_SHORT && name_managed(adv_flags))
6110 return false;
6111
6112 if (data[i + 1] == EIR_APPEARANCE &&
6113 appearance_managed(adv_flags))
Arman Uguray5507e352015-03-25 18:53:44 -07006114 return false;
6115
Arman Uguray24b4f382015-03-23 15:57:12 -07006116 /* If the current field length would exceed the total data
6117 * length, then it's invalid.
6118 */
Arman Uguray4117ed72015-03-23 15:57:14 -07006119 if (i + cur_len >= len)
Arman Uguray24b4f382015-03-23 15:57:12 -07006120 return false;
6121 }
6122
6123 return true;
6124}
6125
Arman Uguray24b4f382015-03-23 15:57:12 -07006126static void add_advertising_complete(struct hci_dev *hdev, u8 status,
6127 u16 opcode)
6128{
6129 struct mgmt_pending_cmd *cmd;
Florian Grandelfffd38b2015-06-18 03:16:47 +02006130 struct mgmt_cp_add_advertising *cp;
Arman Uguray24b4f382015-03-23 15:57:12 -07006131 struct mgmt_rp_add_advertising rp;
Florian Grandelfffd38b2015-06-18 03:16:47 +02006132 struct adv_info *adv_instance, *n;
6133 u8 instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006134
6135 BT_DBG("status %d", status);
6136
6137 hci_dev_lock(hdev);
6138
6139 cmd = pending_find(MGMT_OP_ADD_ADVERTISING, hdev);
6140
Florian Grandelfffd38b2015-06-18 03:16:47 +02006141 list_for_each_entry_safe(adv_instance, n, &hdev->adv_instances, list) {
6142 if (!adv_instance->pending)
6143 continue;
6144
6145 if (!status) {
6146 adv_instance->pending = false;
6147 continue;
6148 }
6149
6150 instance = adv_instance->instance;
6151
6152 if (hdev->cur_adv_instance == instance)
6153 cancel_adv_timeout(hdev);
6154
6155 hci_remove_adv_instance(hdev, instance);
Johan Hedbergf2252572015-11-18 12:49:20 +02006156 mgmt_advertising_removed(cmd ? cmd->sk : NULL, hdev, instance);
Arman Uguray24b4f382015-03-23 15:57:12 -07006157 }
6158
6159 if (!cmd)
6160 goto unlock;
6161
Florian Grandelfffd38b2015-06-18 03:16:47 +02006162 cp = cmd->param;
6163 rp.instance = cp->instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006164
6165 if (status)
6166 mgmt_cmd_status(cmd->sk, cmd->index, cmd->opcode,
6167 mgmt_status(status));
6168 else
6169 mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode,
6170 mgmt_status(status), &rp, sizeof(rp));
6171
6172 mgmt_pending_remove(cmd);
6173
6174unlock:
6175 hci_dev_unlock(hdev);
6176}
6177
6178static int add_advertising(struct sock *sk, struct hci_dev *hdev,
6179 void *data, u16 data_len)
6180{
6181 struct mgmt_cp_add_advertising *cp = data;
6182 struct mgmt_rp_add_advertising rp;
6183 u32 flags;
Arman Uguray089fa8c2015-03-25 18:53:45 -07006184 u32 supported_flags;
Arman Uguray24b4f382015-03-23 15:57:12 -07006185 u8 status;
Florian Grandelfffd38b2015-06-18 03:16:47 +02006186 u16 timeout, duration;
6187 unsigned int prev_instance_cnt = hdev->adv_instance_cnt;
6188 u8 schedule_instance = 0;
6189 struct adv_info *next_instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006190 int err;
6191 struct mgmt_pending_cmd *cmd;
6192 struct hci_request req;
6193
6194 BT_DBG("%s", hdev->name);
6195
6196 status = mgmt_le_support(hdev);
6197 if (status)
6198 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6199 status);
6200
Marcel Holtmannceff86a2015-11-19 16:16:41 +01006201 if (cp->instance < 1 || cp->instance > HCI_MAX_ADV_INSTANCES)
6202 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6203 MGMT_STATUS_INVALID_PARAMS);
6204
Johan Hedberg6a0e7802016-03-11 09:56:33 +02006205 if (data_len != sizeof(*cp) + cp->adv_data_len + cp->scan_rsp_len)
6206 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6207 MGMT_STATUS_INVALID_PARAMS);
6208
Arman Uguray24b4f382015-03-23 15:57:12 -07006209 flags = __le32_to_cpu(cp->flags);
Arman Uguray912098a2015-03-23 15:57:15 -07006210 timeout = __le16_to_cpu(cp->timeout);
Florian Grandelfffd38b2015-06-18 03:16:47 +02006211 duration = __le16_to_cpu(cp->duration);
Arman Uguray24b4f382015-03-23 15:57:12 -07006212
Florian Grandelfffd38b2015-06-18 03:16:47 +02006213 /* The current implementation only supports a subset of the specified
6214 * flags.
Arman Uguray089fa8c2015-03-25 18:53:45 -07006215 */
6216 supported_flags = get_supported_adv_flags(hdev);
Florian Grandelfffd38b2015-06-18 03:16:47 +02006217 if (flags & ~supported_flags)
Arman Uguray24b4f382015-03-23 15:57:12 -07006218 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6219 MGMT_STATUS_INVALID_PARAMS);
6220
6221 hci_dev_lock(hdev);
6222
Arman Uguray912098a2015-03-23 15:57:15 -07006223 if (timeout && !hdev_is_powered(hdev)) {
6224 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6225 MGMT_STATUS_REJECTED);
6226 goto unlock;
6227 }
6228
Arman Uguray24b4f382015-03-23 15:57:12 -07006229 if (pending_find(MGMT_OP_ADD_ADVERTISING, hdev) ||
Arman Ugurayda9293352015-03-23 15:57:13 -07006230 pending_find(MGMT_OP_REMOVE_ADVERTISING, hdev) ||
Arman Uguray24b4f382015-03-23 15:57:12 -07006231 pending_find(MGMT_OP_SET_LE, hdev)) {
6232 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6233 MGMT_STATUS_BUSY);
6234 goto unlock;
6235 }
6236
Szymon Janc5e2c59e2016-09-18 12:50:04 +02006237 if (!tlv_data_is_valid(flags, cp->data, cp->adv_data_len, true) ||
6238 !tlv_data_is_valid(flags, cp->data + cp->adv_data_len,
Arman Ugurayb44133f2015-03-25 18:53:41 -07006239 cp->scan_rsp_len, false)) {
Arman Uguray24b4f382015-03-23 15:57:12 -07006240 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6241 MGMT_STATUS_INVALID_PARAMS);
6242 goto unlock;
6243 }
6244
Florian Grandelfffd38b2015-06-18 03:16:47 +02006245 err = hci_add_adv_instance(hdev, cp->instance, flags,
6246 cp->adv_data_len, cp->data,
6247 cp->scan_rsp_len,
6248 cp->data + cp->adv_data_len,
6249 timeout, duration);
6250 if (err < 0) {
6251 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6252 MGMT_STATUS_FAILED);
6253 goto unlock;
6254 }
Arman Uguray24b4f382015-03-23 15:57:12 -07006255
Florian Grandelfffd38b2015-06-18 03:16:47 +02006256 /* Only trigger an advertising added event if a new instance was
6257 * actually added.
6258 */
6259 if (hdev->adv_instance_cnt > prev_instance_cnt)
Johan Hedbergf2252572015-11-18 12:49:20 +02006260 mgmt_advertising_added(sk, hdev, cp->instance);
Arman Uguray24b4f382015-03-23 15:57:12 -07006261
Florian Grandelfffd38b2015-06-18 03:16:47 +02006262 if (hdev->cur_adv_instance == cp->instance) {
6263 /* If the currently advertised instance is being changed then
6264 * cancel the current advertising and schedule the next
6265 * instance. If there is only one instance then the overridden
6266 * advertising data will be visible right away.
6267 */
6268 cancel_adv_timeout(hdev);
Arman Uguray912098a2015-03-23 15:57:15 -07006269
Florian Grandelfffd38b2015-06-18 03:16:47 +02006270 next_instance = hci_get_next_instance(hdev, cp->instance);
6271 if (next_instance)
6272 schedule_instance = next_instance->instance;
6273 } else if (!hdev->adv_instance_timeout) {
6274 /* Immediately advertise the new instance if no other
6275 * instance is currently being advertised.
6276 */
6277 schedule_instance = cp->instance;
6278 }
Arman Uguray912098a2015-03-23 15:57:15 -07006279
Florian Grandelfffd38b2015-06-18 03:16:47 +02006280 /* If the HCI_ADVERTISING flag is set or the device isn't powered or
6281 * there is no instance to be advertised then we have no HCI
6282 * communication to make. Simply return.
Arman Uguray24b4f382015-03-23 15:57:12 -07006283 */
6284 if (!hdev_is_powered(hdev) ||
Florian Grandelfffd38b2015-06-18 03:16:47 +02006285 hci_dev_test_flag(hdev, HCI_ADVERTISING) ||
6286 !schedule_instance) {
6287 rp.instance = cp->instance;
Arman Uguray24b4f382015-03-23 15:57:12 -07006288 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,
6289 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
6290 goto unlock;
6291 }
6292
6293 /* We're good to go, update advertising data, parameters, and start
6294 * advertising.
6295 */
6296 cmd = mgmt_pending_add(sk, MGMT_OP_ADD_ADVERTISING, hdev, data,
6297 data_len);
6298 if (!cmd) {
6299 err = -ENOMEM;
6300 goto unlock;
6301 }
6302
6303 hci_req_init(&req, hdev);
6304
Johan Hedbergf2252572015-11-18 12:49:20 +02006305 err = __hci_req_schedule_adv_instance(&req, schedule_instance, true);
Arman Uguray24b4f382015-03-23 15:57:12 -07006306
Florian Grandelfffd38b2015-06-18 03:16:47 +02006307 if (!err)
6308 err = hci_req_run(&req, add_advertising_complete);
6309
Arman Uguray24b4f382015-03-23 15:57:12 -07006310 if (err < 0)
6311 mgmt_pending_remove(cmd);
6312
6313unlock:
6314 hci_dev_unlock(hdev);
6315
6316 return err;
6317}
6318
Arman Ugurayda9293352015-03-23 15:57:13 -07006319static void remove_advertising_complete(struct hci_dev *hdev, u8 status,
6320 u16 opcode)
6321{
6322 struct mgmt_pending_cmd *cmd;
Florian Grandel01948332015-06-18 03:16:48 +02006323 struct mgmt_cp_remove_advertising *cp;
Arman Ugurayda9293352015-03-23 15:57:13 -07006324 struct mgmt_rp_remove_advertising rp;
6325
6326 BT_DBG("status %d", status);
6327
6328 hci_dev_lock(hdev);
6329
6330 /* A failure status here only means that we failed to disable
6331 * advertising. Otherwise, the advertising instance has been removed,
6332 * so report success.
6333 */
6334 cmd = pending_find(MGMT_OP_REMOVE_ADVERTISING, hdev);
6335 if (!cmd)
6336 goto unlock;
6337
Florian Grandel01948332015-06-18 03:16:48 +02006338 cp = cmd->param;
6339 rp.instance = cp->instance;
Arman Ugurayda9293352015-03-23 15:57:13 -07006340
6341 mgmt_cmd_complete(cmd->sk, cmd->index, cmd->opcode, MGMT_STATUS_SUCCESS,
6342 &rp, sizeof(rp));
6343 mgmt_pending_remove(cmd);
6344
6345unlock:
6346 hci_dev_unlock(hdev);
6347}
6348
6349static int remove_advertising(struct sock *sk, struct hci_dev *hdev,
6350 void *data, u16 data_len)
6351{
6352 struct mgmt_cp_remove_advertising *cp = data;
6353 struct mgmt_rp_remove_advertising rp;
Arman Ugurayda9293352015-03-23 15:57:13 -07006354 struct mgmt_pending_cmd *cmd;
6355 struct hci_request req;
Johan Hedberg952497b2015-06-18 21:05:31 +03006356 int err;
Arman Ugurayda9293352015-03-23 15:57:13 -07006357
6358 BT_DBG("%s", hdev->name);
6359
Arman Ugurayda9293352015-03-23 15:57:13 -07006360 hci_dev_lock(hdev);
6361
Johan Hedberg952497b2015-06-18 21:05:31 +03006362 if (cp->instance && !hci_find_adv_instance(hdev, cp->instance)) {
Florian Grandel01948332015-06-18 03:16:48 +02006363 err = mgmt_cmd_status(sk, hdev->id,
6364 MGMT_OP_REMOVE_ADVERTISING,
6365 MGMT_STATUS_INVALID_PARAMS);
6366 goto unlock;
6367 }
6368
Arman Ugurayda9293352015-03-23 15:57:13 -07006369 if (pending_find(MGMT_OP_ADD_ADVERTISING, hdev) ||
6370 pending_find(MGMT_OP_REMOVE_ADVERTISING, hdev) ||
6371 pending_find(MGMT_OP_SET_LE, hdev)) {
6372 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_REMOVE_ADVERTISING,
6373 MGMT_STATUS_BUSY);
6374 goto unlock;
6375 }
6376
Johan Hedberg17fd08f2015-11-26 12:15:59 +02006377 if (list_empty(&hdev->adv_instances)) {
Arman Ugurayda9293352015-03-23 15:57:13 -07006378 err = mgmt_cmd_status(sk, hdev->id, MGMT_OP_REMOVE_ADVERTISING,
6379 MGMT_STATUS_INVALID_PARAMS);
6380 goto unlock;
6381 }
6382
Florian Grandel01948332015-06-18 03:16:48 +02006383 hci_req_init(&req, hdev);
Arman Uguray912098a2015-03-23 15:57:15 -07006384
Johan Hedberg37d3a1f2016-08-28 20:53:34 +03006385 hci_req_clear_adv_instance(hdev, sk, &req, cp->instance, true);
Arman Ugurayda9293352015-03-23 15:57:13 -07006386
Florian Grandel01948332015-06-18 03:16:48 +02006387 if (list_empty(&hdev->adv_instances))
Johan Hedbergf2252572015-11-18 12:49:20 +02006388 __hci_req_disable_advertising(&req);
Arman Ugurayda9293352015-03-23 15:57:13 -07006389
Florian Grandel01948332015-06-18 03:16:48 +02006390 /* If no HCI commands have been collected so far or the HCI_ADVERTISING
6391 * flag is set or the device isn't powered then we have no HCI
6392 * communication to make. Simply return.
Arman Ugurayda9293352015-03-23 15:57:13 -07006393 */
Florian Grandel01948332015-06-18 03:16:48 +02006394 if (skb_queue_empty(&req.cmd_q) ||
6395 !hdev_is_powered(hdev) ||
Arman Ugurayda9293352015-03-23 15:57:13 -07006396 hci_dev_test_flag(hdev, HCI_ADVERTISING)) {
Florian Grandel01948332015-06-18 03:16:48 +02006397 rp.instance = cp->instance;
Arman Ugurayda9293352015-03-23 15:57:13 -07006398 err = mgmt_cmd_complete(sk, hdev->id,
6399 MGMT_OP_REMOVE_ADVERTISING,
6400 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
6401 goto unlock;
6402 }
6403
6404 cmd = mgmt_pending_add(sk, MGMT_OP_REMOVE_ADVERTISING, hdev, data,
6405 data_len);
6406 if (!cmd) {
6407 err = -ENOMEM;
6408 goto unlock;
6409 }
6410
Arman Ugurayda9293352015-03-23 15:57:13 -07006411 err = hci_req_run(&req, remove_advertising_complete);
6412 if (err < 0)
6413 mgmt_pending_remove(cmd);
6414
6415unlock:
6416 hci_dev_unlock(hdev);
6417
6418 return err;
6419}
6420
Marcel Holtmann40b25fe2015-11-19 16:16:43 +01006421static int get_adv_size_info(struct sock *sk, struct hci_dev *hdev,
6422 void *data, u16 data_len)
6423{
6424 struct mgmt_cp_get_adv_size_info *cp = data;
6425 struct mgmt_rp_get_adv_size_info rp;
6426 u32 flags, supported_flags;
6427 int err;
6428
6429 BT_DBG("%s", hdev->name);
6430
6431 if (!lmp_le_capable(hdev))
6432 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6433 MGMT_STATUS_REJECTED);
6434
6435 if (cp->instance < 1 || cp->instance > HCI_MAX_ADV_INSTANCES)
6436 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6437 MGMT_STATUS_INVALID_PARAMS);
6438
6439 flags = __le32_to_cpu(cp->flags);
6440
6441 /* The current implementation only supports a subset of the specified
6442 * flags.
6443 */
6444 supported_flags = get_supported_adv_flags(hdev);
6445 if (flags & ~supported_flags)
6446 return mgmt_cmd_status(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6447 MGMT_STATUS_INVALID_PARAMS);
6448
6449 rp.instance = cp->instance;
6450 rp.flags = cp->flags;
6451 rp.max_adv_data_len = tlv_data_max_len(flags, true);
6452 rp.max_scan_rsp_len = tlv_data_max_len(flags, false);
6453
6454 err = mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_ADV_SIZE_INFO,
6455 MGMT_STATUS_SUCCESS, &rp, sizeof(rp));
6456
6457 return err;
6458}
6459
Johan Hedberg6d785aa32015-03-06 21:08:51 +02006460static const struct hci_mgmt_handler mgmt_handlers[] = {
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02006461 { NULL }, /* 0x0000 (no command) */
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006462 { read_version, MGMT_READ_VERSION_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006463 HCI_MGMT_NO_HDEV |
6464 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006465 { read_commands, MGMT_READ_COMMANDS_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006466 HCI_MGMT_NO_HDEV |
6467 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006468 { read_index_list, MGMT_READ_INDEX_LIST_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006469 HCI_MGMT_NO_HDEV |
6470 HCI_MGMT_UNTRUSTED },
6471 { read_controller_info, MGMT_READ_INFO_SIZE,
6472 HCI_MGMT_UNTRUSTED },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006473 { set_powered, MGMT_SETTING_SIZE },
6474 { set_discoverable, MGMT_SET_DISCOVERABLE_SIZE },
6475 { set_connectable, MGMT_SETTING_SIZE },
6476 { set_fast_connectable, MGMT_SETTING_SIZE },
6477 { set_bondable, MGMT_SETTING_SIZE },
6478 { set_link_security, MGMT_SETTING_SIZE },
6479 { set_ssp, MGMT_SETTING_SIZE },
6480 { set_hs, MGMT_SETTING_SIZE },
6481 { set_le, MGMT_SETTING_SIZE },
6482 { set_dev_class, MGMT_SET_DEV_CLASS_SIZE },
6483 { set_local_name, MGMT_SET_LOCAL_NAME_SIZE },
6484 { add_uuid, MGMT_ADD_UUID_SIZE },
6485 { remove_uuid, MGMT_REMOVE_UUID_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006486 { load_link_keys, MGMT_LOAD_LINK_KEYS_SIZE,
6487 HCI_MGMT_VAR_LEN },
6488 { load_long_term_keys, MGMT_LOAD_LONG_TERM_KEYS_SIZE,
6489 HCI_MGMT_VAR_LEN },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006490 { disconnect, MGMT_DISCONNECT_SIZE },
6491 { get_connections, MGMT_GET_CONNECTIONS_SIZE },
6492 { pin_code_reply, MGMT_PIN_CODE_REPLY_SIZE },
6493 { pin_code_neg_reply, MGMT_PIN_CODE_NEG_REPLY_SIZE },
6494 { set_io_capability, MGMT_SET_IO_CAPABILITY_SIZE },
6495 { pair_device, MGMT_PAIR_DEVICE_SIZE },
6496 { cancel_pair_device, MGMT_CANCEL_PAIR_DEVICE_SIZE },
6497 { unpair_device, MGMT_UNPAIR_DEVICE_SIZE },
6498 { user_confirm_reply, MGMT_USER_CONFIRM_REPLY_SIZE },
6499 { user_confirm_neg_reply, MGMT_USER_CONFIRM_NEG_REPLY_SIZE },
6500 { user_passkey_reply, MGMT_USER_PASSKEY_REPLY_SIZE },
6501 { user_passkey_neg_reply, MGMT_USER_PASSKEY_NEG_REPLY_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006502 { read_local_oob_data, MGMT_READ_LOCAL_OOB_DATA_SIZE },
6503 { add_remote_oob_data, MGMT_ADD_REMOTE_OOB_DATA_SIZE,
6504 HCI_MGMT_VAR_LEN },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006505 { remove_remote_oob_data, MGMT_REMOVE_REMOTE_OOB_DATA_SIZE },
6506 { start_discovery, MGMT_START_DISCOVERY_SIZE },
6507 { stop_discovery, MGMT_STOP_DISCOVERY_SIZE },
6508 { confirm_name, MGMT_CONFIRM_NAME_SIZE },
6509 { block_device, MGMT_BLOCK_DEVICE_SIZE },
6510 { unblock_device, MGMT_UNBLOCK_DEVICE_SIZE },
6511 { set_device_id, MGMT_SET_DEVICE_ID_SIZE },
6512 { set_advertising, MGMT_SETTING_SIZE },
6513 { set_bredr, MGMT_SETTING_SIZE },
6514 { set_static_address, MGMT_SET_STATIC_ADDRESS_SIZE },
6515 { set_scan_params, MGMT_SET_SCAN_PARAMS_SIZE },
6516 { set_secure_conn, MGMT_SETTING_SIZE },
6517 { set_debug_keys, MGMT_SETTING_SIZE },
6518 { set_privacy, MGMT_SET_PRIVACY_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006519 { load_irks, MGMT_LOAD_IRKS_SIZE,
6520 HCI_MGMT_VAR_LEN },
Marcel Holtmann7aea86162015-03-14 19:28:02 -07006521 { get_conn_info, MGMT_GET_CONN_INFO_SIZE },
6522 { get_clock_info, MGMT_GET_CLOCK_INFO_SIZE },
6523 { add_device, MGMT_ADD_DEVICE_SIZE },
6524 { remove_device, MGMT_REMOVE_DEVICE_SIZE },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006525 { load_conn_param, MGMT_LOAD_CONN_PARAM_SIZE,
6526 HCI_MGMT_VAR_LEN },
6527 { read_unconf_index_list, MGMT_READ_UNCONF_INDEX_LIST_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006528 HCI_MGMT_NO_HDEV |
6529 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006530 { read_config_info, MGMT_READ_CONFIG_INFO_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006531 HCI_MGMT_UNCONFIGURED |
6532 HCI_MGMT_UNTRUSTED },
Johan Hedbergb9a245fb2015-03-06 21:08:52 +02006533 { set_external_config, MGMT_SET_EXTERNAL_CONFIG_SIZE,
6534 HCI_MGMT_UNCONFIGURED },
6535 { set_public_address, MGMT_SET_PUBLIC_ADDRESS_SIZE,
6536 HCI_MGMT_UNCONFIGURED },
6537 { start_service_discovery, MGMT_START_SERVICE_DISCOVERY_SIZE,
6538 HCI_MGMT_VAR_LEN },
Marcel Holtmann4f0f1552015-03-14 22:43:19 -07006539 { read_local_oob_ext_data, MGMT_READ_LOCAL_OOB_EXT_DATA_SIZE },
Marcel Holtmann96f14742015-03-14 19:27:57 -07006540 { read_ext_index_list, MGMT_READ_EXT_INDEX_LIST_SIZE,
Marcel Holtmannc91041d2015-03-14 19:28:01 -07006541 HCI_MGMT_NO_HDEV |
6542 HCI_MGMT_UNTRUSTED },
Marcel Holtmannd3d53052015-03-14 20:53:25 -07006543 { read_adv_features, MGMT_READ_ADV_FEATURES_SIZE },
Arman Uguray24b4f382015-03-23 15:57:12 -07006544 { add_advertising, MGMT_ADD_ADVERTISING_SIZE,
6545 HCI_MGMT_VAR_LEN },
Arman Ugurayda9293352015-03-23 15:57:13 -07006546 { remove_advertising, MGMT_REMOVE_ADVERTISING_SIZE },
Marcel Holtmann40b25fe2015-11-19 16:16:43 +01006547 { get_adv_size_info, MGMT_GET_ADV_SIZE_INFO_SIZE },
Johan Hedberg78b781c2016-01-05 13:19:32 +02006548 { start_limited_discovery, MGMT_START_DISCOVERY_SIZE },
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02006549 { read_ext_controller_info,MGMT_READ_EXT_INFO_SIZE,
6550 HCI_MGMT_UNTRUSTED },
MichaƂ Narajowskic4960ec2016-09-18 12:50:03 +02006551 { set_appearance, MGMT_SET_APPEARANCE_SIZE },
Johan Hedberg0f4e68c2012-02-28 17:18:30 +02006552};
6553
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07006554void mgmt_index_added(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02006555{
Marcel Holtmannced85542015-03-14 19:27:56 -07006556 struct mgmt_ev_ext_index ev;
Andrei Emeltchenkobb4b2a92012-07-19 17:03:40 +03006557
Marcel Holtmann0602a8a2014-07-02 21:30:54 +02006558 if (test_bit(HCI_QUIRK_RAW_DEVICE, &hdev->quirks))
6559 return;
6560
Marcel Holtmannf9207332015-03-14 19:27:55 -07006561 switch (hdev->dev_type) {
Marcel Holtmannca8bee52016-07-05 14:30:14 +02006562 case HCI_PRIMARY:
Marcel Holtmannf9207332015-03-14 19:27:55 -07006563 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) {
6564 mgmt_index_event(MGMT_EV_UNCONF_INDEX_ADDED, hdev,
6565 NULL, 0, HCI_MGMT_UNCONF_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006566 ev.type = 0x01;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006567 } else {
6568 mgmt_index_event(MGMT_EV_INDEX_ADDED, hdev, NULL, 0,
6569 HCI_MGMT_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006570 ev.type = 0x00;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006571 }
6572 break;
Marcel Holtmannced85542015-03-14 19:27:56 -07006573 case HCI_AMP:
6574 ev.type = 0x02;
6575 break;
6576 default:
6577 return;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006578 }
Marcel Holtmannced85542015-03-14 19:27:56 -07006579
6580 ev.bus = hdev->bus;
6581
6582 mgmt_index_event(MGMT_EV_EXT_INDEX_ADDED, hdev, &ev, sizeof(ev),
6583 HCI_MGMT_EXT_INDEX_EVENTS);
Johan Hedbergc71e97b2010-12-13 21:07:07 +02006584}
6585
Marcel Holtmannbf6b56d2013-10-06 23:55:45 -07006586void mgmt_index_removed(struct hci_dev *hdev)
Johan Hedbergc71e97b2010-12-13 21:07:07 +02006587{
Marcel Holtmannced85542015-03-14 19:27:56 -07006588 struct mgmt_ev_ext_index ev;
Johan Hedberg5f159032012-03-02 03:13:19 +02006589 u8 status = MGMT_STATUS_INVALID_INDEX;
Johan Hedbergb24752f2011-11-03 14:40:33 +02006590
Marcel Holtmann0602a8a2014-07-02 21:30:54 +02006591 if (test_bit(HCI_QUIRK_RAW_DEVICE, &hdev->quirks))
6592 return;
6593
Marcel Holtmannf9207332015-03-14 19:27:55 -07006594 switch (hdev->dev_type) {
Marcel Holtmannca8bee52016-07-05 14:30:14 +02006595 case HCI_PRIMARY:
Marcel Holtmannf9207332015-03-14 19:27:55 -07006596 mgmt_pending_foreach(0, hdev, cmd_complete_rsp, &status);
Johan Hedbergb24752f2011-11-03 14:40:33 +02006597
Marcel Holtmannf9207332015-03-14 19:27:55 -07006598 if (hci_dev_test_flag(hdev, HCI_UNCONFIGURED)) {
6599 mgmt_index_event(MGMT_EV_UNCONF_INDEX_REMOVED, hdev,
6600 NULL, 0, HCI_MGMT_UNCONF_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006601 ev.type = 0x01;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006602 } else {
6603 mgmt_index_event(MGMT_EV_INDEX_REMOVED, hdev, NULL, 0,
6604 HCI_MGMT_INDEX_EVENTS);
Marcel Holtmannced85542015-03-14 19:27:56 -07006605 ev.type = 0x00;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006606 }
6607 break;
Marcel Holtmannced85542015-03-14 19:27:56 -07006608 case HCI_AMP:
6609 ev.type = 0x02;
6610 break;
6611 default:
6612 return;
Marcel Holtmannf9207332015-03-14 19:27:55 -07006613 }
Marcel Holtmannced85542015-03-14 19:27:56 -07006614
6615 ev.bus = hdev->bus;
6616
6617 mgmt_index_event(MGMT_EV_EXT_INDEX_REMOVED, hdev, &ev, sizeof(ev),
6618 HCI_MGMT_EXT_INDEX_EVENTS);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02006619}
6620
Andre Guedes6046dc32014-02-26 20:21:51 -03006621/* This function requires the caller holds hdev->lock */
Johan Hedbergaf02dd42015-11-11 08:11:21 +02006622static void restart_le_actions(struct hci_dev *hdev)
Andre Guedes6046dc32014-02-26 20:21:51 -03006623{
6624 struct hci_conn_params *p;
6625
6626 list_for_each_entry(p, &hdev->le_conn_params, list) {
Johan Hedbergd7347f32014-07-04 12:37:23 +03006627 /* Needed for AUTO_OFF case where might not "really"
6628 * have been powered off.
6629 */
6630 list_del_init(&p->action);
6631
6632 switch (p->auto_connect) {
Marcel Holtmann4b9e7e72014-07-23 21:55:23 +02006633 case HCI_AUTO_CONN_DIRECT:
Johan Hedbergd7347f32014-07-04 12:37:23 +03006634 case HCI_AUTO_CONN_ALWAYS:
6635 list_add(&p->action, &hdev->pend_le_conns);
6636 break;
6637 case HCI_AUTO_CONN_REPORT:
6638 list_add(&p->action, &hdev->pend_le_reports);
6639 break;
6640 default:
6641 break;
Marcel Holtmannc83ed192014-07-01 19:28:24 +02006642 }
Andre Guedes6046dc32014-02-26 20:21:51 -03006643 }
6644}
6645
Johan Hedberg2ff13892015-11-25 16:15:44 +02006646void mgmt_power_on(struct hci_dev *hdev, int err)
Johan Hedberg229ab392013-03-15 17:06:53 -05006647{
6648 struct cmd_lookup match = { NULL, hdev };
6649
Johan Hedberg2ff13892015-11-25 16:15:44 +02006650 BT_DBG("err %d", err);
Johan Hedberg229ab392013-03-15 17:06:53 -05006651
Johan Hedberg2ff13892015-11-25 16:15:44 +02006652 hci_dev_lock(hdev);
6653
6654 if (!err) {
Johan Hedbergaf02dd42015-11-11 08:11:21 +02006655 restart_le_actions(hdev);
6656 hci_update_background_scan(hdev);
Marcel Holtmann162a3ba2015-01-14 15:43:11 -08006657 }
6658
Johan Hedberg229ab392013-03-15 17:06:53 -05006659 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
6660
6661 new_settings(hdev, match.sk);
6662
Johan Hedberg229ab392013-03-15 17:06:53 -05006663 if (match.sk)
6664 sock_put(match.sk);
Johan Hedberg2ff13892015-11-25 16:15:44 +02006665
6666 hci_dev_unlock(hdev);
Johan Hedberg229ab392013-03-15 17:06:53 -05006667}
6668
Johan Hedberg2ff13892015-11-25 16:15:44 +02006669void __mgmt_power_off(struct hci_dev *hdev)
Johan Hedberg5add6af2010-12-16 10:00:37 +02006670{
Johan Hedberg76a7f3a2012-02-17 00:34:40 +02006671 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg98459042014-12-12 11:15:21 +02006672 u8 status, zero_cod[] = { 0, 0, 0 };
Johan Hedbergb24752f2011-11-03 14:40:33 +02006673
Johan Hedberg229ab392013-03-15 17:06:53 -05006674 mgmt_pending_foreach(MGMT_OP_SET_POWERED, hdev, settings_rsp, &match);
Johan Hedberg98459042014-12-12 11:15:21 +02006675
6676 /* If the power off is because of hdev unregistration let
6677 * use the appropriate INVALID_INDEX status. Otherwise use
6678 * NOT_POWERED. We cover both scenarios here since later in
6679 * mgmt_index_removed() any hci_conn callbacks will have already
6680 * been triggered, potentially causing misleading DISCONNECTED
6681 * status responses.
6682 */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07006683 if (hci_dev_test_flag(hdev, HCI_UNREGISTER))
Johan Hedberg98459042014-12-12 11:15:21 +02006684 status = MGMT_STATUS_INVALID_INDEX;
6685 else
6686 status = MGMT_STATUS_NOT_POWERED;
6687
6688 mgmt_pending_foreach(0, hdev, cmd_complete_rsp, &status);
Johan Hedberg229ab392013-03-15 17:06:53 -05006689
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02006690 if (memcmp(hdev->dev_class, zero_cod, sizeof(zero_cod)) != 0) {
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02006691 mgmt_limited_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev,
6692 zero_cod, sizeof(zero_cod),
6693 HCI_MGMT_DEV_CLASS_EVENTS, NULL);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02006694 ext_info_changed(hdev, NULL);
6695 }
Johan Hedberg229ab392013-03-15 17:06:53 -05006696
Johan Hedberg2ff13892015-11-25 16:15:44 +02006697 new_settings(hdev, match.sk);
Johan Hedbergeec8d2b2010-12-16 10:17:38 +02006698
6699 if (match.sk)
6700 sock_put(match.sk);
Johan Hedberg5add6af2010-12-16 10:00:37 +02006701}
Johan Hedberg73f22f62010-12-29 16:00:25 +02006702
Marcel Holtmann3eec7052013-10-06 23:55:46 -07006703void mgmt_set_powered_failed(struct hci_dev *hdev, int err)
Johan Hedberg96570ff2013-05-29 09:51:29 +03006704{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006705 struct mgmt_pending_cmd *cmd;
Johan Hedberg96570ff2013-05-29 09:51:29 +03006706 u8 status;
6707
Johan Hedberg333ae952015-03-17 13:48:47 +02006708 cmd = pending_find(MGMT_OP_SET_POWERED, hdev);
Johan Hedberg96570ff2013-05-29 09:51:29 +03006709 if (!cmd)
Marcel Holtmann3eec7052013-10-06 23:55:46 -07006710 return;
Johan Hedberg96570ff2013-05-29 09:51:29 +03006711
6712 if (err == -ERFKILL)
6713 status = MGMT_STATUS_RFKILLED;
6714 else
6715 status = MGMT_STATUS_FAILED;
6716
Johan Hedberga69e8372015-03-06 21:08:53 +02006717 mgmt_cmd_status(cmd->sk, hdev->id, MGMT_OP_SET_POWERED, status);
Johan Hedberg96570ff2013-05-29 09:51:29 +03006718
6719 mgmt_pending_remove(cmd);
Johan Hedberg96570ff2013-05-29 09:51:29 +03006720}
6721
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07006722void mgmt_new_link_key(struct hci_dev *hdev, struct link_key *key,
6723 bool persistent)
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006724{
Johan Hedberg86742e12011-11-07 23:13:38 +02006725 struct mgmt_ev_new_link_key ev;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006726
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006727 memset(&ev, 0, sizeof(ev));
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006728
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006729 ev.store_hint = persistent;
Johan Hedbergd753fdc2012-02-17 14:06:34 +02006730 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03006731 ev.key.addr.type = BDADDR_BREDR;
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006732 ev.key.type = key->type;
Andrei Emeltchenko9b3b4462012-05-23 11:31:20 +03006733 memcpy(ev.key.val, key->val, HCI_LINK_KEY_SIZE);
Vinicius Costa Gomesa492cd52011-08-25 20:02:29 -03006734 ev.key.pin_len = key->pin_len;
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006735
Marcel Holtmanndc4a5ee2013-10-15 10:15:57 -07006736 mgmt_event(MGMT_EV_NEW_LINK_KEY, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg55ed8ca12011-01-17 14:41:05 +02006737}
Johan Hedbergf7520542011-01-20 12:34:39 +02006738
Johan Hedbergd7b25452014-05-23 13:19:53 +03006739static u8 mgmt_ltk_type(struct smp_ltk *ltk)
6740{
Johan Hedberg23fb8de2014-05-23 13:15:37 +03006741 switch (ltk->type) {
6742 case SMP_LTK:
6743 case SMP_LTK_SLAVE:
6744 if (ltk->authenticated)
6745 return MGMT_LTK_AUTHENTICATED;
6746 return MGMT_LTK_UNAUTHENTICATED;
6747 case SMP_LTK_P256:
6748 if (ltk->authenticated)
6749 return MGMT_LTK_P256_AUTH;
6750 return MGMT_LTK_P256_UNAUTH;
6751 case SMP_LTK_P256_DEBUG:
6752 return MGMT_LTK_P256_DEBUG;
6753 }
Johan Hedbergd7b25452014-05-23 13:19:53 +03006754
6755 return MGMT_LTK_UNAUTHENTICATED;
6756}
6757
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006758void mgmt_new_ltk(struct hci_dev *hdev, struct smp_ltk *key, bool persistent)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006759{
6760 struct mgmt_ev_new_long_term_key ev;
6761
6762 memset(&ev, 0, sizeof(ev));
6763
Marcel Holtmann5192d302014-02-19 17:11:58 -08006764 /* Devices using resolvable or non-resolvable random addresses
Florian Grandelf72186d2015-05-26 03:31:09 +02006765 * without providing an identity resolving key don't require
Marcel Holtmann5192d302014-02-19 17:11:58 -08006766 * to store long term keys. Their addresses will change the
6767 * next time around.
6768 *
6769 * Only when a remote device provides an identity address
6770 * make sure the long term key is stored. If the remote
6771 * identity is known, the long term keys are internally
6772 * mapped to the identity address. So allow static random
6773 * and public addresses here.
6774 */
Johan Hedbergba74b662014-02-19 14:57:45 +02006775 if (key->bdaddr_type == ADDR_LE_DEV_RANDOM &&
6776 (key->bdaddr.b[5] & 0xc0) != 0xc0)
6777 ev.store_hint = 0x00;
6778 else
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006779 ev.store_hint = persistent;
Johan Hedbergba74b662014-02-19 14:57:45 +02006780
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006781 bacpy(&ev.key.addr.bdaddr, &key->bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03006782 ev.key.addr.type = link_to_bdaddr(LE_LINK, key->bdaddr_type);
Johan Hedbergd7b25452014-05-23 13:19:53 +03006783 ev.key.type = mgmt_ltk_type(key);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006784 ev.key.enc_size = key->enc_size;
6785 ev.key.ediv = key->ediv;
Marcel Holtmannfe39c7b2014-02-27 16:00:28 -08006786 ev.key.rand = key->rand;
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006787
Johan Hedberg2ceba532014-06-16 19:25:16 +03006788 if (key->type == SMP_LTK)
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006789 ev.key.master = 1;
6790
Johan Hedberg1fc62c52015-06-10 11:11:20 +03006791 /* Make sure we copy only the significant bytes based on the
6792 * encryption key size, and set the rest of the value to zeroes.
6793 */
Jakub Pawlowskicb922052015-08-05 23:16:29 +02006794 memcpy(ev.key.val, key->val, key->enc_size);
Johan Hedberg1fc62c52015-06-10 11:11:20 +03006795 memset(ev.key.val + key->enc_size, 0,
6796 sizeof(ev.key.val) - key->enc_size);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006797
Marcel Holtmann083368f2013-10-15 14:26:29 -07006798 mgmt_event(MGMT_EV_NEW_LONG_TERM_KEY, hdev, &ev, sizeof(ev), NULL);
Vinicius Costa Gomes346af672012-02-02 21:08:02 -03006799}
6800
Johan Hedbergcad20c22015-10-12 13:36:19 +02006801void mgmt_new_irk(struct hci_dev *hdev, struct smp_irk *irk, bool persistent)
Johan Hedberg95fbac82014-02-19 15:18:31 +02006802{
6803 struct mgmt_ev_new_irk ev;
6804
6805 memset(&ev, 0, sizeof(ev));
6806
Johan Hedbergcad20c22015-10-12 13:36:19 +02006807 ev.store_hint = persistent;
Marcel Holtmannbab6d1e2014-02-19 11:51:54 -08006808
Johan Hedberg95fbac82014-02-19 15:18:31 +02006809 bacpy(&ev.rpa, &irk->rpa);
6810 bacpy(&ev.irk.addr.bdaddr, &irk->bdaddr);
6811 ev.irk.addr.type = link_to_bdaddr(LE_LINK, irk->addr_type);
6812 memcpy(ev.irk.val, irk->val, sizeof(irk->val));
6813
6814 mgmt_event(MGMT_EV_NEW_IRK, hdev, &ev, sizeof(ev), NULL);
6815}
6816
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006817void mgmt_new_csrk(struct hci_dev *hdev, struct smp_csrk *csrk,
6818 bool persistent)
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006819{
6820 struct mgmt_ev_new_csrk ev;
6821
6822 memset(&ev, 0, sizeof(ev));
6823
6824 /* Devices using resolvable or non-resolvable random addresses
Florian Grandelf72186d2015-05-26 03:31:09 +02006825 * without providing an identity resolving key don't require
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006826 * to store signature resolving keys. Their addresses will change
6827 * the next time around.
6828 *
6829 * Only when a remote device provides an identity address
6830 * make sure the signature resolving key is stored. So allow
6831 * static random and public addresses here.
6832 */
6833 if (csrk->bdaddr_type == ADDR_LE_DEV_RANDOM &&
6834 (csrk->bdaddr.b[5] & 0xc0) != 0xc0)
6835 ev.store_hint = 0x00;
6836 else
Marcel Holtmann53ac6ab2014-03-09 23:38:42 -07006837 ev.store_hint = persistent;
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006838
6839 bacpy(&ev.key.addr.bdaddr, &csrk->bdaddr);
6840 ev.key.addr.type = link_to_bdaddr(LE_LINK, csrk->bdaddr_type);
Johan Hedberg4cd39282015-02-27 10:11:13 +02006841 ev.key.type = csrk->type;
Marcel Holtmann7ee4ea32014-03-09 12:19:17 -07006842 memcpy(ev.key.val, csrk->val, sizeof(csrk->val));
6843
6844 mgmt_event(MGMT_EV_NEW_CSRK, hdev, &ev, sizeof(ev), NULL);
6845}
6846
Andre Guedesffb5a8272014-07-01 18:10:11 -03006847void mgmt_new_conn_param(struct hci_dev *hdev, bdaddr_t *bdaddr,
Johan Hedbergf4869e22014-07-02 17:37:32 +03006848 u8 bdaddr_type, u8 store_hint, u16 min_interval,
6849 u16 max_interval, u16 latency, u16 timeout)
Andre Guedesffb5a8272014-07-01 18:10:11 -03006850{
6851 struct mgmt_ev_new_conn_param ev;
6852
Johan Hedbergc103aea2014-07-02 17:37:34 +03006853 if (!hci_is_identity_address(bdaddr, bdaddr_type))
6854 return;
6855
Andre Guedesffb5a8272014-07-01 18:10:11 -03006856 memset(&ev, 0, sizeof(ev));
6857 bacpy(&ev.addr.bdaddr, bdaddr);
6858 ev.addr.type = link_to_bdaddr(LE_LINK, bdaddr_type);
Johan Hedbergf4869e22014-07-02 17:37:32 +03006859 ev.store_hint = store_hint;
Andre Guedesffb5a8272014-07-01 18:10:11 -03006860 ev.min_interval = cpu_to_le16(min_interval);
6861 ev.max_interval = cpu_to_le16(max_interval);
6862 ev.latency = cpu_to_le16(latency);
6863 ev.timeout = cpu_to_le16(timeout);
6864
6865 mgmt_event(MGMT_EV_NEW_CONN_PARAM, hdev, &ev, sizeof(ev), NULL);
6866}
6867
Alfonso Acosta48ec92f2014-10-07 08:44:10 +00006868void mgmt_device_connected(struct hci_dev *hdev, struct hci_conn *conn,
6869 u32 flags, u8 *name, u8 name_len)
Johan Hedbergf7520542011-01-20 12:34:39 +02006870{
Johan Hedbergb644ba32012-01-17 21:48:47 +02006871 char buf[512];
6872 struct mgmt_ev_device_connected *ev = (void *) buf;
6873 u16 eir_len = 0;
Johan Hedbergf7520542011-01-20 12:34:39 +02006874
Alfonso Acosta48ec92f2014-10-07 08:44:10 +00006875 bacpy(&ev->addr.bdaddr, &conn->dst);
6876 ev->addr.type = link_to_bdaddr(conn->type, conn->dst_type);
Johan Hedbergf7520542011-01-20 12:34:39 +02006877
Johan Hedbergc95f0ba2012-02-23 22:54:38 +02006878 ev->flags = __cpu_to_le32(flags);
Johan Hedberg08c79b62012-02-23 22:31:51 +02006879
Alfonso Acostafd45ada2014-10-07 08:44:11 +00006880 /* We must ensure that the EIR Data fields are ordered and
6881 * unique. Keep it simple for now and avoid the problem by not
6882 * adding any BR/EDR data to the LE adv.
6883 */
6884 if (conn->le_adv_data_len > 0) {
6885 memcpy(&ev->eir[eir_len],
6886 conn->le_adv_data, conn->le_adv_data_len);
6887 eir_len = conn->le_adv_data_len;
6888 } else {
6889 if (name_len > 0)
6890 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE,
6891 name, name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02006892
Alfonso Acostaddbea5c2014-10-07 08:44:12 +00006893 if (memcmp(conn->dev_class, "\0\0\0", 3) != 0)
Alfonso Acostafd45ada2014-10-07 08:44:11 +00006894 eir_len = eir_append_data(ev->eir, eir_len,
6895 EIR_CLASS_OF_DEV,
6896 conn->dev_class, 3);
6897 }
Johan Hedbergb644ba32012-01-17 21:48:47 +02006898
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02006899 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02006900
Marcel Holtmannecd90ae2013-10-06 23:55:49 -07006901 mgmt_event(MGMT_EV_DEVICE_CONNECTED, hdev, buf,
6902 sizeof(*ev) + eir_len, NULL);
Johan Hedbergf7520542011-01-20 12:34:39 +02006903}
6904
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006905static void disconnect_rsp(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedberg8962ee72011-01-20 12:40:27 +02006906{
Johan Hedberg8962ee72011-01-20 12:40:27 +02006907 struct sock **sk = data;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006908
Johan Hedbergf5818c22014-12-05 13:36:02 +02006909 cmd->cmd_complete(cmd, 0);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006910
6911 *sk = cmd->sk;
6912 sock_hold(*sk);
6913
Johan Hedberga664b5b2011-02-19 12:06:02 -03006914 mgmt_pending_remove(cmd);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006915}
6916
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006917static void unpair_device_rsp(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedberga8a1d192011-11-10 15:54:38 +02006918{
Johan Hedbergb1078ad2012-02-09 17:21:16 +02006919 struct hci_dev *hdev = data;
Johan Hedberg124f6e32012-02-09 13:50:12 +02006920 struct mgmt_cp_unpair_device *cp = cmd->param;
Johan Hedberga8a1d192011-11-10 15:54:38 +02006921
Johan Hedbergb1078ad2012-02-09 17:21:16 +02006922 device_unpaired(hdev, &cp->addr.bdaddr, cp->addr.type, cmd->sk);
6923
Johan Hedbergd8b7b1e2014-12-05 13:36:05 +02006924 cmd->cmd_complete(cmd, 0);
Johan Hedberga8a1d192011-11-10 15:54:38 +02006925 mgmt_pending_remove(cmd);
6926}
6927
Johan Hedberg84c61d92014-08-01 11:13:30 +03006928bool mgmt_powering_down(struct hci_dev *hdev)
6929{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006930 struct mgmt_pending_cmd *cmd;
Johan Hedberg84c61d92014-08-01 11:13:30 +03006931 struct mgmt_mode *cp;
6932
Johan Hedberg333ae952015-03-17 13:48:47 +02006933 cmd = pending_find(MGMT_OP_SET_POWERED, hdev);
Johan Hedberg84c61d92014-08-01 11:13:30 +03006934 if (!cmd)
6935 return false;
6936
6937 cp = cmd->param;
6938 if (!cp->val)
6939 return true;
6940
6941 return false;
6942}
6943
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07006944void mgmt_device_disconnected(struct hci_dev *hdev, bdaddr_t *bdaddr,
Johan Hedberg12d4a3b2014-02-24 14:52:18 +02006945 u8 link_type, u8 addr_type, u8 reason,
6946 bool mgmt_connected)
Johan Hedbergf7520542011-01-20 12:34:39 +02006947{
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02006948 struct mgmt_ev_device_disconnected ev;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006949 struct sock *sk = NULL;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006950
Johan Hedberg84c61d92014-08-01 11:13:30 +03006951 /* The connection is still in hci_conn_hash so test for 1
6952 * instead of 0 to know if this is the last one.
6953 */
6954 if (mgmt_powering_down(hdev) && hci_conn_count(hdev) == 1) {
6955 cancel_delayed_work(&hdev->power_off);
6956 queue_work(hdev->req_workqueue, &hdev->power_off.work);
Johan Hedberg8b064a32014-02-24 14:52:22 +02006957 }
6958
Johan Hedberg12d4a3b2014-02-24 14:52:18 +02006959 if (!mgmt_connected)
6960 return;
6961
Andre Guedes57eb7762013-10-30 19:01:41 -03006962 if (link_type != ACL_LINK && link_type != LE_LINK)
6963 return;
6964
Johan Hedberg744cf192011-11-08 20:40:14 +02006965 mgmt_pending_foreach(MGMT_OP_DISCONNECT, hdev, disconnect_rsp, &sk);
Johan Hedbergf7520542011-01-20 12:34:39 +02006966
Mikel Astizf0d6a0e2012-08-09 09:52:30 +02006967 bacpy(&ev.addr.bdaddr, bdaddr);
6968 ev.addr.type = link_to_bdaddr(link_type, addr_type);
6969 ev.reason = reason;
Johan Hedbergf7520542011-01-20 12:34:39 +02006970
Marcel Holtmann9b80ec52013-10-06 23:55:50 -07006971 mgmt_event(MGMT_EV_DEVICE_DISCONNECTED, hdev, &ev, sizeof(ev), sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006972
6973 if (sk)
Szymon Jancd97dcb62012-03-16 16:02:56 +01006974 sock_put(sk);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006975
Johan Hedberg124f6e32012-02-09 13:50:12 +02006976 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03006977 hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006978}
6979
Marcel Holtmann78929242013-10-06 23:55:47 -07006980void mgmt_disconnect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr,
6981 u8 link_type, u8 addr_type, u8 status)
Johan Hedberg8962ee72011-01-20 12:40:27 +02006982{
Andre Guedes3655bba2013-10-30 19:01:40 -03006983 u8 bdaddr_type = link_to_bdaddr(link_type, addr_type);
6984 struct mgmt_cp_disconnect *cp;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02006985 struct mgmt_pending_cmd *cmd;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006986
Jefferson Delfes36a75f12012-09-18 13:36:54 -04006987 mgmt_pending_foreach(MGMT_OP_UNPAIR_DEVICE, hdev, unpair_device_rsp,
6988 hdev);
6989
Johan Hedberg333ae952015-03-17 13:48:47 +02006990 cmd = pending_find(MGMT_OP_DISCONNECT, hdev);
Johan Hedberg8962ee72011-01-20 12:40:27 +02006991 if (!cmd)
Marcel Holtmann78929242013-10-06 23:55:47 -07006992 return;
Johan Hedberg8962ee72011-01-20 12:40:27 +02006993
Andre Guedes3655bba2013-10-30 19:01:40 -03006994 cp = cmd->param;
6995
6996 if (bacmp(bdaddr, &cp->addr.bdaddr))
6997 return;
6998
6999 if (cp->addr.type != bdaddr_type)
7000 return;
7001
Johan Hedbergf5818c22014-12-05 13:36:02 +02007002 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03007003 mgmt_pending_remove(cmd);
Johan Hedbergf7520542011-01-20 12:34:39 +02007004}
Johan Hedberg17d5c042011-01-22 06:09:08 +02007005
Marcel Holtmann445608d2013-10-06 23:55:48 -07007006void mgmt_connect_failed(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
7007 u8 addr_type, u8 status)
Johan Hedberg17d5c042011-01-22 06:09:08 +02007008{
7009 struct mgmt_ev_connect_failed ev;
Johan Hedbergc9910d02014-02-27 14:35:12 +02007010
Johan Hedberg84c61d92014-08-01 11:13:30 +03007011 /* The connection is still in hci_conn_hash so test for 1
7012 * instead of 0 to know if this is the last one.
7013 */
7014 if (mgmt_powering_down(hdev) && hci_conn_count(hdev) == 1) {
7015 cancel_delayed_work(&hdev->power_off);
7016 queue_work(hdev->req_workqueue, &hdev->power_off.work);
Johan Hedbergc9910d02014-02-27 14:35:12 +02007017 }
Johan Hedberg17d5c042011-01-22 06:09:08 +02007018
Johan Hedberg4c659c32011-11-07 23:13:39 +02007019 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007020 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergca69b792011-11-11 18:10:00 +02007021 ev.status = mgmt_status(status);
Johan Hedberg17d5c042011-01-22 06:09:08 +02007022
Marcel Holtmann445608d2013-10-06 23:55:48 -07007023 mgmt_event(MGMT_EV_CONNECT_FAILED, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg17d5c042011-01-22 06:09:08 +02007024}
Johan Hedberg980e1a52011-01-22 06:10:07 +02007025
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07007026void mgmt_pin_code_request(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 secure)
Johan Hedberg980e1a52011-01-22 06:10:07 +02007027{
7028 struct mgmt_ev_pin_code_request ev;
7029
Johan Hedbergd8457692012-02-17 14:24:57 +02007030 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes591f47f2012-04-24 21:02:49 -03007031 ev.addr.type = BDADDR_BREDR;
Waldemar Rymarkiewicza770bb52011-04-28 12:07:59 +02007032 ev.secure = secure;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007033
Marcel Holtmannce0e4a02013-10-15 14:26:20 -07007034 mgmt_event(MGMT_EV_PIN_CODE_REQUEST, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007035}
7036
Marcel Holtmanne669cf82013-10-15 14:26:21 -07007037void mgmt_pin_code_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
7038 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02007039{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007040 struct mgmt_pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007041
Johan Hedberg333ae952015-03-17 13:48:47 +02007042 cmd = pending_find(MGMT_OP_PIN_CODE_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007043 if (!cmd)
Marcel Holtmanne669cf82013-10-15 14:26:21 -07007044 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007045
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007046 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03007047 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007048}
7049
Marcel Holtmann3eb38522013-10-15 14:26:22 -07007050void mgmt_pin_code_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
7051 u8 status)
Johan Hedberg980e1a52011-01-22 06:10:07 +02007052{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007053 struct mgmt_pending_cmd *cmd;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007054
Johan Hedberg333ae952015-03-17 13:48:47 +02007055 cmd = pending_find(MGMT_OP_PIN_CODE_NEG_REPLY, hdev);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007056 if (!cmd)
Marcel Holtmann3eb38522013-10-15 14:26:22 -07007057 return;
Johan Hedberg980e1a52011-01-22 06:10:07 +02007058
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007059 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03007060 mgmt_pending_remove(cmd);
Johan Hedberg980e1a52011-01-22 06:10:07 +02007061}
Johan Hedberga5c29682011-02-19 12:05:57 -03007062
Johan Hedberg744cf192011-11-08 20:40:14 +02007063int mgmt_user_confirm_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Johan Hedberg39adbff2014-03-20 08:18:14 +02007064 u8 link_type, u8 addr_type, u32 value,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007065 u8 confirm_hint)
Johan Hedberga5c29682011-02-19 12:05:57 -03007066{
7067 struct mgmt_ev_user_confirm_request ev;
7068
Johan Hedberg744cf192011-11-08 20:40:14 +02007069 BT_DBG("%s", hdev->name);
Johan Hedberga5c29682011-02-19 12:05:57 -03007070
Johan Hedberg272d90d2012-02-09 15:26:12 +02007071 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007072 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberg55bc1a32011-04-28 11:28:56 -07007073 ev.confirm_hint = confirm_hint;
Johan Hedberg39adbff2014-03-20 08:18:14 +02007074 ev.value = cpu_to_le32(value);
Johan Hedberga5c29682011-02-19 12:05:57 -03007075
Johan Hedberg744cf192011-11-08 20:40:14 +02007076 return mgmt_event(MGMT_EV_USER_CONFIRM_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007077 NULL);
Johan Hedberga5c29682011-02-19 12:05:57 -03007078}
7079
Johan Hedberg272d90d2012-02-09 15:26:12 +02007080int mgmt_user_passkey_request(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03007081 u8 link_type, u8 addr_type)
Brian Gix604086b2011-11-23 08:28:33 -08007082{
7083 struct mgmt_ev_user_passkey_request ev;
7084
7085 BT_DBG("%s", hdev->name);
7086
Johan Hedberg272d90d2012-02-09 15:26:12 +02007087 bacpy(&ev.addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007088 ev.addr.type = link_to_bdaddr(link_type, addr_type);
Brian Gix604086b2011-11-23 08:28:33 -08007089
7090 return mgmt_event(MGMT_EV_USER_PASSKEY_REQUEST, hdev, &ev, sizeof(ev),
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007091 NULL);
Brian Gix604086b2011-11-23 08:28:33 -08007092}
7093
Brian Gix0df4c182011-11-16 13:53:13 -08007094static int user_pairing_resp_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo Padovan8ce8e2b2012-05-17 00:36:20 -03007095 u8 link_type, u8 addr_type, u8 status,
7096 u8 opcode)
Johan Hedberga5c29682011-02-19 12:05:57 -03007097{
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007098 struct mgmt_pending_cmd *cmd;
Johan Hedberga5c29682011-02-19 12:05:57 -03007099
Johan Hedberg333ae952015-03-17 13:48:47 +02007100 cmd = pending_find(opcode, hdev);
Johan Hedberga5c29682011-02-19 12:05:57 -03007101 if (!cmd)
7102 return -ENOENT;
7103
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007104 cmd->cmd_complete(cmd, mgmt_status(status));
Johan Hedberga664b5b2011-02-19 12:06:02 -03007105 mgmt_pending_remove(cmd);
Johan Hedberga5c29682011-02-19 12:05:57 -03007106
Johan Hedberg7776d1d2014-12-05 13:36:03 +02007107 return 0;
Johan Hedberga5c29682011-02-19 12:05:57 -03007108}
7109
Johan Hedberg744cf192011-11-08 20:40:14 +02007110int mgmt_user_confirm_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007111 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03007112{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007113 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007114 status, MGMT_OP_USER_CONFIRM_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03007115}
7116
Johan Hedberg272d90d2012-02-09 15:26:12 +02007117int mgmt_user_confirm_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007118 u8 link_type, u8 addr_type, u8 status)
Johan Hedberga5c29682011-02-19 12:05:57 -03007119{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007120 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03007121 status,
7122 MGMT_OP_USER_CONFIRM_NEG_REPLY);
Johan Hedberga5c29682011-02-19 12:05:57 -03007123}
Johan Hedberg2a611692011-02-19 12:06:00 -03007124
Brian Gix604086b2011-11-23 08:28:33 -08007125int mgmt_user_passkey_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007126 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08007127{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007128 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007129 status, MGMT_OP_USER_PASSKEY_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08007130}
7131
Johan Hedberg272d90d2012-02-09 15:26:12 +02007132int mgmt_user_passkey_neg_reply_complete(struct hci_dev *hdev, bdaddr_t *bdaddr,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007133 u8 link_type, u8 addr_type, u8 status)
Brian Gix604086b2011-11-23 08:28:33 -08007134{
Johan Hedberg272d90d2012-02-09 15:26:12 +02007135 return user_pairing_resp_complete(hdev, bdaddr, link_type, addr_type,
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -03007136 status,
7137 MGMT_OP_USER_PASSKEY_NEG_REPLY);
Brian Gix604086b2011-11-23 08:28:33 -08007138}
7139
Johan Hedberg92a25252012-09-06 18:39:26 +03007140int mgmt_user_passkey_notify(struct hci_dev *hdev, bdaddr_t *bdaddr,
7141 u8 link_type, u8 addr_type, u32 passkey,
7142 u8 entered)
7143{
7144 struct mgmt_ev_passkey_notify ev;
7145
7146 BT_DBG("%s", hdev->name);
7147
7148 bacpy(&ev.addr.bdaddr, bdaddr);
7149 ev.addr.type = link_to_bdaddr(link_type, addr_type);
7150 ev.passkey = __cpu_to_le32(passkey);
7151 ev.entered = entered;
7152
7153 return mgmt_event(MGMT_EV_PASSKEY_NOTIFY, hdev, &ev, sizeof(ev), NULL);
7154}
7155
Johan Hedberge1e930f2014-09-08 17:09:49 -07007156void mgmt_auth_failed(struct hci_conn *conn, u8 hci_status)
Johan Hedberg2a611692011-02-19 12:06:00 -03007157{
7158 struct mgmt_ev_auth_failed ev;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007159 struct mgmt_pending_cmd *cmd;
Johan Hedberge1e930f2014-09-08 17:09:49 -07007160 u8 status = mgmt_status(hci_status);
Johan Hedberg2a611692011-02-19 12:06:00 -03007161
Johan Hedberge1e930f2014-09-08 17:09:49 -07007162 bacpy(&ev.addr.bdaddr, &conn->dst);
7163 ev.addr.type = link_to_bdaddr(conn->type, conn->dst_type);
7164 ev.status = status;
Johan Hedberg2a611692011-02-19 12:06:00 -03007165
Johan Hedberge1e930f2014-09-08 17:09:49 -07007166 cmd = find_pairing(conn);
7167
7168 mgmt_event(MGMT_EV_AUTH_FAILED, conn->hdev, &ev, sizeof(ev),
7169 cmd ? cmd->sk : NULL);
7170
Johan Hedberga511b352014-12-11 21:45:45 +02007171 if (cmd) {
7172 cmd->cmd_complete(cmd, status);
7173 mgmt_pending_remove(cmd);
7174 }
Johan Hedberg2a611692011-02-19 12:06:00 -03007175}
Johan Hedbergb312b1612011-03-16 14:29:37 +02007176
Marcel Holtmann464996a2013-10-15 14:26:24 -07007177void mgmt_auth_enable_complete(struct hci_dev *hdev, u8 status)
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007178{
7179 struct cmd_lookup match = { NULL, hdev };
Marcel Holtmann464996a2013-10-15 14:26:24 -07007180 bool changed;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007181
7182 if (status) {
7183 u8 mgmt_err = mgmt_status(status);
7184 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007185 cmd_status_rsp, &mgmt_err);
Marcel Holtmann464996a2013-10-15 14:26:24 -07007186 return;
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007187 }
7188
Marcel Holtmann464996a2013-10-15 14:26:24 -07007189 if (test_bit(HCI_AUTH, &hdev->flags))
Marcel Holtmann238be782015-03-13 02:11:06 -07007190 changed = !hci_dev_test_and_set_flag(hdev, HCI_LINK_SECURITY);
Marcel Holtmann464996a2013-10-15 14:26:24 -07007191 else
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007192 changed = hci_dev_test_and_clear_flag(hdev, HCI_LINK_SECURITY);
Johan Hedberg47990ea2012-02-22 11:58:37 +02007193
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007194 mgmt_pending_foreach(MGMT_OP_SET_LINK_SECURITY, hdev, settings_rsp,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007195 &match);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007196
Johan Hedberg47990ea2012-02-22 11:58:37 +02007197 if (changed)
Marcel Holtmann464996a2013-10-15 14:26:24 -07007198 new_settings(hdev, match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007199
7200 if (match.sk)
7201 sock_put(match.sk);
Johan Hedberg33ef95e2012-02-16 23:56:27 +02007202}
7203
Johan Hedberg890ea892013-03-15 17:06:52 -05007204static void clear_eir(struct hci_request *req)
Johan Hedbergcacaf522012-02-21 00:52:42 +02007205{
Johan Hedberg890ea892013-03-15 17:06:52 -05007206 struct hci_dev *hdev = req->hdev;
Johan Hedbergcacaf522012-02-21 00:52:42 +02007207 struct hci_cp_write_eir cp;
7208
Johan Hedberg976eb202012-10-24 21:12:01 +03007209 if (!lmp_ext_inq_capable(hdev))
Johan Hedberg890ea892013-03-15 17:06:52 -05007210 return;
Johan Hedbergcacaf522012-02-21 00:52:42 +02007211
Johan Hedbergc80da272012-02-22 15:38:48 +02007212 memset(hdev->eir, 0, sizeof(hdev->eir));
7213
Johan Hedbergcacaf522012-02-21 00:52:42 +02007214 memset(&cp, 0, sizeof(cp));
7215
Johan Hedberg890ea892013-03-15 17:06:52 -05007216 hci_req_add(req, HCI_OP_WRITE_EIR, sizeof(cp), &cp);
Johan Hedbergcacaf522012-02-21 00:52:42 +02007217}
7218
Marcel Holtmann3e248562013-10-15 14:26:25 -07007219void mgmt_ssp_enable_complete(struct hci_dev *hdev, u8 enable, u8 status)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007220{
7221 struct cmd_lookup match = { NULL, hdev };
Johan Hedberg890ea892013-03-15 17:06:52 -05007222 struct hci_request req;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007223 bool changed = false;
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007224
7225 if (status) {
7226 u8 mgmt_err = mgmt_status(status);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007227
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007228 if (enable && hci_dev_test_and_clear_flag(hdev,
7229 HCI_SSP_ENABLED)) {
Marcel Holtmanna358dc12015-03-13 02:11:02 -07007230 hci_dev_clear_flag(hdev, HCI_HS_ENABLED);
Marcel Holtmann3e248562013-10-15 14:26:25 -07007231 new_settings(hdev, NULL);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07007232 }
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007233
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007234 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, cmd_status_rsp,
7235 &mgmt_err);
Marcel Holtmann3e248562013-10-15 14:26:25 -07007236 return;
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007237 }
7238
7239 if (enable) {
Marcel Holtmann238be782015-03-13 02:11:06 -07007240 changed = !hci_dev_test_and_set_flag(hdev, HCI_SSP_ENABLED);
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007241 } else {
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007242 changed = hci_dev_test_and_clear_flag(hdev, HCI_SSP_ENABLED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07007243 if (!changed)
Marcel Holtmanna69d8922015-03-13 02:11:05 -07007244 changed = hci_dev_test_and_clear_flag(hdev,
7245 HCI_HS_ENABLED);
Marcel Holtmann9ecb3e22013-10-10 03:08:11 -07007246 else
Marcel Holtmanna358dc12015-03-13 02:11:02 -07007247 hci_dev_clear_flag(hdev, HCI_HS_ENABLED);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007248 }
7249
7250 mgmt_pending_foreach(MGMT_OP_SET_SSP, hdev, settings_rsp, &match);
7251
Johan Hedbergc0ecddc2012-02-22 12:38:31 +02007252 if (changed)
Marcel Holtmann3e248562013-10-15 14:26:25 -07007253 new_settings(hdev, match.sk);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007254
Johan Hedberg5fc6ebb2012-02-22 15:10:59 +02007255 if (match.sk)
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007256 sock_put(match.sk);
7257
Johan Hedberg890ea892013-03-15 17:06:52 -05007258 hci_req_init(&req, hdev);
7259
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07007260 if (hci_dev_test_flag(hdev, HCI_SSP_ENABLED)) {
7261 if (hci_dev_test_flag(hdev, HCI_USE_DEBUG_KEYS))
Johan Hedberg37699722014-06-24 14:00:27 +03007262 hci_req_add(&req, HCI_OP_WRITE_SSP_DEBUG_MODE,
7263 sizeof(enable), &enable);
Johan Hedbergb1a89172015-11-25 16:15:42 +02007264 __hci_req_update_eir(&req);
Johan Hedberg37699722014-06-24 14:00:27 +03007265 } else {
Johan Hedberg890ea892013-03-15 17:06:52 -05007266 clear_eir(&req);
Johan Hedberg37699722014-06-24 14:00:27 +03007267 }
Johan Hedberg890ea892013-03-15 17:06:52 -05007268
7269 hci_req_run(&req, NULL);
Johan Hedberged2c4ee2012-02-17 00:56:28 +02007270}
7271
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007272static void sk_lookup(struct mgmt_pending_cmd *cmd, void *data)
Johan Hedberg90e70452012-02-23 23:09:40 +02007273{
7274 struct cmd_lookup *match = data;
7275
Johan Hedberg90e70452012-02-23 23:09:40 +02007276 if (match->sk == NULL) {
7277 match->sk = cmd->sk;
7278 sock_hold(match->sk);
7279 }
Johan Hedberg90e70452012-02-23 23:09:40 +02007280}
7281
Marcel Holtmann4e1b0242013-10-15 14:26:26 -07007282void mgmt_set_class_of_dev_complete(struct hci_dev *hdev, u8 *dev_class,
7283 u8 status)
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01007284{
Johan Hedberg90e70452012-02-23 23:09:40 +02007285 struct cmd_lookup match = { NULL, hdev, mgmt_status(status) };
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01007286
Johan Hedberg92da6092013-03-15 17:06:55 -05007287 mgmt_pending_foreach(MGMT_OP_SET_DEV_CLASS, hdev, sk_lookup, &match);
7288 mgmt_pending_foreach(MGMT_OP_ADD_UUID, hdev, sk_lookup, &match);
7289 mgmt_pending_foreach(MGMT_OP_REMOVE_UUID, hdev, sk_lookup, &match);
Johan Hedberg90e70452012-02-23 23:09:40 +02007290
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02007291 if (!status) {
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02007292 mgmt_limited_event(MGMT_EV_CLASS_OF_DEV_CHANGED, hdev, dev_class,
7293 3, HCI_MGMT_DEV_CLASS_EVENTS, NULL);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02007294 ext_info_changed(hdev, NULL);
7295 }
Johan Hedberg90e70452012-02-23 23:09:40 +02007296
7297 if (match.sk)
7298 sock_put(match.sk);
Marcel Holtmann7f9a9032012-02-22 18:38:01 +01007299}
7300
Marcel Holtmann7667da32013-10-15 14:26:27 -07007301void mgmt_set_local_name_complete(struct hci_dev *hdev, u8 *name, u8 status)
Johan Hedbergb312b1612011-03-16 14:29:37 +02007302{
Johan Hedbergb312b1612011-03-16 14:29:37 +02007303 struct mgmt_cp_set_local_name ev;
Johan Hedberg3b0602c2015-03-06 21:08:55 +02007304 struct mgmt_pending_cmd *cmd;
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02007305
Johan Hedberg13928972013-03-15 17:07:00 -05007306 if (status)
Marcel Holtmann7667da32013-10-15 14:26:27 -07007307 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02007308
7309 memset(&ev, 0, sizeof(ev));
7310 memcpy(ev.name, name, HCI_MAX_NAME_LENGTH);
Johan Hedberg28cc7bd2012-02-22 21:06:55 +02007311 memcpy(ev.short_name, hdev->short_name, HCI_MAX_SHORT_NAME_LENGTH);
Johan Hedbergb312b1612011-03-16 14:29:37 +02007312
Johan Hedberg333ae952015-03-17 13:48:47 +02007313 cmd = pending_find(MGMT_OP_SET_LOCAL_NAME, hdev);
Johan Hedberg13928972013-03-15 17:07:00 -05007314 if (!cmd) {
7315 memcpy(hdev->dev_name, name, sizeof(hdev->dev_name));
Johan Hedbergb312b1612011-03-16 14:29:37 +02007316
Johan Hedberg13928972013-03-15 17:07:00 -05007317 /* If this is a HCI command related to powering on the
7318 * HCI dev don't send any mgmt signals.
7319 */
Johan Hedberg333ae952015-03-17 13:48:47 +02007320 if (pending_find(MGMT_OP_SET_POWERED, hdev))
Marcel Holtmann7667da32013-10-15 14:26:27 -07007321 return;
Johan Hedbergb312b1612011-03-16 14:29:37 +02007322 }
7323
Marcel Holtmann5504c3a2016-08-29 06:19:46 +02007324 mgmt_limited_event(MGMT_EV_LOCAL_NAME_CHANGED, hdev, &ev, sizeof(ev),
7325 HCI_MGMT_LOCAL_NAME_EVENTS, cmd ? cmd->sk : NULL);
Marcel Holtmann321c6fe2016-09-01 16:46:23 +02007326 ext_info_changed(hdev, cmd ? cmd->sk : NULL);
Johan Hedbergb312b1612011-03-16 14:29:37 +02007327}
Szymon Jancc35938b2011-03-22 13:12:21 +01007328
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007329static inline bool has_uuid(u8 *uuid, u16 uuid_count, u8 (*uuids)[16])
7330{
7331 int i;
7332
7333 for (i = 0; i < uuid_count; i++) {
7334 if (!memcmp(uuid, uuids[i], 16))
7335 return true;
7336 }
7337
7338 return false;
7339}
7340
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007341static bool eir_has_uuids(u8 *eir, u16 eir_len, u16 uuid_count, u8 (*uuids)[16])
7342{
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007343 u16 parsed = 0;
7344
7345 while (parsed < eir_len) {
7346 u8 field_len = eir[0];
7347 u8 uuid[16];
7348 int i;
7349
7350 if (field_len == 0)
7351 break;
7352
7353 if (eir_len - parsed < field_len + 1)
7354 break;
7355
7356 switch (eir[1]) {
7357 case EIR_UUID16_ALL:
7358 case EIR_UUID16_SOME:
7359 for (i = 0; i + 3 <= field_len; i += 2) {
Johan Hedberg189f6ad2014-12-05 13:40:01 +02007360 memcpy(uuid, bluetooth_base_uuid, 16);
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007361 uuid[13] = eir[i + 3];
7362 uuid[12] = eir[i + 2];
7363 if (has_uuid(uuid, uuid_count, uuids))
7364 return true;
7365 }
7366 break;
7367 case EIR_UUID32_ALL:
7368 case EIR_UUID32_SOME:
7369 for (i = 0; i + 5 <= field_len; i += 4) {
Johan Hedberg189f6ad2014-12-05 13:40:01 +02007370 memcpy(uuid, bluetooth_base_uuid, 16);
Jakub Pawlowski799ce932014-12-05 10:55:58 +01007371 uuid[15] = eir[i + 5];
7372 uuid[14] = eir[i + 4];
7373 uuid[13] = eir[i + 3];
7374 uuid[12] = eir[i + 2];
7375 if (has_uuid(uuid, uuid_count, uuids))
7376 return true;
7377 }
7378 break;
7379 case EIR_UUID128_ALL:
7380 case EIR_UUID128_SOME:
7381 for (i = 0; i + 17 <= field_len; i += 16) {
7382 memcpy(uuid, eir + i + 2, 16);
7383 if (has_uuid(uuid, uuid_count, uuids))
7384 return true;
7385 }
7386 break;
7387 }
7388
7389 parsed += field_len + 1;
7390 eir += field_len + 1;
7391 }
7392
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007393 return false;
7394}
7395
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007396static void restart_le_scan(struct hci_dev *hdev)
7397{
7398 /* If controller is not scanning we are done. */
Marcel Holtmannd7a5a112015-03-13 02:11:00 -07007399 if (!hci_dev_test_flag(hdev, HCI_LE_SCAN))
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007400 return;
7401
7402 if (time_after(jiffies + DISCOV_LE_RESTART_DELAY,
7403 hdev->discovery.scan_start +
7404 hdev->discovery.scan_duration))
7405 return;
7406
Johan Hedberg7c1fbed2015-11-11 08:11:23 +02007407 queue_delayed_work(hdev->req_workqueue, &hdev->le_scan_restart,
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007408 DISCOV_LE_RESTART_DELAY);
7409}
7410
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007411static bool is_filter_match(struct hci_dev *hdev, s8 rssi, u8 *eir,
7412 u16 eir_len, u8 *scan_rsp, u8 scan_rsp_len)
7413{
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007414 /* If a RSSI threshold has been specified, and
7415 * HCI_QUIRK_STRICT_DUPLICATE_FILTER is not set, then all results with
7416 * a RSSI smaller than the RSSI threshold will be dropped. If the quirk
7417 * is set, let it through for further processing, as we might need to
7418 * restart the scan.
7419 *
7420 * For BR/EDR devices (pre 1.2) providing no RSSI during inquiry,
7421 * the results are also dropped.
7422 */
7423 if (hdev->discovery.rssi != HCI_RSSI_INVALID &&
7424 (rssi == HCI_RSSI_INVALID ||
7425 (rssi < hdev->discovery.rssi &&
7426 !test_bit(HCI_QUIRK_STRICT_DUPLICATE_FILTER, &hdev->quirks))))
7427 return false;
7428
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007429 if (hdev->discovery.uuid_count != 0) {
7430 /* If a list of UUIDs is provided in filter, results with no
7431 * matching UUID should be dropped.
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007432 */
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007433 if (!eir_has_uuids(eir, eir_len, hdev->discovery.uuid_count,
7434 hdev->discovery.uuids) &&
7435 !eir_has_uuids(scan_rsp, scan_rsp_len,
7436 hdev->discovery.uuid_count,
7437 hdev->discovery.uuids))
7438 return false;
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007439 }
7440
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007441 /* If duplicate filtering does not report RSSI changes, then restart
7442 * scanning to ensure updated result with updated RSSI values.
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007443 */
Jakub Pawlowski2976cde2015-03-04 16:24:25 -08007444 if (test_bit(HCI_QUIRK_STRICT_DUPLICATE_FILTER, &hdev->quirks)) {
7445 restart_le_scan(hdev);
7446
7447 /* Validate RSSI value against the RSSI threshold once more. */
7448 if (hdev->discovery.rssi != HCI_RSSI_INVALID &&
7449 rssi < hdev->discovery.rssi)
7450 return false;
7451 }
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007452
7453 return true;
7454}
7455
Marcel Holtmann901801b2013-10-06 23:55:51 -07007456void mgmt_device_found(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
Marcel Holtmannaf589252014-07-01 14:11:20 +02007457 u8 addr_type, u8 *dev_class, s8 rssi, u32 flags,
7458 u8 *eir, u16 eir_len, u8 *scan_rsp, u8 scan_rsp_len)
Johan Hedberge17acd42011-03-30 23:57:16 +03007459{
Johan Hedberge319d2e2012-01-15 19:51:59 +02007460 char buf[512];
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007461 struct mgmt_ev_device_found *ev = (void *)buf;
Johan Hedberg1dc06092012-01-15 21:01:23 +02007462 size_t ev_size;
Johan Hedberge17acd42011-03-30 23:57:16 +03007463
Johan Hedberg75ce2082014-07-02 22:42:01 +03007464 /* Don't send events for a non-kernel initiated discovery. With
7465 * LE one exception is if we have pend_le_reports > 0 in which
7466 * case we're doing passive scanning and want these events.
7467 */
7468 if (!hci_discovery_active(hdev)) {
7469 if (link_type == ACL_LINK)
7470 return;
Johan Hedberg66f84552014-07-04 12:37:18 +03007471 if (link_type == LE_LINK && list_empty(&hdev->pend_le_reports))
Johan Hedberg75ce2082014-07-02 22:42:01 +03007472 return;
7473 }
Andre Guedes12602d02013-04-30 15:29:40 -03007474
Jakub Pawlowski82f8b652015-03-04 16:24:26 -08007475 if (hdev->discovery.result_filtering) {
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007476 /* We are using service discovery */
7477 if (!is_filter_match(hdev, rssi, eir, eir_len, scan_rsp,
7478 scan_rsp_len))
7479 return;
7480 }
Marcel Holtmannbda157a2014-12-05 10:55:56 +01007481
Johan Hedberg78b781c2016-01-05 13:19:32 +02007482 if (hdev->discovery.limited) {
7483 /* Check for limited discoverable bit */
7484 if (dev_class) {
7485 if (!(dev_class[1] & 0x20))
7486 return;
7487 } else {
7488 u8 *flags = eir_get_data(eir, eir_len, EIR_FLAGS, NULL);
7489 if (!flags || !(flags[0] & LE_AD_LIMITED))
7490 return;
7491 }
7492 }
7493
Johan Hedberg5d2e9fa2014-03-25 10:30:47 +02007494 /* Make sure that the buffer is big enough. The 5 extra bytes
7495 * are for the potential CoD field.
7496 */
7497 if (sizeof(*ev) + eir_len + scan_rsp_len + 5 > sizeof(buf))
Marcel Holtmann901801b2013-10-06 23:55:51 -07007498 return;
Andre Guedes7d262f82012-01-10 18:20:49 -03007499
Johan Hedberg1dc06092012-01-15 21:01:23 +02007500 memset(buf, 0, sizeof(buf));
7501
Marcel Holtmannda25cf62014-12-05 13:03:35 +01007502 /* In case of device discovery with BR/EDR devices (pre 1.2), the
7503 * RSSI value was reported as 0 when not available. This behavior
7504 * is kept when using device discovery. This is required for full
7505 * backwards compatibility with the API.
7506 *
7507 * However when using service discovery, the value 127 will be
7508 * returned when the RSSI is not available.
7509 */
Szymon Janc91200e92015-01-22 16:57:05 +01007510 if (rssi == HCI_RSSI_INVALID && !hdev->discovery.report_invalid_rssi &&
7511 link_type == ACL_LINK)
Marcel Holtmannefb25132014-12-05 13:03:34 +01007512 rssi = 0;
7513
Johan Hedberg841c5642014-07-07 12:45:54 +03007514 bacpy(&ev->addr.bdaddr, bdaddr);
7515 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedberge319d2e2012-01-15 19:51:59 +02007516 ev->rssi = rssi;
Marcel Holtmannaf589252014-07-01 14:11:20 +02007517 ev->flags = cpu_to_le32(flags);
Johan Hedberge17acd42011-03-30 23:57:16 +03007518
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007519 if (eir_len > 0)
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007520 /* Copy EIR or advertising data into event */
Johan Hedberge319d2e2012-01-15 19:51:59 +02007521 memcpy(ev->eir, eir, eir_len);
Johan Hedberge17acd42011-03-30 23:57:16 +03007522
Johan Hedberg0d3b7f62016-01-05 13:19:31 +02007523 if (dev_class && !eir_get_data(ev->eir, eir_len, EIR_CLASS_OF_DEV,
7524 NULL))
Johan Hedberg1dc06092012-01-15 21:01:23 +02007525 eir_len = eir_append_data(ev->eir, eir_len, EIR_CLASS_OF_DEV,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007526 dev_class, 3);
Johan Hedberg1dc06092012-01-15 21:01:23 +02007527
Jakub Pawlowski48f86b72015-03-04 16:24:24 -08007528 if (scan_rsp_len > 0)
Marcel Holtmannb487b9c2014-12-05 10:55:57 +01007529 /* Append scan response data to event */
Johan Hedberg5d2e9fa2014-03-25 10:30:47 +02007530 memcpy(ev->eir + eir_len, scan_rsp, scan_rsp_len);
Jakub Pawlowski4b0e0ce2015-02-01 23:07:55 -08007531
Johan Hedberg5d2e9fa2014-03-25 10:30:47 +02007532 ev->eir_len = cpu_to_le16(eir_len + scan_rsp_len);
7533 ev_size = sizeof(*ev) + eir_len + scan_rsp_len;
Andre Guedesf8523592011-09-09 18:56:26 -03007534
Marcel Holtmann901801b2013-10-06 23:55:51 -07007535 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, ev_size, NULL);
Johan Hedberge17acd42011-03-30 23:57:16 +03007536}
Johan Hedberga88a9652011-03-30 13:18:12 +03007537
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07007538void mgmt_remote_name(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 link_type,
7539 u8 addr_type, s8 rssi, u8 *name, u8 name_len)
Johan Hedberga88a9652011-03-30 13:18:12 +03007540{
Johan Hedbergb644ba32012-01-17 21:48:47 +02007541 struct mgmt_ev_device_found *ev;
7542 char buf[sizeof(*ev) + HCI_MAX_NAME_LENGTH + 2];
7543 u16 eir_len;
Johan Hedberga88a9652011-03-30 13:18:12 +03007544
Johan Hedbergb644ba32012-01-17 21:48:47 +02007545 ev = (struct mgmt_ev_device_found *) buf;
Johan Hedberga88a9652011-03-30 13:18:12 +03007546
Johan Hedbergb644ba32012-01-17 21:48:47 +02007547 memset(buf, 0, sizeof(buf));
Johan Hedberga88a9652011-03-30 13:18:12 +03007548
Johan Hedbergb644ba32012-01-17 21:48:47 +02007549 bacpy(&ev->addr.bdaddr, bdaddr);
Andre Guedes57c14772012-04-24 21:02:50 -03007550 ev->addr.type = link_to_bdaddr(link_type, addr_type);
Johan Hedbergb644ba32012-01-17 21:48:47 +02007551 ev->rssi = rssi;
7552
7553 eir_len = eir_append_data(ev->eir, 0, EIR_NAME_COMPLETE, name,
Gustavo F. Padovan04124682012-03-08 01:25:00 -03007554 name_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02007555
Marcel Holtmanneb55ef02012-03-14 18:08:46 +02007556 ev->eir_len = cpu_to_le16(eir_len);
Johan Hedbergb644ba32012-01-17 21:48:47 +02007557
Marcel Holtmann9cf12ae2013-10-06 23:55:52 -07007558 mgmt_event(MGMT_EV_DEVICE_FOUND, hdev, ev, sizeof(*ev) + eir_len, NULL);
Johan Hedberga88a9652011-03-30 13:18:12 +03007559}
Johan Hedberg314b2382011-04-27 10:29:57 -04007560
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07007561void mgmt_discovering(struct hci_dev *hdev, u8 discovering)
Johan Hedberg314b2382011-04-27 10:29:57 -04007562{
Johan Hedbergf963e8e2012-02-20 23:30:44 +02007563 struct mgmt_ev_discovering ev;
Johan Hedberg164a6e72011-11-01 17:06:44 +02007564
Andre Guedes343fb142011-11-22 17:14:19 -03007565 BT_DBG("%s discovering %u", hdev->name, discovering);
7566
Johan Hedbergf963e8e2012-02-20 23:30:44 +02007567 memset(&ev, 0, sizeof(ev));
7568 ev.type = hdev->discovery.type;
7569 ev.discovering = discovering;
7570
Marcel Holtmann2f1e0632013-10-06 23:55:53 -07007571 mgmt_event(MGMT_EV_DISCOVERING, hdev, &ev, sizeof(ev), NULL);
Johan Hedberg314b2382011-04-27 10:29:57 -04007572}
Antti Julku5e762442011-08-25 16:48:02 +03007573
Johan Hedberg6d785aa32015-03-06 21:08:51 +02007574static struct hci_mgmt_chan chan = {
7575 .channel = HCI_CHANNEL_CONTROL,
7576 .handler_count = ARRAY_SIZE(mgmt_handlers),
7577 .handlers = mgmt_handlers,
Johan Hedberg88b94ce2015-03-17 13:48:49 +02007578 .hdev_init = mgmt_init_hdev,
Johan Hedberg6d785aa32015-03-06 21:08:51 +02007579};
7580
7581int mgmt_init(void)
7582{
7583 return hci_mgmt_chan_register(&chan);
7584}
7585
7586void mgmt_exit(void)
7587{
7588 hci_mgmt_chan_unregister(&chan);
7589}