Commit fe39c7b2 authored by Marcel Holtmann's avatar Marcel Holtmann Committed by Johan Hedberg

Bluetooth: Use __le64 type for LE random numbers

The random numbers in Bluetooth Low Energy are 64-bit numbers and should
also be little endian since the HCI specification is little endian.

Change the whole Low Energy pairing to use __le64 instead of a byte
array.
Signed-off-by: default avatarMarcel Holtmann <marcel@holtmann.org>
Signed-off-by: default avatarJohan Hedberg <johan.hedberg@intel.com>
parent a3172b7e
...@@ -1234,7 +1234,7 @@ struct hci_cp_le_conn_update { ...@@ -1234,7 +1234,7 @@ struct hci_cp_le_conn_update {
#define HCI_OP_LE_START_ENC 0x2019 #define HCI_OP_LE_START_ENC 0x2019
struct hci_cp_le_start_enc { struct hci_cp_le_start_enc {
__le16 handle; __le16 handle;
__u8 rand[8]; __le64 rand;
__le16 ediv; __le16 ediv;
__u8 ltk[16]; __u8 ltk[16];
} __packed; } __packed;
...@@ -1646,7 +1646,7 @@ struct hci_ev_le_conn_complete { ...@@ -1646,7 +1646,7 @@ struct hci_ev_le_conn_complete {
#define HCI_EV_LE_LTK_REQ 0x05 #define HCI_EV_LE_LTK_REQ 0x05
struct hci_ev_le_ltk_req { struct hci_ev_le_ltk_req {
__le16 handle; __le16 handle;
__u8 random[8]; __le64 rand;
__le16 ediv; __le16 ediv;
} __packed; } __packed;
......
...@@ -99,7 +99,7 @@ struct smp_ltk { ...@@ -99,7 +99,7 @@ struct smp_ltk {
u8 type; u8 type;
u8 enc_size; u8 enc_size;
__le16 ediv; __le16 ediv;
u8 rand[8]; __le64 rand;
u8 val[16]; u8 val[16];
}; };
...@@ -828,11 +828,11 @@ void hci_link_keys_clear(struct hci_dev *hdev); ...@@ -828,11 +828,11 @@ void hci_link_keys_clear(struct hci_dev *hdev);
struct link_key *hci_find_link_key(struct hci_dev *hdev, bdaddr_t *bdaddr); struct link_key *hci_find_link_key(struct hci_dev *hdev, bdaddr_t *bdaddr);
int hci_add_link_key(struct hci_dev *hdev, struct hci_conn *conn, int new_key, int hci_add_link_key(struct hci_dev *hdev, struct hci_conn *conn, int new_key,
bdaddr_t *bdaddr, u8 *val, u8 type, u8 pin_len); bdaddr_t *bdaddr, u8 *val, u8 type, u8 pin_len);
struct smp_ltk *hci_find_ltk(struct hci_dev *hdev, __le16 ediv, u8 rand[8], struct smp_ltk *hci_find_ltk(struct hci_dev *hdev, __le16 ediv, __le64 rand,
bool master); bool master);
struct smp_ltk *hci_add_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr, struct smp_ltk *hci_add_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr,
u8 addr_type, u8 type, u8 authenticated, u8 addr_type, u8 type, u8 authenticated,
u8 tk[16], u8 enc_size, __le16 ediv, u8 rand[8]); u8 tk[16], u8 enc_size, __le16 ediv, __le64 rand);
struct smp_ltk *hci_find_ltk_by_addr(struct hci_dev *hdev, bdaddr_t *bdaddr, struct smp_ltk *hci_find_ltk_by_addr(struct hci_dev *hdev, bdaddr_t *bdaddr,
u8 addr_type, bool master); u8 addr_type, bool master);
int hci_remove_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 bdaddr_type); int hci_remove_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr, u8 bdaddr_type);
...@@ -1293,7 +1293,7 @@ struct hci_sec_filter { ...@@ -1293,7 +1293,7 @@ struct hci_sec_filter {
void hci_le_conn_update(struct hci_conn *conn, u16 min, u16 max, void hci_le_conn_update(struct hci_conn *conn, u16 min, u16 max,
u16 latency, u16 to_multiplier); u16 latency, u16 to_multiplier);
void hci_le_start_enc(struct hci_conn *conn, __le16 ediv, __u8 rand[8], void hci_le_start_enc(struct hci_conn *conn, __le16 ediv, __le64 rand,
__u8 ltk[16]); __u8 ltk[16]);
int hci_update_random_address(struct hci_request *req, bool require_privacy, int hci_update_random_address(struct hci_request *req, bool require_privacy,
......
...@@ -187,7 +187,7 @@ struct mgmt_ltk_info { ...@@ -187,7 +187,7 @@ struct mgmt_ltk_info {
__u8 master; __u8 master;
__u8 enc_size; __u8 enc_size;
__le16 ediv; __le16 ediv;
__u8 rand[8]; __le64 rand;
__u8 val[16]; __u8 val[16];
} __packed; } __packed;
......
...@@ -231,7 +231,7 @@ void hci_le_conn_update(struct hci_conn *conn, u16 min, u16 max, ...@@ -231,7 +231,7 @@ void hci_le_conn_update(struct hci_conn *conn, u16 min, u16 max,
hci_send_cmd(hdev, HCI_OP_LE_CONN_UPDATE, sizeof(cp), &cp); hci_send_cmd(hdev, HCI_OP_LE_CONN_UPDATE, sizeof(cp), &cp);
} }
void hci_le_start_enc(struct hci_conn *conn, __le16 ediv, __u8 rand[8], void hci_le_start_enc(struct hci_conn *conn, __le16 ediv, __le64 rand,
__u8 ltk[16]) __u8 ltk[16])
{ {
struct hci_dev *hdev = conn->hdev; struct hci_dev *hdev = conn->hdev;
...@@ -242,9 +242,9 @@ void hci_le_start_enc(struct hci_conn *conn, __le16 ediv, __u8 rand[8], ...@@ -242,9 +242,9 @@ void hci_le_start_enc(struct hci_conn *conn, __le16 ediv, __u8 rand[8],
memset(&cp, 0, sizeof(cp)); memset(&cp, 0, sizeof(cp));
cp.handle = cpu_to_le16(conn->handle); cp.handle = cpu_to_le16(conn->handle);
memcpy(cp.ltk, ltk, sizeof(cp.ltk)); cp.rand = rand;
cp.ediv = ediv; cp.ediv = ediv;
memcpy(cp.rand, rand, sizeof(cp.rand)); memcpy(cp.ltk, ltk, sizeof(cp.ltk));
hci_send_cmd(hdev, HCI_OP_LE_START_ENC, sizeof(cp), &cp); hci_send_cmd(hdev, HCI_OP_LE_START_ENC, sizeof(cp), &cp);
} }
......
...@@ -765,10 +765,10 @@ static int long_term_keys_show(struct seq_file *f, void *ptr) ...@@ -765,10 +765,10 @@ static int long_term_keys_show(struct seq_file *f, void *ptr)
hci_dev_lock(hdev); hci_dev_lock(hdev);
list_for_each_safe(p, n, &hdev->long_term_keys) { list_for_each_safe(p, n, &hdev->long_term_keys) {
struct smp_ltk *ltk = list_entry(p, struct smp_ltk, list); struct smp_ltk *ltk = list_entry(p, struct smp_ltk, list);
seq_printf(f, "%pMR (type %u) %u 0x%02x %u %.4x %*phN %*phN\n", seq_printf(f, "%pMR (type %u) %u 0x%02x %u %.4x %.16llx %*phN\n",
&ltk->bdaddr, ltk->bdaddr_type, ltk->authenticated, &ltk->bdaddr, ltk->bdaddr_type, ltk->authenticated,
ltk->type, ltk->enc_size, __le16_to_cpu(ltk->ediv), ltk->type, ltk->enc_size, __le16_to_cpu(ltk->ediv),
8, ltk->rand, 16, ltk->val); __le64_to_cpu(ltk->rand), 16, ltk->val);
} }
hci_dev_unlock(hdev); hci_dev_unlock(hdev);
...@@ -2921,14 +2921,13 @@ static bool ltk_type_master(u8 type) ...@@ -2921,14 +2921,13 @@ static bool ltk_type_master(u8 type)
return false; return false;
} }
struct smp_ltk *hci_find_ltk(struct hci_dev *hdev, __le16 ediv, u8 rand[8], struct smp_ltk *hci_find_ltk(struct hci_dev *hdev, __le16 ediv, __le64 rand,
bool master) bool master)
{ {
struct smp_ltk *k; struct smp_ltk *k;
list_for_each_entry(k, &hdev->long_term_keys, list) { list_for_each_entry(k, &hdev->long_term_keys, list) {
if (k->ediv != ediv || if (k->ediv != ediv || k->rand != rand)
memcmp(rand, k->rand, sizeof(k->rand)))
continue; continue;
if (ltk_type_master(k->type) != master) if (ltk_type_master(k->type) != master)
...@@ -3046,7 +3045,7 @@ int hci_add_link_key(struct hci_dev *hdev, struct hci_conn *conn, int new_key, ...@@ -3046,7 +3045,7 @@ int hci_add_link_key(struct hci_dev *hdev, struct hci_conn *conn, int new_key,
struct smp_ltk *hci_add_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr, struct smp_ltk *hci_add_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr,
u8 addr_type, u8 type, u8 authenticated, u8 addr_type, u8 type, u8 authenticated,
u8 tk[16], u8 enc_size, __le16 ediv, u8 rand[8]) u8 tk[16], u8 enc_size, __le16 ediv, __le64 rand)
{ {
struct smp_ltk *key, *old_key; struct smp_ltk *key, *old_key;
bool master = ltk_type_master(type); bool master = ltk_type_master(type);
...@@ -3066,9 +3065,9 @@ struct smp_ltk *hci_add_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr, ...@@ -3066,9 +3065,9 @@ struct smp_ltk *hci_add_ltk(struct hci_dev *hdev, bdaddr_t *bdaddr,
memcpy(key->val, tk, sizeof(key->val)); memcpy(key->val, tk, sizeof(key->val));
key->authenticated = authenticated; key->authenticated = authenticated;
key->ediv = ediv; key->ediv = ediv;
key->rand = rand;
key->enc_size = enc_size; key->enc_size = enc_size;
key->type = type; key->type = type;
memcpy(key->rand, rand, sizeof(key->rand));
return key; return key;
} }
......
...@@ -3843,7 +3843,7 @@ static void hci_le_ltk_request_evt(struct hci_dev *hdev, struct sk_buff *skb) ...@@ -3843,7 +3843,7 @@ static void hci_le_ltk_request_evt(struct hci_dev *hdev, struct sk_buff *skb)
if (conn == NULL) if (conn == NULL)
goto not_found; goto not_found;
ltk = hci_find_ltk(hdev, ev->ediv, ev->random, conn->out); ltk = hci_find_ltk(hdev, ev->ediv, ev->rand, conn->out);
if (ltk == NULL) if (ltk == NULL)
goto not_found; goto not_found;
......
...@@ -5025,11 +5025,11 @@ void mgmt_new_ltk(struct hci_dev *hdev, struct smp_ltk *key) ...@@ -5025,11 +5025,11 @@ void mgmt_new_ltk(struct hci_dev *hdev, struct smp_ltk *key)
ev.key.type = key->authenticated; ev.key.type = key->authenticated;
ev.key.enc_size = key->enc_size; ev.key.enc_size = key->enc_size;
ev.key.ediv = key->ediv; ev.key.ediv = key->ediv;
ev.key.rand = key->rand;
if (key->type == HCI_SMP_LTK) if (key->type == HCI_SMP_LTK)
ev.key.master = 1; ev.key.master = 1;
memcpy(ev.key.rand, key->rand, sizeof(key->rand));
memcpy(ev.key.val, key->val, sizeof(key->val)); memcpy(ev.key.val, key->val, sizeof(key->val));
mgmt_event(MGMT_EV_NEW_LONG_TERM_KEY, hdev, &ev, sizeof(ev), NULL); mgmt_event(MGMT_EV_NEW_LONG_TERM_KEY, hdev, &ev, sizeof(ev), NULL);
......
...@@ -517,11 +517,9 @@ static void random_work(struct work_struct *work) ...@@ -517,11 +517,9 @@ static void random_work(struct work_struct *work)
} }
if (hcon->out) { if (hcon->out) {
u8 stk[16], rand[8]; u8 stk[16];
__le16 ediv; __le64 rand = 0;
__le16 ediv = 0;
memset(rand, 0, sizeof(rand));
ediv = 0;
smp_s1(tfm, smp->tk, smp->rrnd, smp->prnd, key); smp_s1(tfm, smp->tk, smp->rrnd, smp->prnd, key);
swap128(key, stk); swap128(key, stk);
...@@ -537,11 +535,9 @@ static void random_work(struct work_struct *work) ...@@ -537,11 +535,9 @@ static void random_work(struct work_struct *work)
hci_le_start_enc(hcon, ediv, rand, stk); hci_le_start_enc(hcon, ediv, rand, stk);
hcon->enc_key_size = smp->enc_key_size; hcon->enc_key_size = smp->enc_key_size;
} else { } else {
u8 stk[16], r[16], rand[8]; u8 stk[16], r[16];
__le16 ediv; __le64 rand = 0;
__le16 ediv = 0;
memset(rand, 0, sizeof(rand));
ediv = 0;
swap128(smp->prnd, r); swap128(smp->prnd, r);
smp_send_cmd(conn, SMP_CMD_PAIRING_RANDOM, sizeof(r), r); smp_send_cmd(conn, SMP_CMD_PAIRING_RANDOM, sizeof(r), r);
...@@ -1205,20 +1201,22 @@ int smp_distribute_keys(struct l2cap_conn *conn) ...@@ -1205,20 +1201,22 @@ int smp_distribute_keys(struct l2cap_conn *conn)
struct smp_ltk *ltk; struct smp_ltk *ltk;
u8 authenticated; u8 authenticated;
__le16 ediv; __le16 ediv;
__le64 rand;
get_random_bytes(enc.ltk, sizeof(enc.ltk)); get_random_bytes(enc.ltk, sizeof(enc.ltk));
get_random_bytes(&ediv, sizeof(ediv)); get_random_bytes(&ediv, sizeof(ediv));
get_random_bytes(ident.rand, sizeof(ident.rand)); get_random_bytes(&rand, sizeof(rand));
smp_send_cmd(conn, SMP_CMD_ENCRYPT_INFO, sizeof(enc), &enc); smp_send_cmd(conn, SMP_CMD_ENCRYPT_INFO, sizeof(enc), &enc);
authenticated = hcon->sec_level == BT_SECURITY_HIGH; authenticated = hcon->sec_level == BT_SECURITY_HIGH;
ltk = hci_add_ltk(hdev, &hcon->dst, hcon->dst_type, ltk = hci_add_ltk(hdev, &hcon->dst, hcon->dst_type,
HCI_SMP_LTK_SLAVE, authenticated, enc.ltk, HCI_SMP_LTK_SLAVE, authenticated, enc.ltk,
smp->enc_key_size, ediv, ident.rand); smp->enc_key_size, ediv, rand);
smp->slave_ltk = ltk; smp->slave_ltk = ltk;
ident.ediv = ediv; ident.ediv = ediv;
ident.rand = rand;
smp_send_cmd(conn, SMP_CMD_MASTER_IDENT, sizeof(ident), &ident); smp_send_cmd(conn, SMP_CMD_MASTER_IDENT, sizeof(ident), &ident);
......
...@@ -78,7 +78,7 @@ struct smp_cmd_encrypt_info { ...@@ -78,7 +78,7 @@ struct smp_cmd_encrypt_info {
#define SMP_CMD_MASTER_IDENT 0x07 #define SMP_CMD_MASTER_IDENT 0x07
struct smp_cmd_master_ident { struct smp_cmd_master_ident {
__le16 ediv; __le16 ediv;
__u8 rand[8]; __le64 rand;
} __packed; } __packed;
#define SMP_CMD_IDENT_INFO 0x08 #define SMP_CMD_IDENT_INFO 0x08
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment