|
@@ -179,7 +179,7 @@ static int read_controller_info(struct sock *sk, u16 index)
|
|
|
|
|
|
hci_del_off_timer(hdev);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
set_bit(HCI_MGMT, &hdev->flags);
|
|
|
|
|
@@ -208,7 +208,7 @@ static int read_controller_info(struct sock *sk, u16 index)
|
|
|
|
|
|
memcpy(rp.name, hdev->dev_name, sizeof(hdev->dev_name));
|
|
|
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return cmd_complete(sk, index, MGMT_OP_READ_INFO, &rp, sizeof(rp));
|
|
@@ -316,7 +316,7 @@ static int set_powered(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_POWERED, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
up = test_bit(HCI_UP, &hdev->flags);
|
|
|
if ((cp->val && up) || (!cp->val && !up)) {
|
|
@@ -343,7 +343,7 @@ static int set_powered(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
err = 0;
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
return err;
|
|
|
}
|
|
@@ -368,7 +368,7 @@ static int set_discoverable(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_DISCOVERABLE, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (!test_bit(HCI_UP, &hdev->flags)) {
|
|
|
err = cmd_status(sk, index, MGMT_OP_SET_DISCOVERABLE, ENETDOWN);
|
|
@@ -403,7 +403,7 @@ static int set_discoverable(struct sock *sk, u16 index, unsigned char *data,
|
|
|
mgmt_pending_remove(cmd);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -429,7 +429,7 @@ static int set_connectable(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_CONNECTABLE, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (!test_bit(HCI_UP, &hdev->flags)) {
|
|
|
err = cmd_status(sk, index, MGMT_OP_SET_CONNECTABLE, ENETDOWN);
|
|
@@ -463,7 +463,7 @@ static int set_connectable(struct sock *sk, u16 index, unsigned char *data,
|
|
|
mgmt_pending_remove(cmd);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -522,7 +522,7 @@ static int set_pairable(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_PAIRABLE, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (cp->val)
|
|
|
set_bit(HCI_PAIRABLE, &hdev->flags);
|
|
@@ -538,7 +538,7 @@ static int set_pairable(struct sock *sk, u16 index, unsigned char *data,
|
|
|
err = mgmt_event(MGMT_EV_PAIRABLE, index, &ev, sizeof(ev), sk);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -739,7 +739,7 @@ static int add_uuid(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_ADD_UUID, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
uuid = kmalloc(sizeof(*uuid), GFP_ATOMIC);
|
|
|
if (!uuid) {
|
|
@@ -763,7 +763,7 @@ static int add_uuid(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
err = cmd_complete(sk, index, MGMT_OP_ADD_UUID, NULL, 0);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -788,7 +788,7 @@ static int remove_uuid(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_REMOVE_UUID, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (memcmp(cp->uuid, bt_uuid_any, 16) == 0) {
|
|
|
err = hci_uuids_clear(hdev);
|
|
@@ -823,7 +823,7 @@ static int remove_uuid(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
err = cmd_complete(sk, index, MGMT_OP_REMOVE_UUID, NULL, 0);
|
|
|
|
|
|
unlock:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -847,7 +847,7 @@ static int set_dev_class(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_DEV_CLASS, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
hdev->major_class = cp->major;
|
|
|
hdev->minor_class = cp->minor;
|
|
@@ -857,7 +857,7 @@ static int set_dev_class(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (err == 0)
|
|
|
err = cmd_complete(sk, index, MGMT_OP_SET_DEV_CLASS, NULL, 0);
|
|
|
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -879,7 +879,7 @@ static int set_service_cache(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_SERVICE_CACHE, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
BT_DBG("hci%u enable %d", index, cp->enable);
|
|
|
|
|
@@ -897,7 +897,7 @@ static int set_service_cache(struct sock *sk, u16 index, unsigned char *data,
|
|
|
err = cmd_complete(sk, index, MGMT_OP_SET_SERVICE_CACHE, NULL,
|
|
|
0);
|
|
|
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -908,7 +908,7 @@ static int load_keys(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
struct hci_dev *hdev;
|
|
|
struct mgmt_cp_load_keys *cp;
|
|
|
u16 key_count, expected_len;
|
|
|
- int i;
|
|
|
+ int i, err;
|
|
|
|
|
|
cp = (void *) data;
|
|
|
|
|
@@ -918,9 +918,9 @@ static int load_keys(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
key_count = get_unaligned_le16(&cp->key_count);
|
|
|
|
|
|
expected_len = sizeof(*cp) + key_count * sizeof(struct mgmt_key_info);
|
|
|
- if (expected_len != len) {
|
|
|
- BT_ERR("load_keys: expected %u bytes, got %u bytes",
|
|
|
- len, expected_len);
|
|
|
+ if (expected_len > len) {
|
|
|
+ BT_ERR("load_keys: expected at least %u bytes, got %u bytes",
|
|
|
+ expected_len, len);
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
|
|
@@ -931,7 +931,7 @@ static int load_keys(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
BT_DBG("hci%u debug_keys %u key_count %u", index, cp->debug_keys,
|
|
|
key_count);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
hci_link_keys_clear(hdev);
|
|
|
|
|
@@ -942,17 +942,36 @@ static int load_keys(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
else
|
|
|
clear_bit(HCI_DEBUG_KEYS, &hdev->flags);
|
|
|
|
|
|
- for (i = 0; i < key_count; i++) {
|
|
|
- struct mgmt_key_info *key = &cp->keys[i];
|
|
|
+ len -= sizeof(*cp);
|
|
|
+ i = 0;
|
|
|
+
|
|
|
+ while (i < len) {
|
|
|
+ struct mgmt_key_info *key = (void *) cp->keys + i;
|
|
|
+
|
|
|
+ i += sizeof(*key) + key->dlen;
|
|
|
+
|
|
|
+ if (key->type == HCI_LK_SMP_LTK) {
|
|
|
+ struct key_master_id *id = (void *) key->data;
|
|
|
+
|
|
|
+ if (key->dlen != sizeof(struct key_master_id))
|
|
|
+ continue;
|
|
|
+
|
|
|
+ hci_add_ltk(hdev, 0, &key->bdaddr, key->pin_len,
|
|
|
+ id->ediv, id->rand, key->val);
|
|
|
+
|
|
|
+ continue;
|
|
|
+ }
|
|
|
|
|
|
hci_add_link_key(hdev, NULL, 0, &key->bdaddr, key->val, key->type,
|
|
|
key->pin_len);
|
|
|
}
|
|
|
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ err = cmd_complete(sk, index, MGMT_OP_LOAD_KEYS, NULL, 0);
|
|
|
+
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
- return 0;
|
|
|
+ return err;
|
|
|
}
|
|
|
|
|
|
static int remove_key(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
@@ -971,7 +990,7 @@ static int remove_key(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_REMOVE_KEY, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
err = hci_remove_link_key(hdev, &cp->bdaddr);
|
|
|
if (err < 0) {
|
|
@@ -994,7 +1013,7 @@ static int remove_key(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
}
|
|
|
|
|
|
unlock:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1020,7 +1039,7 @@ static int disconnect(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_DISCONNECT, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (!test_bit(HCI_UP, &hdev->flags)) {
|
|
|
err = cmd_status(sk, index, MGMT_OP_DISCONNECT, ENETDOWN);
|
|
@@ -1055,7 +1074,7 @@ static int disconnect(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
mgmt_pending_remove(cmd);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1076,7 +1095,7 @@ static int get_connections(struct sock *sk, u16 index)
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_GET_CONNECTIONS, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
count = 0;
|
|
|
list_for_each(p, &hdev->conn_hash.list) {
|
|
@@ -1103,7 +1122,7 @@ static int get_connections(struct sock *sk, u16 index)
|
|
|
|
|
|
unlock:
|
|
|
kfree(rp);
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
return err;
|
|
|
}
|
|
@@ -1149,7 +1168,7 @@ static int pin_code_reply(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_PIN_CODE_REPLY, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (!test_bit(HCI_UP, &hdev->flags)) {
|
|
|
err = cmd_status(sk, index, MGMT_OP_PIN_CODE_REPLY, ENETDOWN);
|
|
@@ -1190,7 +1209,7 @@ static int pin_code_reply(struct sock *sk, u16 index, unsigned char *data,
|
|
|
mgmt_pending_remove(cmd);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1216,7 +1235,7 @@ static int pin_code_neg_reply(struct sock *sk, u16 index, unsigned char *data,
|
|
|
return cmd_status(sk, index, MGMT_OP_PIN_CODE_NEG_REPLY,
|
|
|
ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (!test_bit(HCI_UP, &hdev->flags)) {
|
|
|
err = cmd_status(sk, index, MGMT_OP_PIN_CODE_NEG_REPLY,
|
|
@@ -1227,7 +1246,7 @@ static int pin_code_neg_reply(struct sock *sk, u16 index, unsigned char *data,
|
|
|
err = send_pin_code_neg_reply(sk, index, hdev, cp);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1250,14 +1269,14 @@ static int set_io_capability(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_IO_CAPABILITY, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
hdev->io_capability = cp->io_capability;
|
|
|
|
|
|
BT_DBG("%s IO capability set to 0x%02x", hdev->name,
|
|
|
hdev->io_capability);
|
|
|
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return cmd_complete(sk, index, MGMT_OP_SET_IO_CAPABILITY, NULL, 0);
|
|
@@ -1343,7 +1362,7 @@ static int pair_device(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_PAIR_DEVICE, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (cp->io_cap == 0x03) {
|
|
|
sec_level = BT_SECURITY_MEDIUM;
|
|
@@ -1385,7 +1404,7 @@ static int pair_device(struct sock *sk, u16 index, unsigned char *data, u16 len)
|
|
|
err = 0;
|
|
|
|
|
|
unlock:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1417,7 +1436,7 @@ static int user_confirm_reply(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, mgmt_op, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (!test_bit(HCI_UP, &hdev->flags)) {
|
|
|
err = cmd_status(sk, index, mgmt_op, ENETDOWN);
|
|
@@ -1435,7 +1454,7 @@ static int user_confirm_reply(struct sock *sk, u16 index, unsigned char *data,
|
|
|
mgmt_pending_remove(cmd);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1459,7 +1478,7 @@ static int set_local_name(struct sock *sk, u16 index, unsigned char *data,
|
|
|
if (!hdev)
|
|
|
return cmd_status(sk, index, MGMT_OP_SET_LOCAL_NAME, ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
cmd = mgmt_pending_add(sk, MGMT_OP_SET_LOCAL_NAME, index, data, len);
|
|
|
if (!cmd) {
|
|
@@ -1474,7 +1493,7 @@ static int set_local_name(struct sock *sk, u16 index, unsigned char *data,
|
|
|
mgmt_pending_remove(cmd);
|
|
|
|
|
|
failed:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1493,7 +1512,7 @@ static int read_local_oob_data(struct sock *sk, u16 index)
|
|
|
return cmd_status(sk, index, MGMT_OP_READ_LOCAL_OOB_DATA,
|
|
|
ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
if (!test_bit(HCI_UP, &hdev->flags)) {
|
|
|
err = cmd_status(sk, index, MGMT_OP_READ_LOCAL_OOB_DATA,
|
|
@@ -1523,7 +1542,7 @@ static int read_local_oob_data(struct sock *sk, u16 index)
|
|
|
mgmt_pending_remove(cmd);
|
|
|
|
|
|
unlock:
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1547,7 +1566,7 @@ static int add_remote_oob_data(struct sock *sk, u16 index, unsigned char *data,
|
|
|
return cmd_status(sk, index, MGMT_OP_ADD_REMOTE_OOB_DATA,
|
|
|
ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
err = hci_add_remote_oob_data(hdev, &cp->bdaddr, cp->hash,
|
|
|
cp->randomizer);
|
|
@@ -1557,7 +1576,7 @@ static int add_remote_oob_data(struct sock *sk, u16 index, unsigned char *data,
|
|
|
err = cmd_complete(sk, index, MGMT_OP_ADD_REMOTE_OOB_DATA, NULL,
|
|
|
0);
|
|
|
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1581,7 +1600,7 @@ static int remove_remote_oob_data(struct sock *sk, u16 index,
|
|
|
return cmd_status(sk, index, MGMT_OP_REMOVE_REMOTE_OOB_DATA,
|
|
|
ENODEV);
|
|
|
|
|
|
- hci_dev_lock(hdev);
|
|
|
+ hci_dev_lock_bh(hdev);
|
|
|
|
|
|
err = hci_remove_remote_oob_data(hdev, &cp->bdaddr);
|
|
|
if (err < 0)
|
|
@@ -1591,7 +1610,7 @@ static int remove_remote_oob_data(struct sock *sk, u16 index,
|
|
|
err = cmd_complete(sk, index, MGMT_OP_REMOVE_REMOTE_OOB_DATA,
|
|
|
NULL, 0);
|
|
|
|
|
|
- hci_dev_unlock(hdev);
|
|
|
+ hci_dev_unlock_bh(hdev);
|
|
|
hci_dev_put(hdev);
|
|
|
|
|
|
return err;
|
|
@@ -1958,17 +1977,28 @@ int mgmt_connectable(u16 index, u8 connectable)
|
|
|
|
|
|
int mgmt_new_key(u16 index, struct link_key *key, u8 persistent)
|
|
|
{
|
|
|
- struct mgmt_ev_new_key ev;
|
|
|
+ struct mgmt_ev_new_key *ev;
|
|
|
+ int err, total;
|
|
|
|
|
|
- memset(&ev, 0, sizeof(ev));
|
|
|
+ total = sizeof(struct mgmt_ev_new_key) + key->dlen;
|
|
|
+ ev = kzalloc(total, GFP_ATOMIC);
|
|
|
+ if (!ev)
|
|
|
+ return -ENOMEM;
|
|
|
|
|
|
- ev.store_hint = persistent;
|
|
|
- bacpy(&ev.key.bdaddr, &key->bdaddr);
|
|
|
- ev.key.type = key->type;
|
|
|
- memcpy(ev.key.val, key->val, 16);
|
|
|
- ev.key.pin_len = key->pin_len;
|
|
|
+ bacpy(&ev->key.bdaddr, &key->bdaddr);
|
|
|
+ ev->key.type = key->type;
|
|
|
+ memcpy(ev->key.val, key->val, 16);
|
|
|
+ ev->key.pin_len = key->pin_len;
|
|
|
+ ev->key.dlen = key->dlen;
|
|
|
+ ev->store_hint = persistent;
|
|
|
|
|
|
- return mgmt_event(MGMT_EV_NEW_KEY, index, &ev, sizeof(ev), NULL);
|
|
|
+ memcpy(ev->key.data, key->data, key->dlen);
|
|
|
+
|
|
|
+ err = mgmt_event(MGMT_EV_NEW_KEY, index, ev, total, NULL);
|
|
|
+
|
|
|
+ kfree(ev);
|
|
|
+
|
|
|
+ return err;
|
|
|
}
|
|
|
|
|
|
int mgmt_connected(u16 index, bdaddr_t *bdaddr)
|