Blame tools/parser/att.c

Packit 34410b
/*
Packit 34410b
 *
Packit 34410b
 *  BlueZ - Bluetooth protocol stack for Linux
Packit 34410b
 *
Packit 34410b
 *  Copyright (C) 2011  André Dieb Martins <andre.dieb@gmail.com>
Packit 34410b
 *
Packit 34410b
 *
Packit 34410b
 *  This program is free software; you can redistribute it and/or modify
Packit 34410b
 *  it under the terms of the GNU General Public License as published by
Packit 34410b
 *  the Free Software Foundation; either version 2 of the License, or
Packit 34410b
 *  (at your option) any later version.
Packit 34410b
 *
Packit 34410b
 *  This program 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
Packit 34410b
 *  GNU General Public License for more details.
Packit 34410b
 *
Packit 34410b
 *  You should have received a copy of the GNU General Public License
Packit 34410b
 *  along with this program; 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 <stdio.h>
Packit 34410b
#include <errno.h>
Packit 34410b
#include <unistd.h>
Packit 34410b
#include <stdlib.h>
Packit 34410b
#include <string.h>
Packit 34410b
Packit 34410b
#include "parser.h"
Packit 34410b
Packit 34410b
#define GATT_PRIM_SVC_UUID		0x2800
Packit 34410b
#define GATT_SND_SVC_UUID		0x2801
Packit 34410b
#define GATT_INCLUDE_UUID		0x2802
Packit 34410b
#define GATT_CHARAC_UUID		0x2803
Packit 34410b
Packit 34410b
#define GATT_CHARAC_DEVICE_NAME			0x2A00
Packit 34410b
#define GATT_CHARAC_APPEARANCE			0x2A01
Packit 34410b
#define GATT_CHARAC_PERIPHERAL_PRIV_FLAG	0x2A02
Packit 34410b
#define GATT_CHARAC_RECONNECTION_ADDRESS	0x2A03
Packit 34410b
#define GATT_CHARAC_PERIPHERAL_PREF_CONN	0x2A04
Packit 34410b
#define GATT_CHARAC_SERVICE_CHANGED		0x2A05
Packit 34410b
Packit 34410b
#define GATT_CHARAC_EXT_PROPER_UUID	0x2900
Packit 34410b
#define GATT_CHARAC_USER_DESC_UUID	0x2901
Packit 34410b
#define GATT_CLIENT_CHARAC_CFG_UUID	0x2902
Packit 34410b
#define GATT_SERVER_CHARAC_CFG_UUID	0x2903
Packit 34410b
#define GATT_CHARAC_FMT_UUID		0x2904
Packit 34410b
#define GATT_CHARAC_AGREG_FMT_UUID	0x2905
Packit 34410b
Packit 34410b
Packit 34410b
Packit 34410b
/* Attribute Protocol Opcodes */
Packit 34410b
#define ATT_OP_ERROR			0x01
Packit 34410b
#define ATT_OP_MTU_REQ			0x02
Packit 34410b
#define ATT_OP_MTU_RESP			0x03
Packit 34410b
#define ATT_OP_FIND_INFO_REQ		0x04
Packit 34410b
#define ATT_OP_FIND_INFO_RESP		0x05
Packit 34410b
#define ATT_OP_FIND_BY_TYPE_REQ		0x06
Packit 34410b
#define ATT_OP_FIND_BY_TYPE_RESP	0x07
Packit 34410b
#define ATT_OP_READ_BY_TYPE_REQ		0x08
Packit 34410b
#define ATT_OP_READ_BY_TYPE_RESP	0x09
Packit 34410b
#define ATT_OP_READ_REQ			0x0A
Packit 34410b
#define ATT_OP_READ_RESP		0x0B
Packit 34410b
#define ATT_OP_READ_BLOB_REQ		0x0C
Packit 34410b
#define ATT_OP_READ_BLOB_RESP		0x0D
Packit 34410b
#define ATT_OP_READ_MULTI_REQ		0x0E
Packit 34410b
#define ATT_OP_READ_MULTI_RESP		0x0F
Packit 34410b
#define ATT_OP_READ_BY_GROUP_REQ	0x10
Packit 34410b
#define ATT_OP_READ_BY_GROUP_RESP	0x11
Packit 34410b
#define ATT_OP_WRITE_REQ		0x12
Packit 34410b
#define ATT_OP_WRITE_RESP		0x13
Packit 34410b
#define ATT_OP_WRITE_CMD		0x52
Packit 34410b
#define ATT_OP_PREP_WRITE_REQ		0x16
Packit 34410b
#define ATT_OP_PREP_WRITE_RESP		0x17
Packit 34410b
#define ATT_OP_EXEC_WRITE_REQ		0x18
Packit 34410b
#define ATT_OP_EXEC_WRITE_RESP		0x19
Packit 34410b
#define ATT_OP_HANDLE_NOTIFY		0x1B
Packit 34410b
#define ATT_OP_HANDLE_IND		0x1D
Packit 34410b
#define ATT_OP_HANDLE_CNF		0x1E
Packit 34410b
#define ATT_OP_SIGNED_WRITE_CMD		0xD2
Packit 34410b
Packit 34410b
/* Error codes for Error response PDU */
Packit 34410b
#define ATT_ECODE_INVALID_HANDLE		0x01
Packit 34410b
#define ATT_ECODE_READ_NOT_PERM			0x02
Packit 34410b
#define ATT_ECODE_WRITE_NOT_PERM		0x03
Packit 34410b
#define ATT_ECODE_INVALID_PDU			0x04
Packit 34410b
#define ATT_ECODE_INSUFF_AUTHEN			0x05
Packit 34410b
#define ATT_ECODE_REQ_NOT_SUPP			0x06
Packit 34410b
#define ATT_ECODE_INVALID_OFFSET		0x07
Packit 34410b
#define ATT_ECODE_INSUFF_AUTHO			0x08
Packit 34410b
#define ATT_ECODE_PREP_QUEUE_FULL		0x09
Packit 34410b
#define ATT_ECODE_ATTR_NOT_FOUND		0x0A
Packit 34410b
#define ATT_ECODE_ATTR_NOT_LONG			0x0B
Packit 34410b
#define ATT_ECODE_INSUFF_ENCR_KEY_SIZE		0x0C
Packit 34410b
#define ATT_ECODE_INVAL_ATTR_VALUE_LEN		0x0D
Packit 34410b
#define ATT_ECODE_UNLIKELY			0x0E
Packit 34410b
#define ATT_ECODE_INSUFF_ENC			0x0F
Packit 34410b
#define ATT_ECODE_UNSUPP_GRP_TYPE		0x10
Packit 34410b
#define ATT_ECODE_INSUFF_RESOURCES		0x11
Packit 34410b
#define ATT_ECODE_IO				0xFF
Packit 34410b
Packit 34410b
Packit 34410b
/* Attribute Protocol Opcodes */
Packit 34410b
static const char *attop2str(uint8_t op)
Packit 34410b
{
Packit 34410b
	switch (op) {
Packit 34410b
	case ATT_OP_ERROR:
Packit 34410b
		return "Error";
Packit 34410b
	case ATT_OP_MTU_REQ:
Packit 34410b
		return "MTU req";
Packit 34410b
	case ATT_OP_MTU_RESP:
Packit 34410b
		return "MTU resp";
Packit 34410b
	case ATT_OP_FIND_INFO_REQ:
Packit 34410b
		return "Find Information req";
Packit 34410b
	case ATT_OP_FIND_INFO_RESP:
Packit 34410b
		return "Find Information resp";
Packit 34410b
	case ATT_OP_FIND_BY_TYPE_REQ:
Packit 34410b
		return "Find By Type req";
Packit 34410b
	case ATT_OP_FIND_BY_TYPE_RESP:
Packit 34410b
		return "Find By Type resp";
Packit 34410b
	case ATT_OP_READ_BY_TYPE_REQ:
Packit 34410b
		return "Read By Type req";
Packit 34410b
	case ATT_OP_READ_BY_TYPE_RESP:
Packit 34410b
		return "Read By Type resp";
Packit 34410b
	case ATT_OP_READ_REQ:
Packit 34410b
		return "Read req";
Packit 34410b
	case ATT_OP_READ_RESP:
Packit 34410b
		return "Read resp";
Packit 34410b
	case ATT_OP_READ_BLOB_REQ:
Packit 34410b
		return "Read Blob req";
Packit 34410b
	case ATT_OP_READ_BLOB_RESP:
Packit 34410b
		return "Read Blob resp";
Packit 34410b
	case ATT_OP_READ_MULTI_REQ:
Packit 34410b
		return "Read Multi req";
Packit 34410b
	case ATT_OP_READ_MULTI_RESP:
Packit 34410b
		return "Read Multi resp";
Packit 34410b
	case ATT_OP_READ_BY_GROUP_REQ:
Packit 34410b
		return "Read By Group req";
Packit 34410b
	case ATT_OP_READ_BY_GROUP_RESP:
Packit 34410b
		return "Read By Group resp";
Packit 34410b
	case ATT_OP_WRITE_REQ:
Packit 34410b
		return "Write req";
Packit 34410b
	case ATT_OP_WRITE_RESP:
Packit 34410b
		return "Write resp";
Packit 34410b
	case ATT_OP_WRITE_CMD:
Packit 34410b
		return "Write cmd";
Packit 34410b
	case ATT_OP_PREP_WRITE_REQ:
Packit 34410b
		return "Prepare Write req";
Packit 34410b
	case ATT_OP_PREP_WRITE_RESP:
Packit 34410b
		return "Prepare Write resp";
Packit 34410b
	case ATT_OP_EXEC_WRITE_REQ:
Packit 34410b
		return "Exec Write req";
Packit 34410b
	case ATT_OP_EXEC_WRITE_RESP:
Packit 34410b
		return "Exec Write resp";
Packit 34410b
	case ATT_OP_HANDLE_NOTIFY:
Packit 34410b
		return "Handle notify";
Packit 34410b
	case ATT_OP_HANDLE_IND:
Packit 34410b
		return "Handle indicate";
Packit 34410b
	case ATT_OP_HANDLE_CNF:
Packit 34410b
		return "Handle CNF";
Packit 34410b
	case ATT_OP_SIGNED_WRITE_CMD:
Packit 34410b
		return "Signed Write Cmd";
Packit 34410b
	default:
Packit 34410b
		return "Unknown";
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static const char * atterror2str(uint8_t err)
Packit 34410b
{
Packit 34410b
	switch (err) {
Packit 34410b
	case ATT_ECODE_INVALID_HANDLE:
Packit 34410b
		return "Invalid handle";
Packit 34410b
	case ATT_ECODE_READ_NOT_PERM:
Packit 34410b
		return "Read not permitted";
Packit 34410b
	case ATT_ECODE_WRITE_NOT_PERM:
Packit 34410b
		return "Write not permitted";
Packit 34410b
	case ATT_ECODE_INVALID_PDU:
Packit 34410b
		return "Invalid PDU";
Packit 34410b
	case ATT_ECODE_INSUFF_AUTHEN:
Packit 34410b
		return "Insufficient authentication";
Packit 34410b
	case ATT_ECODE_REQ_NOT_SUPP:
Packit 34410b
		return "Request not supported";
Packit 34410b
	case ATT_ECODE_INVALID_OFFSET:
Packit 34410b
		return "Invalid offset";
Packit 34410b
	case ATT_ECODE_INSUFF_AUTHO:
Packit 34410b
		return "Insufficient authorization";
Packit 34410b
	case ATT_ECODE_PREP_QUEUE_FULL:
Packit 34410b
		return "Prepare queue full";
Packit 34410b
	case ATT_ECODE_ATTR_NOT_FOUND:
Packit 34410b
		return "Attribute not found";
Packit 34410b
	case ATT_ECODE_ATTR_NOT_LONG:
Packit 34410b
		return "Attribute not long";
Packit 34410b
	case ATT_ECODE_INSUFF_ENCR_KEY_SIZE:
Packit 34410b
		return "Insufficient encryption key size";
Packit 34410b
	case ATT_ECODE_INVAL_ATTR_VALUE_LEN:
Packit 34410b
		return "Invalid attribute value length";
Packit 34410b
	case ATT_ECODE_UNLIKELY:
Packit 34410b
		return "Unlikely error";
Packit 34410b
	case ATT_ECODE_INSUFF_ENC:
Packit 34410b
		return "Insufficient encryption";
Packit 34410b
	case ATT_ECODE_UNSUPP_GRP_TYPE:
Packit 34410b
		return "Unsupported group type";
Packit 34410b
	case ATT_ECODE_INSUFF_RESOURCES:
Packit 34410b
		return "Insufficient resources";
Packit 34410b
	case ATT_ECODE_IO:
Packit 34410b
		return "Application Error";
Packit 34410b
	default:
Packit 34410b
		return "Reserved";
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static const char *uuid2str(uint16_t uuid)
Packit 34410b
{
Packit 34410b
	switch (uuid) {
Packit 34410b
	case GATT_PRIM_SVC_UUID:
Packit 34410b
		return "GATT Primary Service";
Packit 34410b
	case GATT_SND_SVC_UUID:
Packit 34410b
		return "GATT Secondary Service";
Packit 34410b
	case GATT_INCLUDE_UUID:
Packit 34410b
		return "GATT Include";
Packit 34410b
	case GATT_CHARAC_UUID:
Packit 34410b
		return "GATT Characteristic";
Packit 34410b
	case GATT_CHARAC_DEVICE_NAME:
Packit 34410b
		return "GATT(type) Device Name";
Packit 34410b
	case GATT_CHARAC_APPEARANCE:
Packit 34410b
		return "GATT(type) Appearance";
Packit 34410b
	case GATT_CHARAC_PERIPHERAL_PRIV_FLAG:
Packit 34410b
		return "GATT(type) Peripheral Privacy Flag";
Packit 34410b
	case GATT_CHARAC_RECONNECTION_ADDRESS:
Packit 34410b
		return "GATT(type) Characteristic Reconnection Address";
Packit 34410b
	case GATT_CHARAC_PERIPHERAL_PREF_CONN:
Packit 34410b
		return "GATT(type) Characteristic Preferred Connection Parameters";
Packit 34410b
	case GATT_CHARAC_SERVICE_CHANGED:
Packit 34410b
		return "GATT(type) Characteristic Service Changed";
Packit 34410b
	case GATT_CHARAC_EXT_PROPER_UUID:
Packit 34410b
		return "GATT(desc) Characteristic Extended Properties";
Packit 34410b
	case GATT_CHARAC_USER_DESC_UUID:
Packit 34410b
		return "GATT(desc) User Description";
Packit 34410b
	case GATT_CLIENT_CHARAC_CFG_UUID:
Packit 34410b
		return "GATT(desc) Client Characteristic Configuration";
Packit 34410b
	case GATT_SERVER_CHARAC_CFG_UUID:
Packit 34410b
		return "GATT(desc) Server Characteristic Configuration";
Packit 34410b
	case GATT_CHARAC_FMT_UUID:
Packit 34410b
		return "GATT(desc) Format";
Packit 34410b
	case GATT_CHARAC_AGREG_FMT_UUID:
Packit 34410b
		return "GATT(desc) Aggregate Format";
Packit 34410b
	default:
Packit 34410b
		return "Unknown";
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_error_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint8_t op = p_get_u8(frm);
Packit 34410b
	uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
	uint8_t err = p_get_u8(frm);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("Error: %s (%d)\n", atterror2str(err), err);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("%s (0x%.2x) on handle 0x%4.4x\n", attop2str(op), op, handle);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_mtu_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t client_rx_mtu = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("client rx mtu %d\n", client_rx_mtu);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_mtu_resp_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t server_rx_mtu = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("server rx mtu %d\n", server_rx_mtu);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_find_info_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t start = btohs(htons(p_get_u16(frm)));
Packit 34410b
	uint16_t end = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("start 0x%4.4x, end 0x%4.4x\n", start, end);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void print_uuid128(struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint8_t uuid[16];
Packit 34410b
	int i;
Packit 34410b
Packit 34410b
	for (i = 0; i < 16; i++)
Packit 34410b
		uuid[15 - i] = p_get_u8(frm);
Packit 34410b
Packit 34410b
	for (i = 0; i < 16; i++) {
Packit 34410b
		printf("%02x", uuid[i]);
Packit 34410b
		if (i == 3 || i == 5 || i == 7 || i == 9)
Packit 34410b
			printf("-");
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_find_info_resp_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint8_t fmt = p_get_u8(frm);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
Packit 34410b
	if (fmt == 0x01) {
Packit 34410b
		printf("format: uuid-16\n");
Packit 34410b
Packit 34410b
		while (frm->len > 0) {
Packit 34410b
			uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
			uint16_t uuid = btohs(htons(p_get_u16(frm)));
Packit 34410b
			p_indent(level + 1, frm);
Packit 34410b
			printf("handle 0x%4.4x, uuid 0x%4.4x (%s)\n", handle, uuid,
Packit 34410b
					uuid2str(uuid));
Packit 34410b
		}
Packit 34410b
	} else {
Packit 34410b
		printf("format: uuid-128\n");
Packit 34410b
Packit 34410b
		while (frm->len > 0) {
Packit 34410b
			uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
			p_indent(level + 1, frm);
Packit 34410b
			printf("handle 0x%4.4x, uuid ", handle);
Packit 34410b
			print_uuid128(frm);
Packit 34410b
			printf("\n");
Packit 34410b
		}
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_find_by_type_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t start = btohs(htons(p_get_u16(frm)));
Packit 34410b
	uint16_t end = btohs(htons(p_get_u16(frm)));
Packit 34410b
	uint16_t uuid = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("start 0x%4.4x, end 0x%4.4x, uuid 0x%4.4x\n", start, end, uuid);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("value");
Packit 34410b
	while (frm->len > 0)
Packit 34410b
		printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_find_by_type_resp_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	while (frm->len > 0) {
Packit 34410b
		uint16_t uuid = btohs(htons(p_get_u16(frm)));
Packit 34410b
		uint16_t end = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
		p_indent(level, frm);
Packit 34410b
		printf("Found attr 0x%4.4x, group end handle 0x%4.4x\n",
Packit 34410b
								uuid, end);
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_by_type_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t start = btohs(htons(p_get_u16(frm)));
Packit 34410b
	uint16_t end = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("start 0x%4.4x, end 0x%4.4x\n", start, end);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	if (frm->len == 2) {
Packit 34410b
		printf("type-uuid 0x%4.4x\n", btohs(htons(p_get_u16(frm))));
Packit 34410b
	} else if (frm->len == 16) {
Packit 34410b
		printf("type-uuid ");
Packit 34410b
		print_uuid128(frm);
Packit 34410b
		printf("\n");
Packit 34410b
	} else {
Packit 34410b
		printf("malformed uuid (expected 2 or 16 octets)\n");
Packit 34410b
		p_indent(level, frm);
Packit 34410b
		raw_dump(level, frm);
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_by_type_resp_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint8_t length = p_get_u8(frm);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("length: %d\n", length);
Packit 34410b
Packit 34410b
	while (frm->len > 0) {
Packit 34410b
		uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
		int val_len = length - 2;
Packit 34410b
		int i;
Packit 34410b
Packit 34410b
		p_indent(level + 1, frm);
Packit 34410b
		printf("handle 0x%4.4x, value ", handle);
Packit 34410b
		for (i = 0; i < val_len; i++) {
Packit 34410b
			printf("0x%.2x ", p_get_u8(frm));
Packit 34410b
		}
Packit 34410b
		printf("\n");
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("handle 0x%4.4x\n", handle);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_blob_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
	uint16_t offset = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("handle 0x%4.4x offset 0x%4.4x\n", handle, offset);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_blob_resp_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("value");
Packit 34410b
Packit 34410b
	while (frm->len > 0)
Packit 34410b
		printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_multi_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("Handles\n");
Packit 34410b
Packit 34410b
	while (frm->len > 0) {
Packit 34410b
		p_indent(level, frm);
Packit 34410b
		printf("handle 0x%4.4x\n", btohs(htons(p_get_u16(frm))));
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_multi_resp_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("values");
Packit 34410b
Packit 34410b
	while (frm->len > 0)
Packit 34410b
		printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_read_by_group_resp_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint8_t length = p_get_u8(frm);
Packit 34410b
Packit 34410b
	while (frm->len > 0) {
Packit 34410b
		uint16_t attr_handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
		uint16_t end_grp_handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
		uint8_t remaining = length - 4;
Packit 34410b
Packit 34410b
		p_indent(level, frm);
Packit 34410b
		printf("attr handle 0x%4.4x, end group handle 0x%4.4x\n",
Packit 34410b
						attr_handle, end_grp_handle);
Packit 34410b
Packit 34410b
		p_indent(level, frm);
Packit 34410b
		printf("value");
Packit 34410b
		while (remaining > 0) {
Packit 34410b
			printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
			remaining--;
Packit 34410b
		}
Packit 34410b
		printf("\n");
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_write_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("handle 0x%4.4x value ", handle);
Packit 34410b
Packit 34410b
	while (frm->len > 0)
Packit 34410b
		printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_signed_write_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
	int value_len = frm->len - 12; /* handle:2 already accounted, sig: 12 */
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("handle 0x%4.4x value ", handle);
Packit 34410b
Packit 34410b
	while (value_len--)
Packit 34410b
		printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("auth signature ");
Packit 34410b
	while (frm->len > 0)
Packit 34410b
		printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_prep_write_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
	uint16_t val_offset = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("attr handle 0x%4.4x, value offset 0x%4.4x\n", handle,
Packit 34410b
								val_offset);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("part attr value ");
Packit 34410b
	while (frm->len > 0)
Packit 34410b
		printf(" 0x%2.2x", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_exec_write_req_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint8_t flags = p_get_u8(frm);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	if (flags == 0x00)
Packit 34410b
		printf("cancel all prepared writes ");
Packit 34410b
	else
Packit 34410b
		printf("immediatelly write all pending prepared values ");
Packit 34410b
Packit 34410b
	printf("(0x%2.2x)\n", flags);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void att_handle_notify_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint16_t handle = btohs(htons(p_get_u16(frm)));
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("handle 0x%4.4x\n", handle);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("value ");
Packit 34410b
	while (frm->len > 0)
Packit 34410b
		printf("0x%.2x ", p_get_u8(frm));
Packit 34410b
	printf("\n");
Packit 34410b
}
Packit 34410b
Packit 34410b
void att_dump(int level, struct frame *frm)
Packit 34410b
{
Packit 34410b
	uint8_t op;
Packit 34410b
Packit 34410b
	op = p_get_u8(frm);
Packit 34410b
Packit 34410b
	p_indent(level, frm);
Packit 34410b
	printf("ATT: %s (0x%.2x)\n", attop2str(op), op);
Packit 34410b
Packit 34410b
	switch (op) {
Packit 34410b
		case ATT_OP_ERROR:
Packit 34410b
			att_error_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_MTU_REQ:
Packit 34410b
			att_mtu_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_MTU_RESP:
Packit 34410b
			att_mtu_resp_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_FIND_INFO_REQ:
Packit 34410b
			att_find_info_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_FIND_INFO_RESP:
Packit 34410b
			att_find_info_resp_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_FIND_BY_TYPE_REQ:
Packit 34410b
			att_find_by_type_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_FIND_BY_TYPE_RESP:
Packit 34410b
			att_find_by_type_resp_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_BY_TYPE_REQ:
Packit 34410b
		case ATT_OP_READ_BY_GROUP_REQ: /* exact same parsing */
Packit 34410b
			att_read_by_type_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_BY_TYPE_RESP:
Packit 34410b
			att_read_by_type_resp_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_REQ:
Packit 34410b
			att_read_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_RESP:
Packit 34410b
			raw_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_BLOB_REQ:
Packit 34410b
			att_read_blob_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_BLOB_RESP:
Packit 34410b
			att_read_blob_resp_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_MULTI_REQ:
Packit 34410b
			att_read_multi_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_MULTI_RESP:
Packit 34410b
			att_read_multi_resp_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_READ_BY_GROUP_RESP:
Packit 34410b
			att_read_by_group_resp_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_WRITE_REQ:
Packit 34410b
		case ATT_OP_WRITE_CMD:
Packit 34410b
			att_write_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_SIGNED_WRITE_CMD:
Packit 34410b
			att_signed_write_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_PREP_WRITE_REQ:
Packit 34410b
		case ATT_OP_PREP_WRITE_RESP:
Packit 34410b
			att_prep_write_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_EXEC_WRITE_REQ:
Packit 34410b
			att_exec_write_req_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		case ATT_OP_HANDLE_NOTIFY:
Packit 34410b
			att_handle_notify_dump(level + 1, frm);
Packit 34410b
			break;
Packit 34410b
		default:
Packit 34410b
			raw_dump(level, frm);
Packit 34410b
			break;
Packit 34410b
	}
Packit 34410b
}