zephyr/net/bluetooth/keys.c
Luiz Augusto von Dentz 69130ed2c0 Bluetooth: keys: Add bt_keys_find_addr
bt_keys_find_addr lookup for any keys for the given address thus it can
be used to check if bonding procedure has been perfomed.

Change-Id: Ia398866418a908e35c8cd70e19b2bb2aa0a0be96
Signed-off-by: Luiz Augusto von Dentz <luiz.von.dentz@intel.com>
2016-02-05 20:14:44 -05:00

317 lines
7.1 KiB
C

/* keys.c - Bluetooth key handling */
/*
* Copyright (c) 2015 Intel Corporation
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1) Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* 2) Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* 3) Neither the name of Intel Corporation nor the names of its contributors
* may be used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include <string.h>
#include <nanokernel.h>
#include <misc/util.h>
#include <bluetooth/log.h>
#include <bluetooth/bluetooth.h>
#include <bluetooth/hci.h>
#include "hci_core.h"
#include "smp.h"
#include "keys.h"
#if !defined(CONFIG_BLUETOOTH_DEBUG_KEYS)
#undef BT_DBG
#define BT_DBG(fmt, ...)
#endif
#define bt_keys_foreach(list, cur, member) \
for (cur = list; *cur; cur = &(*cur)->member)
static struct bt_keys key_pool[CONFIG_BLUETOOTH_MAX_PAIRED];
static struct bt_keys *ltks;
static struct bt_keys *slave_ltks;
static struct bt_keys *irks;
static struct bt_keys *local_csrks;
static struct bt_keys *remote_csrks;
struct bt_keys *bt_keys_get_addr(const bt_addr_le_t *addr)
{
struct bt_keys *keys;
int i;
BT_DBG("%s\n", bt_addr_le_str(addr));
for (i = 0; i < ARRAY_SIZE(key_pool); i++) {
keys = &key_pool[i];
if (!bt_addr_le_cmp(&keys->addr, addr)) {
return keys;
}
if (!bt_addr_le_cmp(&keys->addr, BT_ADDR_LE_ANY)) {
bt_addr_le_copy(&keys->addr, addr);
BT_DBG("created %p for %s\n", keys,
bt_addr_le_str(addr));
return keys;
}
}
BT_DBG("unable to create keys for %s\n", bt_addr_le_str(addr));
return NULL;
}
void bt_keys_clear(struct bt_keys *keys, int type)
{
struct bt_keys **cur;
BT_DBG("keys for %s type %d\n", bt_addr_le_str(&keys->addr), type);
if (((type & keys->keys) & BT_KEYS_SLAVE_LTK)) {
bt_keys_foreach(&slave_ltks, cur, slave_ltk.next) {
if (*cur == keys) {
*cur = (*cur)->slave_ltk.next;
break;
}
}
keys->keys &= ~BT_KEYS_SLAVE_LTK;
}
if (((type & keys->keys) & BT_KEYS_LTK)) {
bt_keys_foreach(&ltks, cur, ltk.next) {
if (*cur == keys) {
*cur = (*cur)->ltk.next;
break;
}
}
keys->keys &= ~BT_KEYS_LTK;
}
if (((type & keys->keys) & BT_KEYS_IRK)) {
bt_keys_foreach(&irks, cur, irk.next) {
if (*cur == keys) {
*cur = (*cur)->irk.next;
break;
}
}
keys->keys &= ~BT_KEYS_IRK;
}
if (((type & keys->keys) & BT_KEYS_LOCAL_CSRK)) {
bt_keys_foreach(&local_csrks, cur, local_csrk.next) {
if (*cur == keys) {
*cur = (*cur)->local_csrk.next;
break;
}
}
keys->keys &= ~BT_KEYS_LOCAL_CSRK;
}
if (((type & keys->keys) & BT_KEYS_REMOTE_CSRK)) {
bt_keys_foreach(&remote_csrks, cur, remote_csrk.next) {
if (*cur == keys) {
*cur = (*cur)->remote_csrk.next;
break;
}
}
keys->keys &= ~BT_KEYS_REMOTE_CSRK;
}
if (!keys->keys) {
memset(keys, 0, sizeof(*keys));
}
}
struct bt_keys *bt_keys_find(int type, const bt_addr_le_t *addr)
{
struct bt_keys **cur;
BT_DBG("type %d %s\n", type, bt_addr_le_str(addr));
switch (type) {
case BT_KEYS_SLAVE_LTK:
bt_keys_foreach(&slave_ltks, cur, slave_ltk.next) {
if (!bt_addr_le_cmp(&(*cur)->addr, addr)) {
break;
}
}
return *cur;
case BT_KEYS_LTK:
bt_keys_foreach(&ltks, cur, ltk.next) {
if (!bt_addr_le_cmp(&(*cur)->addr, addr)) {
break;
}
}
return *cur;
case BT_KEYS_IRK:
bt_keys_foreach(&irks, cur, irk.next) {
if (!bt_addr_le_cmp(&(*cur)->addr, addr)) {
break;
}
}
return *cur;
case BT_KEYS_LOCAL_CSRK:
bt_keys_foreach(&local_csrks, cur, local_csrk.next) {
if (!bt_addr_le_cmp(&(*cur)->addr, addr)) {
break;
}
}
return *cur;
case BT_KEYS_REMOTE_CSRK:
bt_keys_foreach(&remote_csrks, cur, remote_csrk.next) {
if (!bt_addr_le_cmp(&(*cur)->addr, addr)) {
break;
}
}
return *cur;
default:
return NULL;
}
}
void bt_keys_add_type(struct bt_keys *keys, int type)
{
if (keys->keys & type) {
return;
}
switch (type) {
case BT_KEYS_SLAVE_LTK:
keys->slave_ltk.next = slave_ltks;
slave_ltks = keys;
break;
case BT_KEYS_LTK:
keys->ltk.next = ltks;
ltks = keys;
break;
case BT_KEYS_IRK:
keys->irk.next = irks;
irks = keys;
break;
case BT_KEYS_LOCAL_CSRK:
keys->local_csrk.next = local_csrks;
local_csrks = keys;
break;
case BT_KEYS_REMOTE_CSRK:
keys->remote_csrk.next = remote_csrks;
remote_csrks = keys;
break;
default:
BT_ERR("Unknown key type %d\n", type);
return;
}
keys->keys |= type;
}
struct bt_keys *bt_keys_get_type(int type, const bt_addr_le_t *addr)
{
struct bt_keys *keys;
BT_DBG("type %d %s\n", type, bt_addr_le_str(addr));
keys = bt_keys_find(type, addr);
if (keys) {
return keys;
}
keys = bt_keys_get_addr(addr);
if (!keys) {
return NULL;
}
bt_keys_add_type(keys, type);
return keys;
}
struct bt_keys *bt_keys_find_irk(const bt_addr_le_t *addr)
{
struct bt_keys **cur;
BT_DBG("%s\n", bt_addr_le_str(addr));
if (!bt_addr_le_is_rpa(addr)) {
return NULL;
}
bt_keys_foreach(&irks, cur, irk.next) {
struct bt_irk *irk = &(*cur)->irk;
if (!bt_addr_cmp((bt_addr_t *)addr->val, &irk->rpa)) {
BT_DBG("cached RPA %s for %s\n", bt_addr_str(&irk->rpa),
bt_addr_le_str(&(*cur)->addr));
return *cur;
}
if (bt_smp_irk_matches(irk->val, (bt_addr_t *)addr->val)) {
struct bt_keys *match = *cur;
BT_DBG("RPA %s matches %s\n", bt_addr_str(&irk->rpa),
bt_addr_le_str(&(*cur)->addr));
bt_addr_copy(&irk->rpa, (bt_addr_t *)addr->val);
/* Move to the beginning of the list for faster
* future lookups.
*/
if (match != irks) {
/* Remove match from list */
*cur = irk->next;
/* Add match to the beginning */
irk->next = irks;
irks = match;
}
return match;
}
}
BT_DBG("No IRK for %s\n", bt_addr_le_str(addr));
return NULL;
}
struct bt_keys *bt_keys_find_addr(const bt_addr_le_t *addr)
{
struct bt_keys *keys;
int i;
BT_DBG("%s\n", bt_addr_le_str(addr));
for (i = 0; i < ARRAY_SIZE(key_pool); i++) {
keys = &key_pool[i];
if (!bt_addr_le_cmp(&keys->addr, addr)) {
return keys;
}
}
return NULL;
}