Blame src/shared/hci-crypto.c

Packit 34410b
/*
Packit 34410b
 *
Packit 34410b
 *  BlueZ - Bluetooth protocol stack for Linux
Packit 34410b
 *
Packit 34410b
 *  Copyright (C) 2013-2014  Intel Corporation
Packit 34410b
 *
Packit 34410b
 *
Packit 34410b
 *  This library is free software; you can redistribute it and/or
Packit 34410b
 *  modify it under the terms of the GNU Lesser General Public
Packit 34410b
 *  License as published by the Free Software Foundation; either
Packit 34410b
 *  version 2.1 of the License, or (at your option) any later version.
Packit 34410b
 *
Packit 34410b
 *  This library is distributed in the hope that it will be useful,
Packit 34410b
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 34410b
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 34410b
 *  Lesser General Public License for more details.
Packit 34410b
 *
Packit 34410b
 *  You should have received a copy of the GNU Lesser General Public
Packit 34410b
 *  License along with this library; if not, write to the Free Software
Packit 34410b
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
Packit 34410b
 *
Packit 34410b
 */
Packit 34410b
Packit 34410b
#ifdef HAVE_CONFIG_H
Packit 34410b
#include <config.h>
Packit 34410b
#endif
Packit 34410b
Packit 34410b
#include <string.h>
Packit 34410b
Packit 34410b
#include "monitor/bt.h"
Packit 34410b
#include "src/shared/util.h"
Packit 34410b
#include "src/shared/hci.h"
Packit 34410b
#include "src/shared/hci-crypto.h"
Packit 34410b
Packit 34410b
struct crypto_data {
Packit 34410b
	uint8_t size;
Packit 34410b
	bt_hci_crypto_func_t callback;
Packit 34410b
	void *user_data;
Packit 34410b
};
Packit 34410b
Packit 34410b
static void le_encrypt_callback(const void *response, uint8_t size,
Packit 34410b
							void *user_data)
Packit 34410b
{
Packit 34410b
	struct crypto_data *data = user_data;
Packit 34410b
	const struct bt_hci_rsp_le_encrypt *rsp = response;
Packit 34410b
Packit 34410b
	if (rsp->status) {
Packit 34410b
		data->callback(NULL, 0, data->user_data);
Packit 34410b
		return;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	data->callback(rsp->data, data->size, data->user_data);
Packit 34410b
}
Packit 34410b
Packit 34410b
static bool le_encrypt(struct bt_hci *hci, uint8_t size,
Packit 34410b
			const uint8_t key[16], const uint8_t plaintext[16],
Packit 34410b
			bt_hci_crypto_func_t callback, void *user_data)
Packit 34410b
{
Packit 34410b
	struct crypto_data *data;
Packit 34410b
	struct bt_hci_cmd_le_encrypt cmd;
Packit 34410b
Packit 34410b
	if (!callback || !size || size > 16)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	memcpy(cmd.key, key, 16);
Packit 34410b
	memcpy(cmd.plaintext, plaintext, 16);
Packit 34410b
Packit 34410b
	data = new0(struct crypto_data, 1);
Packit 34410b
	data->size = size;
Packit 34410b
	data->callback = callback;
Packit 34410b
	data->user_data = user_data;
Packit 34410b
Packit 34410b
	if (!bt_hci_send(hci, BT_HCI_CMD_LE_ENCRYPT, &cmd, sizeof(cmd),
Packit 34410b
					le_encrypt_callback, data, free)) {
Packit 34410b
		free(data);
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return true;
Packit 34410b
}
Packit 34410b
Packit 34410b
static void prand_callback(const void *response, uint8_t size,
Packit 34410b
							void *user_data)
Packit 34410b
{
Packit 34410b
	struct crypto_data *data = user_data;
Packit 34410b
	const struct bt_hci_rsp_le_rand *rsp = response;
Packit 34410b
	uint8_t prand[3];
Packit 34410b
Packit 34410b
	if (rsp->status) {
Packit 34410b
		data->callback(NULL, 0, data->user_data);
Packit 34410b
		return;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	prand[0] = (rsp->number & 0xff0000) >> 16;
Packit 34410b
	prand[1] = (rsp->number & 0x00ff00) >> 8;
Packit 34410b
	prand[2] = (rsp->number & 0x00003f) | 0x40;
Packit 34410b
Packit 34410b
	data->callback(prand, 3, data->user_data);
Packit 34410b
}
Packit 34410b
Packit 34410b
bool bt_hci_crypto_prand(struct bt_hci *hci,
Packit 34410b
			bt_hci_crypto_func_t callback, void *user_data)
Packit 34410b
{
Packit 34410b
	struct crypto_data *data;
Packit 34410b
Packit 34410b
	if (!callback)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	data = new0(struct crypto_data, 1);
Packit 34410b
	data->callback = callback;
Packit 34410b
	data->user_data = user_data;
Packit 34410b
Packit 34410b
	if (!bt_hci_send(hci, BT_HCI_CMD_LE_RAND, NULL, 0,
Packit 34410b
					prand_callback, data, free)) {
Packit 34410b
		free(data);
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return true;
Packit 34410b
}
Packit 34410b
Packit 34410b
bool bt_hci_crypto_e(struct bt_hci *hci,
Packit 34410b
			const uint8_t key[16], const uint8_t plaintext[16],
Packit 34410b
			bt_hci_crypto_func_t callback, void *user_data)
Packit 34410b
{
Packit 34410b
	return le_encrypt(hci, 16, key, plaintext, callback, user_data);
Packit 34410b
}
Packit 34410b
Packit 34410b
bool bt_hci_crypto_d1(struct bt_hci *hci,
Packit 34410b
			const uint8_t k[16], uint16_t d, uint16_t r,
Packit 34410b
			bt_hci_crypto_func_t callback, void *user_data)
Packit 34410b
{
Packit 34410b
	uint8_t dp[16];
Packit 34410b
Packit 34410b
	/* d' = padding || r || d */
Packit 34410b
	dp[0] = d & 0xff;
Packit 34410b
	dp[1] = d >> 8;
Packit 34410b
	dp[2] = r & 0xff;
Packit 34410b
	dp[3] = r >> 8;
Packit 34410b
	memset(dp + 4, 0, 12);
Packit 34410b
Packit 34410b
	/* d1(k, d, r) = e(k, d') */
Packit 34410b
	return le_encrypt(hci, 16, k, dp, callback, user_data);
Packit 34410b
}
Packit 34410b
Packit 34410b
bool bt_hci_crypto_dm(struct bt_hci *hci,
Packit 34410b
			const uint8_t k[16], const uint8_t r[8],
Packit 34410b
			bt_hci_crypto_func_t callback, void *user_data)
Packit 34410b
{
Packit 34410b
	uint8_t rp[16];
Packit 34410b
Packit 34410b
	/* r' = padding || r */
Packit 34410b
	memcpy(rp, r, 8);
Packit 34410b
	memset(rp + 8, 0, 8);
Packit 34410b
Packit 34410b
	/* dm(k, r) = e(k, r') mod 2^16 */
Packit 34410b
	return le_encrypt(hci, 8, k, rp, callback, user_data);
Packit 34410b
}
Packit 34410b
Packit 34410b
bool bt_hci_crypto_ah(struct bt_hci *hci,
Packit 34410b
			const uint8_t k[16], const uint8_t r[3],
Packit 34410b
			bt_hci_crypto_func_t callback, void *user_data)
Packit 34410b
{
Packit 34410b
	uint8_t rp[16];
Packit 34410b
Packit 34410b
	/* r' = padding || r */
Packit 34410b
	memcpy(rp, r, 3);
Packit 34410b
	memset(rp + 3, 0, 13);
Packit 34410b
Packit 34410b
	/* ah(k, r) = e(k, r') mod 2^24 */
Packit 34410b
	return le_encrypt(hci, 3, k, rp, callback, user_data);
Packit 34410b
}