Blame src/shared/btsnoop.c

Packit 34410b
/*
Packit 34410b
 *
Packit 34410b
 *  BlueZ - Bluetooth protocol stack for Linux
Packit 34410b
 *
Packit 34410b
 *  Copyright (C) 2012-2014  Intel Corporation. All rights reserved.
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
#define _GNU_SOURCE
Packit 34410b
#include <endian.h>
Packit 34410b
#include <fcntl.h>
Packit 34410b
#include <unistd.h>
Packit 34410b
#include <stdlib.h>
Packit 34410b
#include <string.h>
Packit 34410b
#include <stdio.h>
Packit 34410b
#include <limits.h>
Packit 34410b
#include <arpa/inet.h>
Packit 34410b
#include <sys/stat.h>
Packit 34410b
Packit 34410b
#include "src/shared/btsnoop.h"
Packit 34410b
Packit 34410b
struct btsnoop_hdr {
Packit 34410b
	uint8_t		id[8];		/* Identification Pattern */
Packit 34410b
	uint32_t	version;	/* Version Number = 1 */
Packit 34410b
	uint32_t	type;		/* Datalink Type */
Packit 34410b
} __attribute__ ((packed));
Packit 34410b
#define BTSNOOP_HDR_SIZE (sizeof(struct btsnoop_hdr))
Packit 34410b
Packit 34410b
struct btsnoop_pkt {
Packit 34410b
	uint32_t	size;		/* Original Length */
Packit 34410b
	uint32_t	len;		/* Included Length */
Packit 34410b
	uint32_t	flags;		/* Packet Flags */
Packit 34410b
	uint32_t	drops;		/* Cumulative Drops */
Packit 34410b
	uint64_t	ts;		/* Timestamp microseconds */
Packit 34410b
	uint8_t		data[0];	/* Packet Data */
Packit 34410b
} __attribute__ ((packed));
Packit 34410b
#define BTSNOOP_PKT_SIZE (sizeof(struct btsnoop_pkt))
Packit 34410b
Packit 34410b
static const uint8_t btsnoop_id[] = { 0x62, 0x74, 0x73, 0x6e,
Packit 34410b
				      0x6f, 0x6f, 0x70, 0x00 };
Packit 34410b
Packit 34410b
static const uint32_t btsnoop_version = 1;
Packit 34410b
Packit 34410b
struct pklg_pkt {
Packit 34410b
	uint32_t	len;
Packit 34410b
	uint64_t	ts;
Packit 34410b
	uint8_t		type;
Packit 34410b
} __attribute__ ((packed));
Packit 34410b
#define PKLG_PKT_SIZE (sizeof(struct pklg_pkt))
Packit 34410b
Packit 34410b
struct btsnoop {
Packit 34410b
	int ref_count;
Packit 34410b
	int fd;
Packit 34410b
	unsigned long flags;
Packit 34410b
	uint32_t format;
Packit 34410b
	uint16_t index;
Packit 34410b
	bool aborted;
Packit 34410b
	bool pklg_format;
Packit 34410b
	bool pklg_v2;
Packit 34410b
	const char *path;
Packit 34410b
	size_t max_size;
Packit 34410b
	size_t cur_size;
Packit 34410b
	unsigned int max_count;
Packit 34410b
	unsigned int cur_count;
Packit 34410b
};
Packit 34410b
Packit 34410b
struct btsnoop *btsnoop_open(const char *path, unsigned long flags)
Packit 34410b
{
Packit 34410b
	struct btsnoop *btsnoop;
Packit 34410b
	struct btsnoop_hdr hdr;
Packit 34410b
	ssize_t len;
Packit 34410b
Packit 34410b
	btsnoop = calloc(1, sizeof(*btsnoop));
Packit 34410b
	if (!btsnoop)
Packit 34410b
		return NULL;
Packit 34410b
Packit 34410b
	btsnoop->fd = open(path, O_RDONLY | O_CLOEXEC);
Packit 34410b
	if (btsnoop->fd < 0) {
Packit 34410b
		free(btsnoop);
Packit 34410b
		return NULL;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	btsnoop->flags = flags;
Packit 34410b
Packit 34410b
	len = read(btsnoop->fd, &hdr, BTSNOOP_HDR_SIZE);
Packit 34410b
	if (len < 0 || len != BTSNOOP_HDR_SIZE)
Packit 34410b
		goto failed;
Packit 34410b
Packit 34410b
	if (!memcmp(hdr.id, btsnoop_id, sizeof(btsnoop_id))) {
Packit 34410b
		/* Check for BTSnoop version 1 format */
Packit 34410b
		if (be32toh(hdr.version) != btsnoop_version)
Packit 34410b
			goto failed;
Packit 34410b
Packit 34410b
		btsnoop->format = be32toh(hdr.type);
Packit 34410b
		btsnoop->index = 0xffff;
Packit 34410b
	} else {
Packit 34410b
		if (!(btsnoop->flags & BTSNOOP_FLAG_PKLG_SUPPORT))
Packit 34410b
			goto failed;
Packit 34410b
Packit 34410b
		/* Check for Apple Packet Logger format */
Packit 34410b
		if (hdr.id[0] != 0x00 ||
Packit 34410b
				(hdr.id[1] != 0x00 && hdr.id[1] != 0x01))
Packit 34410b
			goto failed;
Packit 34410b
Packit 34410b
		btsnoop->format = BTSNOOP_FORMAT_MONITOR;
Packit 34410b
		btsnoop->index = 0xffff;
Packit 34410b
		btsnoop->pklg_format = true;
Packit 34410b
		btsnoop->pklg_v2 = (hdr.id[1] == 0x01);
Packit 34410b
Packit 34410b
		/* Apple Packet Logger format has no header */
Packit 34410b
		lseek(btsnoop->fd, 0, SEEK_SET);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return btsnoop_ref(btsnoop);
Packit 34410b
Packit 34410b
failed:
Packit 34410b
	close(btsnoop->fd);
Packit 34410b
	free(btsnoop);
Packit 34410b
Packit 34410b
	return NULL;
Packit 34410b
}
Packit 34410b
Packit 34410b
struct btsnoop *btsnoop_create(const char *path, size_t max_size,
Packit 34410b
					unsigned int max_count, uint32_t format)
Packit 34410b
{
Packit 34410b
	struct btsnoop *btsnoop;
Packit 34410b
	struct btsnoop_hdr hdr;
Packit 34410b
	const char *real_path;
Packit 34410b
	char tmp[PATH_MAX];
Packit 34410b
	ssize_t written;
Packit 34410b
Packit 34410b
	if (!max_size && max_count)
Packit 34410b
		return NULL;
Packit 34410b
Packit 34410b
	btsnoop = calloc(1, sizeof(*btsnoop));
Packit 34410b
	if (!btsnoop)
Packit 34410b
		return NULL;
Packit 34410b
Packit 34410b
	/* If max file size is specified, always add counter to file path */
Packit 34410b
	if (max_size) {
Packit 34410b
		snprintf(tmp, PATH_MAX, "%s.0", path);
Packit 34410b
		real_path = tmp;
Packit 34410b
	} else {
Packit 34410b
		real_path = path;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	btsnoop->fd = open(real_path, O_WRONLY | O_CREAT | O_TRUNC | O_CLOEXEC,
Packit 34410b
					S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
Packit 34410b
	if (btsnoop->fd < 0) {
Packit 34410b
		free(btsnoop);
Packit 34410b
		return NULL;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	btsnoop->format = format;
Packit 34410b
	btsnoop->index = 0xffff;
Packit 34410b
	btsnoop->path = path;
Packit 34410b
	btsnoop->max_count = max_count;
Packit 34410b
	btsnoop->max_size = max_size;
Packit 34410b
Packit 34410b
	memcpy(hdr.id, btsnoop_id, sizeof(btsnoop_id));
Packit 34410b
	hdr.version = htobe32(btsnoop_version);
Packit 34410b
	hdr.type = htobe32(btsnoop->format);
Packit 34410b
Packit 34410b
	written = write(btsnoop->fd, &hdr, BTSNOOP_HDR_SIZE);
Packit 34410b
	if (written < 0) {
Packit 34410b
		close(btsnoop->fd);
Packit 34410b
		free(btsnoop);
Packit 34410b
		return NULL;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	btsnoop->cur_size = BTSNOOP_HDR_SIZE;
Packit 34410b
Packit 34410b
	return btsnoop_ref(btsnoop);
Packit 34410b
}
Packit 34410b
Packit 34410b
struct btsnoop *btsnoop_ref(struct btsnoop *btsnoop)
Packit 34410b
{
Packit 34410b
	if (!btsnoop)
Packit 34410b
		return NULL;
Packit 34410b
Packit 34410b
	__sync_fetch_and_add(&btsnoop->ref_count, 1);
Packit 34410b
Packit 34410b
	return btsnoop;
Packit 34410b
}
Packit 34410b
Packit 34410b
void btsnoop_unref(struct btsnoop *btsnoop)
Packit 34410b
{
Packit 34410b
	if (!btsnoop)
Packit 34410b
		return;
Packit 34410b
Packit 34410b
	if (__sync_sub_and_fetch(&btsnoop->ref_count, 1))
Packit 34410b
		return;
Packit 34410b
Packit 34410b
	if (btsnoop->fd >= 0)
Packit 34410b
		close(btsnoop->fd);
Packit 34410b
Packit 34410b
	free(btsnoop);
Packit 34410b
}
Packit 34410b
Packit 34410b
uint32_t btsnoop_get_format(struct btsnoop *btsnoop)
Packit 34410b
{
Packit 34410b
	if (!btsnoop)
Packit 34410b
		return BTSNOOP_FORMAT_INVALID;
Packit 34410b
Packit 34410b
	return btsnoop->format;
Packit 34410b
}
Packit 34410b
Packit 34410b
static bool btsnoop_rotate(struct btsnoop *btsnoop)
Packit 34410b
{
Packit 34410b
	struct btsnoop_hdr hdr;
Packit 34410b
	char path[PATH_MAX];
Packit 34410b
	ssize_t written;
Packit 34410b
Packit 34410b
	close(btsnoop->fd);
Packit 34410b
Packit 34410b
	/* Check if max number of log files has been reached */
Packit 34410b
	if (btsnoop->max_count && btsnoop->cur_count >= btsnoop->max_count) {
Packit 34410b
		snprintf(path, PATH_MAX, "%s.%u", btsnoop->path,
Packit 34410b
				btsnoop->cur_count - btsnoop->max_count);
Packit 34410b
		unlink(path);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	snprintf(path, PATH_MAX,"%s.%u", btsnoop->path, btsnoop->cur_count);
Packit 34410b
	btsnoop->cur_count++;
Packit 34410b
Packit 34410b
	btsnoop->fd = open(path, O_WRONLY | O_CREAT | O_TRUNC | O_CLOEXEC,
Packit 34410b
					S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
Packit 34410b
	if (btsnoop->fd < 0)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	memcpy(hdr.id, btsnoop_id, sizeof(btsnoop_id));
Packit 34410b
	hdr.version = htobe32(btsnoop_version);
Packit 34410b
	hdr.type = htobe32(btsnoop->format);
Packit 34410b
Packit 34410b
	written = write(btsnoop->fd, &hdr, BTSNOOP_HDR_SIZE);
Packit 34410b
	if (written < 0)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	btsnoop->cur_size = BTSNOOP_HDR_SIZE;
Packit 34410b
Packit 34410b
	return true;
Packit 34410b
}
Packit 34410b
Packit 34410b
bool btsnoop_write(struct btsnoop *btsnoop, struct timeval *tv,
Packit 34410b
			uint32_t flags, uint32_t drops, const void *data,
Packit 34410b
			uint16_t size)
Packit 34410b
{
Packit 34410b
	struct btsnoop_pkt pkt;
Packit 34410b
	uint64_t ts;
Packit 34410b
	ssize_t written;
Packit 34410b
Packit 34410b
	if (!btsnoop || !tv)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	if (btsnoop->max_size && btsnoop->max_size <=
Packit 34410b
			btsnoop->cur_size + size + BTSNOOP_PKT_SIZE)
Packit 34410b
		if (!btsnoop_rotate(btsnoop))
Packit 34410b
			return false;
Packit 34410b
Packit 34410b
	ts = (tv->tv_sec - 946684800ll) * 1000000ll + tv->tv_usec;
Packit 34410b
Packit 34410b
	pkt.size  = htobe32(size);
Packit 34410b
	pkt.len   = htobe32(size);
Packit 34410b
	pkt.flags = htobe32(flags);
Packit 34410b
	pkt.drops = htobe32(drops);
Packit 34410b
	pkt.ts    = htobe64(ts + 0x00E03AB44A676000ll);
Packit 34410b
Packit 34410b
	written = write(btsnoop->fd, &pkt, BTSNOOP_PKT_SIZE);
Packit 34410b
	if (written < 0)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	btsnoop->cur_size += BTSNOOP_PKT_SIZE;
Packit 34410b
Packit 34410b
	if (data && size > 0) {
Packit 34410b
		written = write(btsnoop->fd, data, size);
Packit 34410b
		if (written < 0)
Packit 34410b
			return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	btsnoop->cur_size += size;
Packit 34410b
Packit 34410b
	return true;
Packit 34410b
}
Packit 34410b
Packit 34410b
static uint32_t get_flags_from_opcode(uint16_t opcode)
Packit 34410b
{
Packit 34410b
	switch (opcode) {
Packit 34410b
	case BTSNOOP_OPCODE_NEW_INDEX:
Packit 34410b
	case BTSNOOP_OPCODE_DEL_INDEX:
Packit 34410b
		break;
Packit 34410b
	case BTSNOOP_OPCODE_COMMAND_PKT:
Packit 34410b
		return 0x02;
Packit 34410b
	case BTSNOOP_OPCODE_EVENT_PKT:
Packit 34410b
		return 0x03;
Packit 34410b
	case BTSNOOP_OPCODE_ACL_TX_PKT:
Packit 34410b
		return 0x00;
Packit 34410b
	case BTSNOOP_OPCODE_ACL_RX_PKT:
Packit 34410b
		return 0x01;
Packit 34410b
	case BTSNOOP_OPCODE_SCO_TX_PKT:
Packit 34410b
	case BTSNOOP_OPCODE_SCO_RX_PKT:
Packit 34410b
		break;
Packit 34410b
	case BTSNOOP_OPCODE_OPEN_INDEX:
Packit 34410b
	case BTSNOOP_OPCODE_CLOSE_INDEX:
Packit 34410b
		break;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return 0xff;
Packit 34410b
}
Packit 34410b
Packit 34410b
bool btsnoop_write_hci(struct btsnoop *btsnoop, struct timeval *tv,
Packit 34410b
			uint16_t index, uint16_t opcode, uint32_t drops,
Packit 34410b
			const void *data, uint16_t size)
Packit 34410b
{
Packit 34410b
	uint32_t flags;
Packit 34410b
Packit 34410b
	if (!btsnoop)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	switch (btsnoop->format) {
Packit 34410b
	case BTSNOOP_FORMAT_HCI:
Packit 34410b
		if (btsnoop->index == 0xffff)
Packit 34410b
			btsnoop->index = index;
Packit 34410b
Packit 34410b
		if (index != btsnoop->index)
Packit 34410b
			return false;
Packit 34410b
Packit 34410b
		flags = get_flags_from_opcode(opcode);
Packit 34410b
		if (flags == 0xff)
Packit 34410b
			return false;
Packit 34410b
		break;
Packit 34410b
Packit 34410b
	case BTSNOOP_FORMAT_MONITOR:
Packit 34410b
		flags = (index << 16) | opcode;
Packit 34410b
		break;
Packit 34410b
Packit 34410b
	default:
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return btsnoop_write(btsnoop, tv, flags, drops, data, size);
Packit 34410b
}
Packit 34410b
Packit 34410b
bool btsnoop_write_phy(struct btsnoop *btsnoop, struct timeval *tv,
Packit 34410b
			uint16_t frequency, const void *data, uint16_t size)
Packit 34410b
{
Packit 34410b
	uint32_t flags;
Packit 34410b
Packit 34410b
	if (!btsnoop)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	switch (btsnoop->format) {
Packit 34410b
	case BTSNOOP_FORMAT_SIMULATOR:
Packit 34410b
		flags = (1 << 16) | frequency;
Packit 34410b
		break;
Packit 34410b
Packit 34410b
	default:
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return btsnoop_write(btsnoop, tv, flags, 0, data, size);
Packit 34410b
}
Packit 34410b
Packit 34410b
static bool pklg_read_hci(struct btsnoop *btsnoop, struct timeval *tv,
Packit 34410b
					uint16_t *index, uint16_t *opcode,
Packit 34410b
					void *data, uint16_t *size)
Packit 34410b
{
Packit 34410b
	struct pklg_pkt pkt;
Packit 34410b
	uint32_t toread;
Packit 34410b
	uint64_t ts;
Packit 34410b
	ssize_t len;
Packit 34410b
Packit 34410b
	len = read(btsnoop->fd, &pkt, PKLG_PKT_SIZE);
Packit 34410b
	if (len == 0)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	if (len < 0 || len != PKLG_PKT_SIZE) {
Packit 34410b
		btsnoop->aborted = true;
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	if (btsnoop->pklg_v2) {
Packit 34410b
		toread = le32toh(pkt.len) - (PKLG_PKT_SIZE - 4);
Packit 34410b
Packit 34410b
		ts = le64toh(pkt.ts);
Packit 34410b
		tv->tv_sec = ts & 0xffffffff;
Packit 34410b
		tv->tv_usec = ts >> 32;
Packit 34410b
	} else {
Packit 34410b
		toread = be32toh(pkt.len) - (PKLG_PKT_SIZE - 4);
Packit 34410b
Packit 34410b
		ts = be64toh(pkt.ts);
Packit 34410b
		tv->tv_sec = ts >> 32;
Packit 34410b
		tv->tv_usec = ts & 0xffffffff;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	if (toread > BTSNOOP_MAX_PACKET_SIZE) {
Packit 34410b
                btsnoop->aborted = true;
Packit 34410b
                return false;
Packit 34410b
        }
Packit 34410b
Packit 34410b
	switch (pkt.type) {
Packit 34410b
	case 0x00:
Packit 34410b
		*index = 0x0000;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_COMMAND_PKT;
Packit 34410b
		break;
Packit 34410b
	case 0x01:
Packit 34410b
		*index = 0x0000;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_EVENT_PKT;
Packit 34410b
		break;
Packit 34410b
	case 0x02:
Packit 34410b
		*index = 0x0000;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_ACL_TX_PKT;
Packit 34410b
		break;
Packit 34410b
	case 0x03:
Packit 34410b
		*index = 0x0000;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_ACL_RX_PKT;
Packit 34410b
		break;
Packit 34410b
	case 0x08:
Packit 34410b
		*index = 0x0000;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_SCO_TX_PKT;
Packit 34410b
		break;
Packit 34410b
	case 0x09:
Packit 34410b
		*index = 0x0000;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_SCO_RX_PKT;
Packit 34410b
		break;
Packit 34410b
	case 0x0b:
Packit 34410b
		*index = 0x0000;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_VENDOR_DIAG;
Packit 34410b
		break;
Packit 34410b
	case 0xfc:
Packit 34410b
		*index = 0xffff;
Packit 34410b
		*opcode = BTSNOOP_OPCODE_SYSTEM_NOTE;
Packit 34410b
		break;
Packit 34410b
	default:
Packit 34410b
		*index = 0xffff;
Packit 34410b
		*opcode = 0xffff;
Packit 34410b
		break;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	len = read(btsnoop->fd, data, toread);
Packit 34410b
	if (len < 0) {
Packit 34410b
		btsnoop->aborted = true;
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	*size = toread;
Packit 34410b
Packit 34410b
	return true;
Packit 34410b
}
Packit 34410b
Packit 34410b
static uint16_t get_opcode_from_flags(uint8_t type, uint32_t flags)
Packit 34410b
{
Packit 34410b
	switch (type) {
Packit 34410b
	case 0x01:
Packit 34410b
		return BTSNOOP_OPCODE_COMMAND_PKT;
Packit 34410b
	case 0x02:
Packit 34410b
		if (flags & 0x01)
Packit 34410b
			return BTSNOOP_OPCODE_ACL_RX_PKT;
Packit 34410b
		else
Packit 34410b
			return BTSNOOP_OPCODE_ACL_TX_PKT;
Packit 34410b
	case 0x03:
Packit 34410b
		if (flags & 0x01)
Packit 34410b
			return BTSNOOP_OPCODE_SCO_RX_PKT;
Packit 34410b
		else
Packit 34410b
			return BTSNOOP_OPCODE_SCO_TX_PKT;
Packit 34410b
	case 0x04:
Packit 34410b
		return BTSNOOP_OPCODE_EVENT_PKT;
Packit 34410b
	case 0xff:
Packit 34410b
		if (flags & 0x02) {
Packit 34410b
			if (flags & 0x01)
Packit 34410b
				return BTSNOOP_OPCODE_EVENT_PKT;
Packit 34410b
			else
Packit 34410b
				return BTSNOOP_OPCODE_COMMAND_PKT;
Packit 34410b
		} else {
Packit 34410b
			if (flags & 0x01)
Packit 34410b
				return BTSNOOP_OPCODE_ACL_RX_PKT;
Packit 34410b
			else
Packit 34410b
				return BTSNOOP_OPCODE_ACL_TX_PKT;
Packit 34410b
		}
Packit 34410b
		break;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return 0xffff;
Packit 34410b
}
Packit 34410b
Packit 34410b
bool btsnoop_read_hci(struct btsnoop *btsnoop, struct timeval *tv,
Packit 34410b
					uint16_t *index, uint16_t *opcode,
Packit 34410b
					void *data, uint16_t *size)
Packit 34410b
{
Packit 34410b
	struct btsnoop_pkt pkt;
Packit 34410b
	uint32_t toread, flags;
Packit 34410b
	uint64_t ts;
Packit 34410b
	uint8_t pkt_type;
Packit 34410b
	ssize_t len;
Packit 34410b
Packit 34410b
	if (!btsnoop || btsnoop->aborted)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	if (btsnoop->pklg_format)
Packit 34410b
		return pklg_read_hci(btsnoop, tv, index, opcode, data, size);
Packit 34410b
Packit 34410b
	len = read(btsnoop->fd, &pkt, BTSNOOP_PKT_SIZE);
Packit 34410b
	if (len == 0)
Packit 34410b
		return false;
Packit 34410b
Packit 34410b
	if (len < 0 || len != BTSNOOP_PKT_SIZE) {
Packit 34410b
		btsnoop->aborted = true;
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	toread = be32toh(pkt.size);
Packit 34410b
	if (toread > BTSNOOP_MAX_PACKET_SIZE) {
Packit 34410b
		btsnoop->aborted = true;
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	flags = be32toh(pkt.flags);
Packit 34410b
Packit 34410b
	ts = be64toh(pkt.ts) - 0x00E03AB44A676000ll;
Packit 34410b
	tv->tv_sec = (ts / 1000000ll) + 946684800ll;
Packit 34410b
	tv->tv_usec = ts % 1000000ll;
Packit 34410b
Packit 34410b
	switch (btsnoop->format) {
Packit 34410b
	case BTSNOOP_FORMAT_HCI:
Packit 34410b
		*index = 0;
Packit 34410b
		*opcode = get_opcode_from_flags(0xff, flags);
Packit 34410b
		break;
Packit 34410b
Packit 34410b
	case BTSNOOP_FORMAT_UART:
Packit 34410b
		len = read(btsnoop->fd, &pkt_type, 1);
Packit 34410b
		if (len < 0) {
Packit 34410b
			btsnoop->aborted = true;
Packit 34410b
			return false;
Packit 34410b
		}
Packit 34410b
		toread--;
Packit 34410b
Packit 34410b
		*index = 0;
Packit 34410b
		*opcode = get_opcode_from_flags(pkt_type, flags);
Packit 34410b
		break;
Packit 34410b
Packit 34410b
	case BTSNOOP_FORMAT_MONITOR:
Packit 34410b
		*index = flags >> 16;
Packit 34410b
		*opcode = flags & 0xffff;
Packit 34410b
		break;
Packit 34410b
Packit 34410b
	default:
Packit 34410b
		btsnoop->aborted = true;
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	len = read(btsnoop->fd, data, toread);
Packit 34410b
	if (len < 0) {
Packit 34410b
		btsnoop->aborted = true;
Packit 34410b
		return false;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	*size = toread;
Packit 34410b
Packit 34410b
	return true;
Packit 34410b
}
Packit 34410b
Packit 34410b
bool btsnoop_read_phy(struct btsnoop *btsnoop, struct timeval *tv,
Packit 34410b
			uint16_t *frequency, void *data, uint16_t *size)
Packit 34410b
{
Packit 34410b
	return false;
Packit 34410b
}