blob: efe51ccdc615af5e7439f931363313bc51a0a6e7 [file] [log] [blame]
Anderson Brigliaeb492e02011-06-09 18:50:40 -03001/*
2 BlueZ - Bluetooth protocol stack for Linux
3 Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies).
4
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License version 2 as
7 published by the Free Software Foundation;
8
9 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
10 OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
11 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.
12 IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY
13 CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES
14 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17
18 ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,
19 COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS
20 SOFTWARE IS DISCLAIMED.
21*/
22
Gustavo Padovan8c520a52012-05-23 04:04:22 -030023#include <linux/crypto.h>
24#include <linux/scatterlist.h>
25#include <crypto/b128ops.h>
26
Anderson Brigliaeb492e02011-06-09 18:50:40 -030027#include <net/bluetooth/bluetooth.h>
28#include <net/bluetooth/hci_core.h>
29#include <net/bluetooth/l2cap.h>
Brian Gix2b64d152011-12-21 16:12:12 -080030#include <net/bluetooth/mgmt.h>
Marcel Holtmannac4b7232013-10-10 14:54:16 -070031
32#include "smp.h"
Anderson Brigliad22ef0b2011-06-09 18:50:44 -030033
Marcel Holtmann17b02e62012-03-01 14:32:37 -080034#define SMP_TIMEOUT msecs_to_jiffies(30000)
Vinicius Costa Gomes5d3de7d2011-06-14 13:37:41 -030035
Johan Hedberg065a13e2012-10-11 16:26:06 +020036#define AUTH_REQ_MASK 0x07
37
Anderson Brigliad22ef0b2011-06-09 18:50:44 -030038static inline void swap128(u8 src[16], u8 dst[16])
39{
40 int i;
41 for (i = 0; i < 16; i++)
42 dst[15 - i] = src[i];
43}
44
45static inline void swap56(u8 src[7], u8 dst[7])
46{
47 int i;
48 for (i = 0; i < 7; i++)
49 dst[6 - i] = src[i];
50}
51
52static int smp_e(struct crypto_blkcipher *tfm, const u8 *k, u8 *r)
53{
54 struct blkcipher_desc desc;
55 struct scatterlist sg;
Johan Hedberg201a5922013-12-02 10:49:04 +020056 int err;
Anderson Brigliad22ef0b2011-06-09 18:50:44 -030057
58 if (tfm == NULL) {
59 BT_ERR("tfm %p", tfm);
60 return -EINVAL;
61 }
62
63 desc.tfm = tfm;
64 desc.flags = 0;
65
66 err = crypto_blkcipher_setkey(tfm, k, 16);
67 if (err) {
68 BT_ERR("cipher setkey failed: %d", err);
69 return err;
70 }
71
72 sg_init_one(&sg, r, 16);
73
Anderson Brigliad22ef0b2011-06-09 18:50:44 -030074 err = crypto_blkcipher_encrypt(&desc, &sg, &sg, 16);
75 if (err)
76 BT_ERR("Encrypt data error %d", err);
77
78 return err;
79}
80
81static int smp_c1(struct crypto_blkcipher *tfm, u8 k[16], u8 r[16],
Marcel Holtmannf1560462013-10-13 05:43:25 -070082 u8 preq[7], u8 pres[7], u8 _iat, bdaddr_t *ia,
83 u8 _rat, bdaddr_t *ra, u8 res[16])
Anderson Brigliad22ef0b2011-06-09 18:50:44 -030084{
85 u8 p1[16], p2[16];
86 int err;
87
88 memset(p1, 0, 16);
89
90 /* p1 = pres || preq || _rat || _iat */
91 swap56(pres, p1);
92 swap56(preq, p1 + 7);
93 p1[14] = _rat;
94 p1[15] = _iat;
95
96 memset(p2, 0, 16);
97
98 /* p2 = padding || ia || ra */
99 baswap((bdaddr_t *) (p2 + 4), ia);
100 baswap((bdaddr_t *) (p2 + 10), ra);
101
102 /* res = r XOR p1 */
103 u128_xor((u128 *) res, (u128 *) r, (u128 *) p1);
104
105 /* res = e(k, res) */
106 err = smp_e(tfm, k, res);
107 if (err) {
108 BT_ERR("Encrypt data error");
109 return err;
110 }
111
112 /* res = res XOR p2 */
113 u128_xor((u128 *) res, (u128 *) res, (u128 *) p2);
114
115 /* res = e(k, res) */
116 err = smp_e(tfm, k, res);
117 if (err)
118 BT_ERR("Encrypt data error");
119
120 return err;
121}
122
Marcel Holtmannf1560462013-10-13 05:43:25 -0700123static int smp_s1(struct crypto_blkcipher *tfm, u8 k[16], u8 r1[16],
124 u8 r2[16], u8 _r[16])
Anderson Brigliad22ef0b2011-06-09 18:50:44 -0300125{
126 int err;
127
128 /* Just least significant octets from r1 and r2 are considered */
129 memcpy(_r, r1 + 8, 8);
130 memcpy(_r + 8, r2 + 8, 8);
131
132 err = smp_e(tfm, k, _r);
133 if (err)
134 BT_ERR("Encrypt data error");
135
136 return err;
137}
138
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300139static struct sk_buff *smp_build_cmd(struct l2cap_conn *conn, u8 code,
Marcel Holtmannf1560462013-10-13 05:43:25 -0700140 u16 dlen, void *data)
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300141{
142 struct sk_buff *skb;
143 struct l2cap_hdr *lh;
144 int len;
145
146 len = L2CAP_HDR_SIZE + sizeof(code) + dlen;
147
148 if (len > conn->mtu)
149 return NULL;
150
151 skb = bt_skb_alloc(len, GFP_ATOMIC);
152 if (!skb)
153 return NULL;
154
155 lh = (struct l2cap_hdr *) skb_put(skb, L2CAP_HDR_SIZE);
156 lh->len = cpu_to_le16(sizeof(code) + dlen);
Syam Sidhardhand8aece22012-10-10 22:09:28 +0530157 lh->cid = __constant_cpu_to_le16(L2CAP_CID_SMP);
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300158
159 memcpy(skb_put(skb, sizeof(code)), &code, sizeof(code));
160
161 memcpy(skb_put(skb, dlen), data, dlen);
162
163 return skb;
164}
165
166static void smp_send_cmd(struct l2cap_conn *conn, u8 code, u16 len, void *data)
167{
168 struct sk_buff *skb = smp_build_cmd(conn, code, len, data);
169
170 BT_DBG("code 0x%2.2x", code);
171
172 if (!skb)
173 return;
174
Luiz Augusto von Dentz73d80de2011-11-02 15:52:01 +0200175 skb->priority = HCI_PRIO_MAX;
176 hci_send_acl(conn->hchan, skb, 0);
Vinicius Costa Gomese2dcd112011-08-19 21:06:50 -0300177
Gustavo F. Padovan6c9d42a2011-12-20 10:57:27 -0200178 cancel_delayed_work_sync(&conn->security_timer);
Marcel Holtmann17b02e62012-03-01 14:32:37 -0800179 schedule_delayed_work(&conn->security_timer, SMP_TIMEOUT);
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300180}
181
Brian Gix2b64d152011-12-21 16:12:12 -0800182static __u8 authreq_to_seclevel(__u8 authreq)
183{
184 if (authreq & SMP_AUTH_MITM)
185 return BT_SECURITY_HIGH;
186 else
187 return BT_SECURITY_MEDIUM;
188}
189
190static __u8 seclevel_to_authreq(__u8 sec_level)
191{
192 switch (sec_level) {
193 case BT_SECURITY_HIGH:
194 return SMP_AUTH_MITM | SMP_AUTH_BONDING;
195 case BT_SECURITY_MEDIUM:
196 return SMP_AUTH_BONDING;
197 default:
198 return SMP_AUTH_NONE;
199 }
200}
201
Vinicius Costa Gomesb8e66ea2011-06-09 18:50:52 -0300202static void build_pairing_cmd(struct l2cap_conn *conn,
Marcel Holtmannf1560462013-10-13 05:43:25 -0700203 struct smp_cmd_pairing *req,
204 struct smp_cmd_pairing *rsp, __u8 authreq)
Vinicius Costa Gomesb8e66ea2011-06-09 18:50:52 -0300205{
Brian Gix2b64d152011-12-21 16:12:12 -0800206 u8 dist_keys = 0;
Vinicius Costa Gomes54790f72011-07-07 18:59:38 -0300207
Johan Hedberga8b2d5c2012-01-08 23:11:15 +0200208 if (test_bit(HCI_PAIRABLE, &conn->hcon->hdev->dev_flags)) {
Vinicius Costa Gomesca10b5e2011-08-25 20:02:37 -0300209 dist_keys = SMP_DIST_ENC_KEY;
Vinicius Costa Gomes54790f72011-07-07 18:59:38 -0300210 authreq |= SMP_AUTH_BONDING;
Brian Gix2b64d152011-12-21 16:12:12 -0800211 } else {
212 authreq &= ~SMP_AUTH_BONDING;
Vinicius Costa Gomes54790f72011-07-07 18:59:38 -0300213 }
214
215 if (rsp == NULL) {
216 req->io_capability = conn->hcon->io_capability;
217 req->oob_flag = SMP_OOB_NOT_PRESENT;
218 req->max_key_size = SMP_MAX_ENC_KEY_SIZE;
Johan Hedberg0cf73b92014-01-29 13:55:59 -0800219 req->init_key_dist = dist_keys;
Vinicius Costa Gomes54790f72011-07-07 18:59:38 -0300220 req->resp_key_dist = dist_keys;
Johan Hedberg065a13e2012-10-11 16:26:06 +0200221 req->auth_req = (authreq & AUTH_REQ_MASK);
Vinicius Costa Gomes54790f72011-07-07 18:59:38 -0300222 return;
223 }
224
225 rsp->io_capability = conn->hcon->io_capability;
226 rsp->oob_flag = SMP_OOB_NOT_PRESENT;
227 rsp->max_key_size = SMP_MAX_ENC_KEY_SIZE;
Johan Hedberg0cf73b92014-01-29 13:55:59 -0800228 rsp->init_key_dist = req->init_key_dist & dist_keys;
Vinicius Costa Gomes54790f72011-07-07 18:59:38 -0300229 rsp->resp_key_dist = req->resp_key_dist & dist_keys;
Johan Hedberg065a13e2012-10-11 16:26:06 +0200230 rsp->auth_req = (authreq & AUTH_REQ_MASK);
Vinicius Costa Gomesb8e66ea2011-06-09 18:50:52 -0300231}
232
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300233static u8 check_enc_key_size(struct l2cap_conn *conn, __u8 max_key_size)
234{
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300235 struct smp_chan *smp = conn->smp_chan;
236
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300237 if ((max_key_size > SMP_MAX_ENC_KEY_SIZE) ||
Marcel Holtmannf1560462013-10-13 05:43:25 -0700238 (max_key_size < SMP_MIN_ENC_KEY_SIZE))
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300239 return SMP_ENC_KEY_SIZE;
240
Vinicius Costa Gomesf7aa6112012-01-30 19:29:12 -0300241 smp->enc_key_size = max_key_size;
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300242
243 return 0;
244}
245
Johan Hedberg84794e12013-11-06 11:24:57 +0200246static void smp_failure(struct l2cap_conn *conn, u8 reason)
Brian Gix4f957a72011-11-23 08:28:36 -0800247{
Johan Hedbergbab73cb2012-02-09 16:07:29 +0200248 struct hci_conn *hcon = conn->hcon;
249
Johan Hedberg84794e12013-11-06 11:24:57 +0200250 if (reason)
Brian Gix4f957a72011-11-23 08:28:36 -0800251 smp_send_cmd(conn, SMP_CMD_PAIRING_FAIL, sizeof(reason),
Marcel Holtmannf1560462013-10-13 05:43:25 -0700252 &reason);
Brian Gix4f957a72011-11-23 08:28:36 -0800253
Marcel Holtmannce39fb42013-10-13 02:23:39 -0700254 clear_bit(HCI_CONN_ENCRYPT_PEND, &hcon->flags);
255 mgmt_auth_failed(hcon->hdev, &hcon->dst, hcon->type, hcon->dst_type,
256 HCI_ERROR_AUTH_FAILURE);
Vinicius Costa Gomesf1c09c02012-02-01 18:27:56 -0300257
Andre Guedes61a0cfb2012-08-01 20:34:15 -0300258 cancel_delayed_work_sync(&conn->security_timer);
259
Marcel Holtmannce39fb42013-10-13 02:23:39 -0700260 if (test_and_clear_bit(HCI_CONN_LE_SMP_PEND, &hcon->flags))
Vinicius Costa Gomesf1c09c02012-02-01 18:27:56 -0300261 smp_chan_destroy(conn);
Brian Gix4f957a72011-11-23 08:28:36 -0800262}
263
Brian Gix2b64d152011-12-21 16:12:12 -0800264#define JUST_WORKS 0x00
265#define JUST_CFM 0x01
266#define REQ_PASSKEY 0x02
267#define CFM_PASSKEY 0x03
268#define REQ_OOB 0x04
269#define OVERLAP 0xFF
270
271static const u8 gen_method[5][5] = {
272 { JUST_WORKS, JUST_CFM, REQ_PASSKEY, JUST_WORKS, REQ_PASSKEY },
273 { JUST_WORKS, JUST_CFM, REQ_PASSKEY, JUST_WORKS, REQ_PASSKEY },
274 { CFM_PASSKEY, CFM_PASSKEY, REQ_PASSKEY, JUST_WORKS, CFM_PASSKEY },
275 { JUST_WORKS, JUST_CFM, JUST_WORKS, JUST_WORKS, JUST_CFM },
276 { CFM_PASSKEY, CFM_PASSKEY, REQ_PASSKEY, JUST_WORKS, OVERLAP },
277};
278
279static int tk_request(struct l2cap_conn *conn, u8 remote_oob, u8 auth,
280 u8 local_io, u8 remote_io)
281{
282 struct hci_conn *hcon = conn->hcon;
283 struct smp_chan *smp = conn->smp_chan;
284 u8 method;
285 u32 passkey = 0;
286 int ret = 0;
287
288 /* Initialize key for JUST WORKS */
289 memset(smp->tk, 0, sizeof(smp->tk));
290 clear_bit(SMP_FLAG_TK_VALID, &smp->smp_flags);
291
292 BT_DBG("tk_request: auth:%d lcl:%d rem:%d", auth, local_io, remote_io);
293
294 /* If neither side wants MITM, use JUST WORKS */
295 /* If either side has unknown io_caps, use JUST WORKS */
296 /* Otherwise, look up method from the table */
297 if (!(auth & SMP_AUTH_MITM) ||
Marcel Holtmannf1560462013-10-13 05:43:25 -0700298 local_io > SMP_IO_KEYBOARD_DISPLAY ||
299 remote_io > SMP_IO_KEYBOARD_DISPLAY)
Brian Gix2b64d152011-12-21 16:12:12 -0800300 method = JUST_WORKS;
301 else
Ido Yarivb3ff53f2012-03-05 20:07:08 +0200302 method = gen_method[remote_io][local_io];
Brian Gix2b64d152011-12-21 16:12:12 -0800303
304 /* If not bonding, don't ask user to confirm a Zero TK */
305 if (!(auth & SMP_AUTH_BONDING) && method == JUST_CFM)
306 method = JUST_WORKS;
307
308 /* If Just Works, Continue with Zero TK */
309 if (method == JUST_WORKS) {
310 set_bit(SMP_FLAG_TK_VALID, &smp->smp_flags);
311 return 0;
312 }
313
314 /* Not Just Works/Confirm results in MITM Authentication */
315 if (method != JUST_CFM)
316 set_bit(SMP_FLAG_MITM_AUTH, &smp->smp_flags);
317
318 /* If both devices have Keyoard-Display I/O, the master
319 * Confirms and the slave Enters the passkey.
320 */
321 if (method == OVERLAP) {
322 if (hcon->link_mode & HCI_LM_MASTER)
323 method = CFM_PASSKEY;
324 else
325 method = REQ_PASSKEY;
326 }
327
328 /* Generate random passkey. Not valid until confirmed. */
329 if (method == CFM_PASSKEY) {
330 u8 key[16];
331
332 memset(key, 0, sizeof(key));
333 get_random_bytes(&passkey, sizeof(passkey));
334 passkey %= 1000000;
335 put_unaligned_le32(passkey, key);
336 swap128(key, smp->tk);
337 BT_DBG("PassKey: %d", passkey);
338 }
339
340 hci_dev_lock(hcon->hdev);
341
342 if (method == REQ_PASSKEY)
Marcel Holtmannce39fb42013-10-13 02:23:39 -0700343 ret = mgmt_user_passkey_request(hcon->hdev, &hcon->dst,
Johan Hedberg272d90d2012-02-09 15:26:12 +0200344 hcon->type, hcon->dst_type);
Brian Gix2b64d152011-12-21 16:12:12 -0800345 else
Marcel Holtmannce39fb42013-10-13 02:23:39 -0700346 ret = mgmt_user_confirm_request(hcon->hdev, &hcon->dst,
Johan Hedberg272d90d2012-02-09 15:26:12 +0200347 hcon->type, hcon->dst_type,
Brian Gix2b64d152011-12-21 16:12:12 -0800348 cpu_to_le32(passkey), 0);
349
350 hci_dev_unlock(hcon->hdev);
351
352 return ret;
353}
354
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300355static void confirm_work(struct work_struct *work)
356{
357 struct smp_chan *smp = container_of(work, struct smp_chan, confirm);
358 struct l2cap_conn *conn = smp->conn;
359 struct crypto_blkcipher *tfm;
360 struct smp_cmd_pairing_confirm cp;
361 int ret;
362 u8 res[16], reason;
363
364 BT_DBG("conn %p", conn);
365
366 tfm = crypto_alloc_blkcipher("ecb(aes)", 0, CRYPTO_ALG_ASYNC);
367 if (IS_ERR(tfm)) {
368 reason = SMP_UNSPECIFIED;
369 goto error;
370 }
371
372 smp->tfm = tfm;
373
374 if (conn->hcon->out)
Marcel Holtmannc8462ca2013-10-13 05:24:02 -0700375 ret = smp_c1(tfm, smp->tk, smp->prnd, smp->preq, smp->prsp,
376 conn->hcon->src_type, &conn->hcon->src,
377 conn->hcon->dst_type, &conn->hcon->dst, res);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300378 else
379 ret = smp_c1(tfm, smp->tk, smp->prnd, smp->preq, smp->prsp,
Marcel Holtmannc8462ca2013-10-13 05:24:02 -0700380 conn->hcon->dst_type, &conn->hcon->dst,
381 conn->hcon->src_type, &conn->hcon->src, res);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300382 if (ret) {
383 reason = SMP_UNSPECIFIED;
384 goto error;
385 }
386
Brian Gix2b64d152011-12-21 16:12:12 -0800387 clear_bit(SMP_FLAG_CFM_PENDING, &smp->smp_flags);
388
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300389 swap128(res, cp.confirm_val);
390 smp_send_cmd(smp->conn, SMP_CMD_PAIRING_CONFIRM, sizeof(cp), &cp);
391
392 return;
393
394error:
Johan Hedberg84794e12013-11-06 11:24:57 +0200395 smp_failure(conn, reason);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300396}
397
398static void random_work(struct work_struct *work)
399{
400 struct smp_chan *smp = container_of(work, struct smp_chan, random);
401 struct l2cap_conn *conn = smp->conn;
402 struct hci_conn *hcon = conn->hcon;
403 struct crypto_blkcipher *tfm = smp->tfm;
404 u8 reason, confirm[16], res[16], key[16];
405 int ret;
406
407 if (IS_ERR_OR_NULL(tfm)) {
408 reason = SMP_UNSPECIFIED;
409 goto error;
410 }
411
412 BT_DBG("conn %p %s", conn, conn->hcon->out ? "master" : "slave");
413
414 if (hcon->out)
Marcel Holtmannc8462ca2013-10-13 05:24:02 -0700415 ret = smp_c1(tfm, smp->tk, smp->rrnd, smp->preq, smp->prsp,
416 hcon->src_type, &hcon->src,
417 hcon->dst_type, &hcon->dst, res);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300418 else
419 ret = smp_c1(tfm, smp->tk, smp->rrnd, smp->preq, smp->prsp,
Marcel Holtmannc8462ca2013-10-13 05:24:02 -0700420 hcon->dst_type, &hcon->dst,
421 hcon->src_type, &hcon->src, res);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300422 if (ret) {
423 reason = SMP_UNSPECIFIED;
424 goto error;
425 }
426
427 swap128(res, confirm);
428
429 if (memcmp(smp->pcnf, confirm, sizeof(smp->pcnf)) != 0) {
430 BT_ERR("Pairing failed (confirmation values mismatch)");
431 reason = SMP_CONFIRM_FAILED;
432 goto error;
433 }
434
435 if (hcon->out) {
436 u8 stk[16], rand[8];
437 __le16 ediv;
438
439 memset(rand, 0, sizeof(rand));
440 ediv = 0;
441
442 smp_s1(tfm, smp->tk, smp->rrnd, smp->prnd, key);
443 swap128(key, stk);
444
Vinicius Costa Gomesf7aa6112012-01-30 19:29:12 -0300445 memset(stk + smp->enc_key_size, 0,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300446 SMP_MAX_ENC_KEY_SIZE - smp->enc_key_size);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300447
Johan Hedberg51a8efd2012-01-16 06:10:31 +0200448 if (test_and_set_bit(HCI_CONN_ENCRYPT_PEND, &hcon->flags)) {
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300449 reason = SMP_UNSPECIFIED;
450 goto error;
451 }
452
453 hci_le_start_enc(hcon, ediv, rand, stk);
Vinicius Costa Gomesf7aa6112012-01-30 19:29:12 -0300454 hcon->enc_key_size = smp->enc_key_size;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300455 } else {
456 u8 stk[16], r[16], rand[8];
457 __le16 ediv;
458
459 memset(rand, 0, sizeof(rand));
460 ediv = 0;
461
462 swap128(smp->prnd, r);
463 smp_send_cmd(conn, SMP_CMD_PAIRING_RANDOM, sizeof(r), r);
464
465 smp_s1(tfm, smp->tk, smp->prnd, smp->rrnd, key);
466 swap128(key, stk);
467
Vinicius Costa Gomesf7aa6112012-01-30 19:29:12 -0300468 memset(stk + smp->enc_key_size, 0,
Marcel Holtmannf1560462013-10-13 05:43:25 -0700469 SMP_MAX_ENC_KEY_SIZE - smp->enc_key_size);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300470
Marcel Holtmannce39fb42013-10-13 02:23:39 -0700471 hci_add_ltk(hcon->hdev, &hcon->dst, hcon->dst_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300472 HCI_SMP_STK_SLAVE, 0, 0, stk, smp->enc_key_size,
473 ediv, rand);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300474 }
475
476 return;
477
478error:
Johan Hedberg84794e12013-11-06 11:24:57 +0200479 smp_failure(conn, reason);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300480}
481
482static struct smp_chan *smp_chan_create(struct l2cap_conn *conn)
483{
484 struct smp_chan *smp;
485
Marcel Holtmannf1560462013-10-13 05:43:25 -0700486 smp = kzalloc(sizeof(*smp), GFP_ATOMIC);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300487 if (!smp)
488 return NULL;
489
490 INIT_WORK(&smp->confirm, confirm_work);
491 INIT_WORK(&smp->random, random_work);
492
493 smp->conn = conn;
494 conn->smp_chan = smp;
Brian Gix2b64d152011-12-21 16:12:12 -0800495 conn->hcon->smp_conn = conn;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300496
497 hci_conn_hold(conn->hcon);
498
499 return smp;
500}
501
502void smp_chan_destroy(struct l2cap_conn *conn)
503{
Brian Gixc8eb9692011-11-23 08:28:35 -0800504 struct smp_chan *smp = conn->smp_chan;
505
Vinicius Costa Gomesf1c09c02012-02-01 18:27:56 -0300506 BUG_ON(!smp);
Brian Gixc8eb9692011-11-23 08:28:35 -0800507
508 if (smp->tfm)
509 crypto_free_blkcipher(smp->tfm);
510
511 kfree(smp);
512 conn->smp_chan = NULL;
Brian Gix2b64d152011-12-21 16:12:12 -0800513 conn->hcon->smp_conn = NULL;
David Herrmann76a68ba2013-04-06 20:28:37 +0200514 hci_conn_drop(conn->hcon);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300515}
516
Brian Gix2b64d152011-12-21 16:12:12 -0800517int smp_user_confirm_reply(struct hci_conn *hcon, u16 mgmt_op, __le32 passkey)
518{
519 struct l2cap_conn *conn = hcon->smp_conn;
520 struct smp_chan *smp;
521 u32 value;
522 u8 key[16];
523
524 BT_DBG("");
525
526 if (!conn)
527 return -ENOTCONN;
528
529 smp = conn->smp_chan;
530
531 switch (mgmt_op) {
532 case MGMT_OP_USER_PASSKEY_REPLY:
533 value = le32_to_cpu(passkey);
534 memset(key, 0, sizeof(key));
535 BT_DBG("PassKey: %d", value);
536 put_unaligned_le32(value, key);
537 swap128(key, smp->tk);
538 /* Fall Through */
539 case MGMT_OP_USER_CONFIRM_REPLY:
540 set_bit(SMP_FLAG_TK_VALID, &smp->smp_flags);
541 break;
542 case MGMT_OP_USER_PASSKEY_NEG_REPLY:
543 case MGMT_OP_USER_CONFIRM_NEG_REPLY:
Johan Hedberg84794e12013-11-06 11:24:57 +0200544 smp_failure(conn, SMP_PASSKEY_ENTRY_FAILED);
Brian Gix2b64d152011-12-21 16:12:12 -0800545 return 0;
546 default:
Johan Hedberg84794e12013-11-06 11:24:57 +0200547 smp_failure(conn, SMP_PASSKEY_ENTRY_FAILED);
Brian Gix2b64d152011-12-21 16:12:12 -0800548 return -EOPNOTSUPP;
549 }
550
551 /* If it is our turn to send Pairing Confirm, do so now */
552 if (test_bit(SMP_FLAG_CFM_PENDING, &smp->smp_flags))
553 queue_work(hcon->hdev->workqueue, &smp->confirm);
554
555 return 0;
556}
557
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300558static u8 smp_cmd_pairing_req(struct l2cap_conn *conn, struct sk_buff *skb)
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300559{
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300560 struct smp_cmd_pairing rsp, *req = (void *) skb->data;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300561 struct smp_chan *smp;
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300562 u8 key_size;
Brian Gix2b64d152011-12-21 16:12:12 -0800563 u8 auth = SMP_AUTH_NONE;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300564 int ret;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300565
566 BT_DBG("conn %p", conn);
567
Brian Gix2b64d152011-12-21 16:12:12 -0800568 if (conn->hcon->link_mode & HCI_LM_MASTER)
569 return SMP_CMD_NOTSUPP;
570
Johan Hedberg51a8efd2012-01-16 06:10:31 +0200571 if (!test_and_set_bit(HCI_CONN_LE_SMP_PEND, &conn->hcon->flags))
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300572 smp = smp_chan_create(conn);
Andrei Emeltchenkod08fd0e2012-07-19 17:03:43 +0300573 else
574 smp = conn->smp_chan;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300575
Andrei Emeltchenkod08fd0e2012-07-19 17:03:43 +0300576 if (!smp)
577 return SMP_UNSPECIFIED;
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -0300578
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300579 smp->preq[0] = SMP_CMD_PAIRING_REQ;
580 memcpy(&smp->preq[1], req, sizeof(*req));
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300581 skb_pull(skb, sizeof(*req));
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300582
Brian Gix2b64d152011-12-21 16:12:12 -0800583 /* We didn't start the pairing, so match remote */
584 if (req->auth_req & SMP_AUTH_BONDING)
585 auth = req->auth_req;
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300586
Ido Yarivfdde0a22012-03-05 20:09:38 +0200587 conn->hcon->pending_sec_level = authreq_to_seclevel(auth);
588
Brian Gix2b64d152011-12-21 16:12:12 -0800589 build_pairing_cmd(conn, req, &rsp, auth);
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300590
591 key_size = min(req->max_key_size, rsp.max_key_size);
592 if (check_enc_key_size(conn, key_size))
593 return SMP_ENC_KEY_SIZE;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300594
Johan Hedberge84a6b12013-12-02 10:49:03 +0200595 get_random_bytes(smp->prnd, sizeof(smp->prnd));
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300596
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300597 smp->prsp[0] = SMP_CMD_PAIRING_RSP;
598 memcpy(&smp->prsp[1], &rsp, sizeof(rsp));
Anderson Brigliaf01ead32011-06-09 18:50:45 -0300599
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300600 smp_send_cmd(conn, SMP_CMD_PAIRING_RSP, sizeof(rsp), &rsp);
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300601
Brian Gix2b64d152011-12-21 16:12:12 -0800602 /* Request setup of TK */
603 ret = tk_request(conn, 0, auth, rsp.io_capability, req->io_capability);
604 if (ret)
605 return SMP_UNSPECIFIED;
606
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300607 return 0;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300608}
609
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300610static u8 smp_cmd_pairing_rsp(struct l2cap_conn *conn, struct sk_buff *skb)
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300611{
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300612 struct smp_cmd_pairing *req, *rsp = (void *) skb->data;
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300613 struct smp_chan *smp = conn->smp_chan;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300614 struct hci_dev *hdev = conn->hcon->hdev;
Brian Gix2b64d152011-12-21 16:12:12 -0800615 u8 key_size, auth = SMP_AUTH_NONE;
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300616 int ret;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300617
618 BT_DBG("conn %p", conn);
619
Brian Gix2b64d152011-12-21 16:12:12 -0800620 if (!(conn->hcon->link_mode & HCI_LM_MASTER))
621 return SMP_CMD_NOTSUPP;
622
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300623 skb_pull(skb, sizeof(*rsp));
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300624
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300625 req = (void *) &smp->preq[1];
Vinicius Costa Gomes3158c502011-06-14 13:37:42 -0300626
627 key_size = min(req->max_key_size, rsp->max_key_size);
628 if (check_enc_key_size(conn, key_size))
629 return SMP_ENC_KEY_SIZE;
630
Johan Hedberge84a6b12013-12-02 10:49:03 +0200631 get_random_bytes(smp->prnd, sizeof(smp->prnd));
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300632
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300633 smp->prsp[0] = SMP_CMD_PAIRING_RSP;
634 memcpy(&smp->prsp[1], rsp, sizeof(*rsp));
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300635
Brian Gix2b64d152011-12-21 16:12:12 -0800636 if ((req->auth_req & SMP_AUTH_BONDING) &&
Marcel Holtmannf1560462013-10-13 05:43:25 -0700637 (rsp->auth_req & SMP_AUTH_BONDING))
Brian Gix2b64d152011-12-21 16:12:12 -0800638 auth = SMP_AUTH_BONDING;
639
640 auth |= (req->auth_req | rsp->auth_req) & SMP_AUTH_MITM;
641
Johan Hedberg476585e2012-06-06 18:54:15 +0800642 ret = tk_request(conn, 0, auth, req->io_capability, rsp->io_capability);
Brian Gix2b64d152011-12-21 16:12:12 -0800643 if (ret)
644 return SMP_UNSPECIFIED;
645
646 set_bit(SMP_FLAG_CFM_PENDING, &smp->smp_flags);
647
648 /* Can't compose response until we have been confirmed */
649 if (!test_bit(SMP_FLAG_TK_VALID, &smp->smp_flags))
650 return 0;
651
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300652 queue_work(hdev->workqueue, &smp->confirm);
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300653
654 return 0;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300655}
656
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300657static u8 smp_cmd_pairing_confirm(struct l2cap_conn *conn, struct sk_buff *skb)
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300658{
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300659 struct smp_chan *smp = conn->smp_chan;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300660 struct hci_dev *hdev = conn->hcon->hdev;
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300661
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300662 BT_DBG("conn %p %s", conn, conn->hcon->out ? "master" : "slave");
663
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300664 memcpy(smp->pcnf, skb->data, sizeof(smp->pcnf));
665 skb_pull(skb, sizeof(smp->pcnf));
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300666
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300667 if (conn->hcon->out) {
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300668 u8 random[16];
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300669
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300670 swap128(smp->prnd, random);
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300671 smp_send_cmd(conn, SMP_CMD_PAIRING_RANDOM, sizeof(random),
Marcel Holtmannf1560462013-10-13 05:43:25 -0700672 random);
Brian Gix2b64d152011-12-21 16:12:12 -0800673 } else if (test_bit(SMP_FLAG_TK_VALID, &smp->smp_flags)) {
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300674 queue_work(hdev->workqueue, &smp->confirm);
Brian Gix2b64d152011-12-21 16:12:12 -0800675 } else {
676 set_bit(SMP_FLAG_CFM_PENDING, &smp->smp_flags);
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300677 }
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300678
679 return 0;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300680}
681
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300682static u8 smp_cmd_pairing_random(struct l2cap_conn *conn, struct sk_buff *skb)
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300683{
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300684 struct smp_chan *smp = conn->smp_chan;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300685 struct hci_dev *hdev = conn->hcon->hdev;
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300686
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300687 BT_DBG("conn %p", conn);
Anderson Briglia7d24ddc2011-06-09 18:50:46 -0300688
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300689 swap128(skb->data, smp->rrnd);
690 skb_pull(skb, sizeof(smp->rrnd));
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300691
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300692 queue_work(hdev->workqueue, &smp->random);
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300693
694 return 0;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300695}
696
Johan Hedberg4dab7862012-06-07 14:58:37 +0800697static u8 smp_ltk_encrypt(struct l2cap_conn *conn, u8 sec_level)
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300698{
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -0300699 struct smp_ltk *key;
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300700 struct hci_conn *hcon = conn->hcon;
701
Johan Hedberg98a0b842014-01-30 19:40:00 -0800702 key = hci_find_ltk_by_addr(hcon->hdev, &hcon->dst, hcon->dst_type,
703 hcon->out);
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300704 if (!key)
705 return 0;
706
Johan Hedberg4dab7862012-06-07 14:58:37 +0800707 if (sec_level > BT_SECURITY_MEDIUM && !key->authenticated)
708 return 0;
709
Johan Hedberg51a8efd2012-01-16 06:10:31 +0200710 if (test_and_set_bit(HCI_CONN_ENCRYPT_PEND, &hcon->flags))
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300711 return 1;
712
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -0300713 hci_le_start_enc(hcon, key->ediv, key->rand, key->val);
714 hcon->enc_key_size = key->enc_size;
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300715
716 return 1;
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300717}
Marcel Holtmannf1560462013-10-13 05:43:25 -0700718
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300719static u8 smp_cmd_security_req(struct l2cap_conn *conn, struct sk_buff *skb)
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300720{
721 struct smp_cmd_security_req *rp = (void *) skb->data;
722 struct smp_cmd_pairing cp;
Vinicius Costa Gomesf1cb9af2011-01-26 21:42:57 -0300723 struct hci_conn *hcon = conn->hcon;
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300724 struct smp_chan *smp;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300725
726 BT_DBG("conn %p", conn);
727
Johan Hedberg86ca9ea2013-11-05 11:30:39 +0200728 if (!(conn->hcon->link_mode & HCI_LM_MASTER))
729 return SMP_CMD_NOTSUPP;
730
Brian Gix2b64d152011-12-21 16:12:12 -0800731 hcon->pending_sec_level = authreq_to_seclevel(rp->auth_req);
Vinicius Costa Gomesfeb45eb2011-08-25 20:02:35 -0300732
Johan Hedberg4dab7862012-06-07 14:58:37 +0800733 if (smp_ltk_encrypt(conn, hcon->pending_sec_level))
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300734 return 0;
735
Johan Hedberg51a8efd2012-01-16 06:10:31 +0200736 if (test_and_set_bit(HCI_CONN_LE_SMP_PEND, &hcon->flags))
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300737 return 0;
Vinicius Costa Gomesf1cb9af2011-01-26 21:42:57 -0300738
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300739 smp = smp_chan_create(conn);
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -0300740
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300741 skb_pull(skb, sizeof(*rp));
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300742
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300743 memset(&cp, 0, sizeof(cp));
Vinicius Costa Gomes54790f72011-07-07 18:59:38 -0300744 build_pairing_cmd(conn, &cp, NULL, rp->auth_req);
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300745
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300746 smp->preq[0] = SMP_CMD_PAIRING_REQ;
747 memcpy(&smp->preq[1], &cp, sizeof(cp));
Anderson Brigliaf01ead32011-06-09 18:50:45 -0300748
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300749 smp_send_cmd(conn, SMP_CMD_PAIRING_REQ, sizeof(cp), &cp);
Vinicius Costa Gomesf1cb9af2011-01-26 21:42:57 -0300750
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300751 return 0;
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300752}
753
Johan Hedbergad32a2f2013-05-14 18:05:12 +0300754bool smp_sufficient_security(struct hci_conn *hcon, u8 sec_level)
755{
756 if (sec_level == BT_SECURITY_LOW)
757 return true;
758
759 if (hcon->sec_level >= sec_level)
760 return true;
761
762 return false;
763}
764
Vinicius Costa Gomescc110922012-08-23 21:32:43 -0300765int smp_conn_security(struct hci_conn *hcon, __u8 sec_level)
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300766{
Vinicius Costa Gomescc110922012-08-23 21:32:43 -0300767 struct l2cap_conn *conn = hcon->l2cap_data;
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300768 struct smp_chan *smp = conn->smp_chan;
Brian Gix2b64d152011-12-21 16:12:12 -0800769 __u8 authreq;
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300770
Vinicius Costa Gomes3a0259b2011-06-09 18:50:43 -0300771 BT_DBG("conn %p hcon %p level 0x%2.2x", conn, hcon, sec_level);
772
Johan Hedberg757aee02013-04-24 13:05:32 +0300773 if (!test_bit(HCI_LE_ENABLED, &hcon->hdev->dev_flags))
Andre Guedes2e65c9d2011-06-30 19:20:56 -0300774 return 1;
775
Johan Hedbergad32a2f2013-05-14 18:05:12 +0300776 if (smp_sufficient_security(hcon, sec_level))
Vinicius Costa Gomesf1cb9af2011-01-26 21:42:57 -0300777 return 1;
778
Vinicius Costa Gomes988c5992011-08-25 20:02:28 -0300779 if (hcon->link_mode & HCI_LM_MASTER)
Johan Hedberg4dab7862012-06-07 14:58:37 +0800780 if (smp_ltk_encrypt(conn, sec_level))
Vinicius Costa Gomes02bc7452011-07-07 18:59:41 -0300781 goto done;
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -0300782
Johan Hedberg51a8efd2012-01-16 06:10:31 +0200783 if (test_and_set_bit(HCI_CONN_LE_SMP_PEND, &hcon->flags))
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -0300784 return 0;
785
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -0300786 smp = smp_chan_create(conn);
Brian Gix2b64d152011-12-21 16:12:12 -0800787 if (!smp)
788 return 1;
789
790 authreq = seclevel_to_authreq(sec_level);
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -0300791
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -0300792 if (hcon->link_mode & HCI_LM_MASTER) {
793 struct smp_cmd_pairing cp;
Anderson Brigliaf01ead32011-06-09 18:50:45 -0300794
Brian Gix2b64d152011-12-21 16:12:12 -0800795 build_pairing_cmd(conn, &cp, NULL, authreq);
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300796 smp->preq[0] = SMP_CMD_PAIRING_REQ;
797 memcpy(&smp->preq[1], &cp, sizeof(cp));
Anderson Brigliaf01ead32011-06-09 18:50:45 -0300798
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300799 smp_send_cmd(conn, SMP_CMD_PAIRING_REQ, sizeof(cp), &cp);
800 } else {
801 struct smp_cmd_security_req cp;
Brian Gix2b64d152011-12-21 16:12:12 -0800802 cp.auth_req = authreq;
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300803 smp_send_cmd(conn, SMP_CMD_SECURITY_REQ, sizeof(cp), &cp);
804 }
805
Vinicius Costa Gomes02bc7452011-07-07 18:59:41 -0300806done:
Vinicius Costa Gomesf1cb9af2011-01-26 21:42:57 -0300807 hcon->pending_sec_level = sec_level;
Vinicius Costa Gomesf1cb9af2011-01-26 21:42:57 -0300808
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300809 return 0;
810}
811
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300812static int smp_cmd_encrypt_info(struct l2cap_conn *conn, struct sk_buff *skb)
813{
Vinicius Costa Gomes16b90832011-07-07 18:59:39 -0300814 struct smp_cmd_encrypt_info *rp = (void *) skb->data;
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300815 struct smp_chan *smp = conn->smp_chan;
Vinicius Costa Gomes16b90832011-07-07 18:59:39 -0300816
817 skb_pull(skb, sizeof(*rp));
818
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300819 memcpy(smp->tk, rp->ltk, sizeof(smp->tk));
Vinicius Costa Gomes16b90832011-07-07 18:59:39 -0300820
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300821 return 0;
822}
823
824static int smp_cmd_master_ident(struct l2cap_conn *conn, struct sk_buff *skb)
825{
Vinicius Costa Gomes16b90832011-07-07 18:59:39 -0300826 struct smp_cmd_master_ident *rp = (void *) skb->data;
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300827 struct smp_chan *smp = conn->smp_chan;
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -0300828 struct hci_dev *hdev = conn->hcon->hdev;
829 struct hci_conn *hcon = conn->hcon;
830 u8 authenticated;
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300831
Vinicius Costa Gomes16b90832011-07-07 18:59:39 -0300832 skb_pull(skb, sizeof(*rp));
833
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -0300834 hci_dev_lock(hdev);
Marcel Holtmannce39fb42013-10-13 02:23:39 -0700835 authenticated = (hcon->sec_level == BT_SECURITY_HIGH);
836 hci_add_ltk(hdev, &hcon->dst, hcon->dst_type, HCI_SMP_LTK, 1,
837 authenticated, smp->tk, smp->enc_key_size,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300838 rp->ediv, rp->rand);
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300839 smp_distribute_keys(conn, 1);
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -0300840 hci_dev_unlock(hdev);
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300841
842 return 0;
843}
844
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300845int smp_sig_channel(struct l2cap_conn *conn, struct sk_buff *skb)
846{
Marcel Holtmann7b9899d2013-10-03 00:00:57 -0700847 struct hci_conn *hcon = conn->hcon;
Marcel Holtmann92381f52013-10-03 01:23:08 -0700848 __u8 code, reason;
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300849 int err = 0;
850
Marcel Holtmann7b9899d2013-10-03 00:00:57 -0700851 if (hcon->type != LE_LINK) {
852 kfree_skb(skb);
Johan Hedberg34327112013-10-16 11:37:01 +0300853 return 0;
Marcel Holtmann7b9899d2013-10-03 00:00:57 -0700854 }
855
Marcel Holtmann92381f52013-10-03 01:23:08 -0700856 if (skb->len < 1) {
857 kfree_skb(skb);
858 return -EILSEQ;
859 }
860
Marcel Holtmann06ae3312013-10-18 03:43:00 -0700861 if (!test_bit(HCI_LE_ENABLED, &hcon->hdev->dev_flags)) {
Andre Guedes2e65c9d2011-06-30 19:20:56 -0300862 err = -ENOTSUPP;
863 reason = SMP_PAIRING_NOTSUPP;
864 goto done;
865 }
866
Marcel Holtmann92381f52013-10-03 01:23:08 -0700867 code = skb->data[0];
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300868 skb_pull(skb, sizeof(code));
869
Johan Hedberg8cf9fa12013-01-29 10:44:23 -0600870 /*
871 * The SMP context must be initialized for all other PDUs except
872 * pairing and security requests. If we get any other PDU when
873 * not initialized simply disconnect (done if this function
874 * returns an error).
875 */
876 if (code != SMP_CMD_PAIRING_REQ && code != SMP_CMD_SECURITY_REQ &&
877 !conn->smp_chan) {
878 BT_ERR("Unexpected SMP command 0x%02x. Disconnecting.", code);
879 kfree_skb(skb);
880 return -ENOTSUPP;
881 }
882
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300883 switch (code) {
884 case SMP_CMD_PAIRING_REQ:
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300885 reason = smp_cmd_pairing_req(conn, skb);
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300886 break;
887
888 case SMP_CMD_PAIRING_FAIL:
Johan Hedberg84794e12013-11-06 11:24:57 +0200889 smp_failure(conn, 0);
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300890 reason = 0;
891 err = -EPERM;
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300892 break;
893
894 case SMP_CMD_PAIRING_RSP:
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300895 reason = smp_cmd_pairing_rsp(conn, skb);
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300896 break;
897
898 case SMP_CMD_SECURITY_REQ:
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300899 reason = smp_cmd_security_req(conn, skb);
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300900 break;
901
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300902 case SMP_CMD_PAIRING_CONFIRM:
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300903 reason = smp_cmd_pairing_confirm(conn, skb);
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300904 break;
905
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300906 case SMP_CMD_PAIRING_RANDOM:
Vinicius Costa Gomesda85e5e2011-06-09 18:50:53 -0300907 reason = smp_cmd_pairing_random(conn, skb);
Anderson Briglia88ba43b2011-06-09 18:50:42 -0300908 break;
909
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300910 case SMP_CMD_ENCRYPT_INFO:
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300911 reason = smp_cmd_encrypt_info(conn, skb);
912 break;
913
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300914 case SMP_CMD_MASTER_IDENT:
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300915 reason = smp_cmd_master_ident(conn, skb);
916 break;
917
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300918 case SMP_CMD_IDENT_INFO:
919 case SMP_CMD_IDENT_ADDR_INFO:
920 case SMP_CMD_SIGN_INFO:
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300921 /* Just ignored */
922 reason = 0;
923 break;
924
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300925 default:
926 BT_DBG("Unknown command code 0x%2.2x", code);
927
928 reason = SMP_CMD_NOTSUPP;
Vinicius Costa Gomes3a0259b2011-06-09 18:50:43 -0300929 err = -EOPNOTSUPP;
930 goto done;
931 }
932
933done:
934 if (reason)
Johan Hedberg84794e12013-11-06 11:24:57 +0200935 smp_failure(conn, reason);
Anderson Brigliaeb492e02011-06-09 18:50:40 -0300936
937 kfree_skb(skb);
938 return err;
939}
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300940
941int smp_distribute_keys(struct l2cap_conn *conn, __u8 force)
942{
943 struct smp_cmd_pairing *req, *rsp;
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300944 struct smp_chan *smp = conn->smp_chan;
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300945 __u8 *keydist;
946
947 BT_DBG("conn %p force %d", conn, force);
948
Johan Hedberg51a8efd2012-01-16 06:10:31 +0200949 if (!test_bit(HCI_CONN_LE_SMP_PEND, &conn->hcon->flags))
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -0300950 return 0;
951
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300952 rsp = (void *) &smp->prsp[1];
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300953
954 /* The responder sends its keys first */
955 if (!force && conn->hcon->out && (rsp->resp_key_dist & 0x07))
956 return 0;
957
Vinicius Costa Gomes1c1def02011-09-05 14:31:30 -0300958 req = (void *) &smp->preq[1];
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300959
960 if (conn->hcon->out) {
961 keydist = &rsp->init_key_dist;
962 *keydist &= req->init_key_dist;
963 } else {
964 keydist = &rsp->resp_key_dist;
965 *keydist &= req->resp_key_dist;
966 }
967
968
969 BT_DBG("keydist 0x%x", *keydist);
970
971 if (*keydist & SMP_DIST_ENC_KEY) {
972 struct smp_cmd_encrypt_info enc;
973 struct smp_cmd_master_ident ident;
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -0300974 struct hci_conn *hcon = conn->hcon;
975 u8 authenticated;
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300976 __le16 ediv;
977
978 get_random_bytes(enc.ltk, sizeof(enc.ltk));
979 get_random_bytes(&ediv, sizeof(ediv));
980 get_random_bytes(ident.rand, sizeof(ident.rand));
981
982 smp_send_cmd(conn, SMP_CMD_ENCRYPT_INFO, sizeof(enc), &enc);
983
Vinicius Costa Gomesc9839a12012-02-02 21:08:01 -0300984 authenticated = hcon->sec_level == BT_SECURITY_HIGH;
Marcel Holtmannce39fb42013-10-13 02:23:39 -0700985 hci_add_ltk(hcon->hdev, &hcon->dst, hcon->dst_type,
Gustavo F. Padovan04124682012-03-08 01:25:00 -0300986 HCI_SMP_LTK_SLAVE, 1, authenticated,
987 enc.ltk, smp->enc_key_size, ediv, ident.rand);
Vinicius Costa Gomes16b90832011-07-07 18:59:39 -0300988
Andrei Emeltchenko58115372012-03-12 12:13:06 +0200989 ident.ediv = ediv;
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -0300990
991 smp_send_cmd(conn, SMP_CMD_MASTER_IDENT, sizeof(ident), &ident);
992
993 *keydist &= ~SMP_DIST_ENC_KEY;
994 }
995
996 if (*keydist & SMP_DIST_ID_KEY) {
997 struct smp_cmd_ident_addr_info addrinfo;
998 struct smp_cmd_ident_info idinfo;
999
1000 /* Send a dummy key */
1001 get_random_bytes(idinfo.irk, sizeof(idinfo.irk));
1002
1003 smp_send_cmd(conn, SMP_CMD_IDENT_INFO, sizeof(idinfo), &idinfo);
1004
1005 /* Just public address */
1006 memset(&addrinfo, 0, sizeof(addrinfo));
Marcel Holtmann2b36a562013-10-13 05:24:00 -07001007 bacpy(&addrinfo.bdaddr, &conn->hcon->src);
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -03001008
1009 smp_send_cmd(conn, SMP_CMD_IDENT_ADDR_INFO, sizeof(addrinfo),
Marcel Holtmannf1560462013-10-13 05:43:25 -07001010 &addrinfo);
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -03001011
1012 *keydist &= ~SMP_DIST_ID_KEY;
1013 }
1014
1015 if (*keydist & SMP_DIST_SIGN) {
1016 struct smp_cmd_sign_info sign;
1017
1018 /* Send a dummy key */
1019 get_random_bytes(sign.csrk, sizeof(sign.csrk));
1020
1021 smp_send_cmd(conn, SMP_CMD_SIGN_INFO, sizeof(sign), &sign);
1022
1023 *keydist &= ~SMP_DIST_SIGN;
1024 }
1025
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -03001026 if (conn->hcon->out || force) {
Johan Hedberg51a8efd2012-01-16 06:10:31 +02001027 clear_bit(HCI_CONN_LE_SMP_PEND, &conn->hcon->flags);
Gustavo F. Padovan6c9d42a2011-12-20 10:57:27 -02001028 cancel_delayed_work_sync(&conn->security_timer);
Vinicius Costa Gomes8aab4752011-09-05 14:31:31 -03001029 smp_chan_destroy(conn);
Vinicius Costa Gomesd26a2342011-08-19 21:06:51 -03001030 }
1031
Vinicius Costa Gomes7034b912011-07-07 18:59:34 -03001032 return 0;
1033}