📄 obex.c
字号:
/* * * Bluetooth packet analyzer - OBEX parser * * Copyright (C) 2004-2005 Marcel Holtmann <marcel@holtmann.org> * * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * * $Id: obex.c,v 1.7 2005/05/15 20:55:15 holtmann Exp $ */#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <errno.h>#include <string.h>#include <sys/types.h>#include <netinet/in.h>#include "parser.h"#define TABLE_SIZE 20static struct { uint16_t handle; uint8_t dlci; uint8_t opcode; uint8_t status; struct frame frm;} table[TABLE_SIZE];static void del_frame(uint16_t handle, uint8_t dlci){ int i; for (i = 0; i < TABLE_SIZE; i++) if (table[i].handle == handle && table[i].dlci == dlci) { table[i].handle = 0; table[i].dlci = 0; table[i].opcode = 0; table[i].status = 0; if (table[i].frm.data) free(table[i].frm.data); memset(&table[i].frm, 0, sizeof(struct frame)); break; }}static struct frame *add_frame(struct frame *frm){ struct frame *fr; void *data; int i, pos = -1; for (i = 0; i < TABLE_SIZE; i++) { if (table[i].handle == frm->handle && table[i].dlci == frm->dlci) { pos = i; break; } if (pos < 0 && !table[i].handle && !table[i].dlci) pos = i; } if (pos < 0) return frm; table[pos].handle = frm->handle; table[pos].dlci = frm->dlci; fr = &table[pos].frm; data = malloc(fr->len + frm->len); if (!data) { perror("Can't allocate OBEX stream buffer"); del_frame(frm->handle, frm->dlci); return frm; } if (fr->len > 0) memcpy(data, fr->ptr, fr->len); if (frm->len > 0) memcpy(data + fr->len, frm->ptr, frm->len); if (fr->data) free(fr->data); fr->data = data; fr->data_len = fr->len + frm->len; fr->len = fr->data_len; fr->ptr = fr->data; fr->in = frm->in; fr->ts = frm->ts; fr->handle = frm->handle; fr->cid = frm->cid; fr->num = frm->num; fr->dlci = frm->dlci; fr->channel = frm->channel; return fr;}static uint8_t get_opcode(uint16_t handle, uint8_t dlci){ int i; for (i = 0; i < TABLE_SIZE; i++) if (table[i].handle == handle && table[i].dlci == dlci) return table[i].opcode; return 0x00;}static void set_opcode(uint16_t handle, uint8_t dlci, uint8_t opcode){ int i; for (i = 0; i < TABLE_SIZE; i++) if (table[i].handle == handle && table[i].dlci == dlci) { table[i].opcode = opcode; break; }}static uint8_t get_status(uint16_t handle, uint8_t dlci){ int i; for (i = 0; i < TABLE_SIZE; i++) if (table[i].handle == handle && table[i].dlci == dlci) return table[i].status; return 0x00;}static void set_status(uint16_t handle, uint8_t dlci, uint8_t status){ int i; for (i = 0; i < TABLE_SIZE; i++) if (table[i].handle == handle && table[i].dlci == dlci) { table[i].status = status; break; }}static char *opcode2str(uint8_t opcode){ switch (opcode & 0x7f) { case 0x00: return "Connect"; case 0x01: return "Disconnect"; case 0x02: return "Put"; case 0x03: return "Get"; case 0x04: return "Reserved"; case 0x05: return "SetPath"; case 0x06: return "Reserved"; case 0x07: return "Session"; case 0x7f: return "Abort"; case 0x10: return "Continue"; case 0x20: return "Success"; case 0x21: return "Created"; case 0x22: return "Accepted"; case 0x23: return "Non-authoritative information"; case 0x24: return "No content"; case 0x25: return "Reset content"; case 0x26: return "Partial content"; case 0x30: return "Multiple choices"; case 0x31: return "Moved permanently"; case 0x32: return "Moved temporarily"; case 0x33: return "See other"; case 0x34: return "Not modified"; case 0x35: return "Use Proxy"; case 0x40: return "Bad request"; case 0x41: return "Unauthorized"; case 0x42: return "Payment required"; case 0x43: return "Forbidden"; case 0x44: return "Not found"; case 0x45: return "Method not allowed"; case 0x46: return "Not acceptable"; case 0x47: return "Proxy authentication required"; case 0x48: return "Request timeout"; case 0x49: return "Conflict"; case 0x4a: return "Gone"; case 0x4b: return "Length required"; case 0x4c: return "Precondition failed"; case 0x4d: return "Requested entity too large"; case 0x4e: return "Requested URL too large"; case 0x4f: return "Unsupported media type"; case 0x50: return "Internal server error"; case 0x51: return "Not implemented"; case 0x52: return "Bad gateway"; case 0x53: return "Service unavailable"; case 0x54: return "Gateway timeout"; case 0x55: return "HTTP version not supported"; case 0x60: return "Database full"; case 0x61: return "Database locked"; default: return "Unknown"; }}static char *hi2str(uint8_t hi){ switch (hi & 0x3f) { case 0x00: return "Count"; case 0x01: return "Name"; case 0x02: return "Type"; case 0x03: return "Length"; case 0x04: return "Time"; case 0x05: return "Description"; case 0x06: return "Target"; case 0x07: return "HTTP"; case 0x08: return "Body"; case 0x09: return "End of Body"; case 0x0a: return "Who"; case 0x0b: return "Connection ID"; case 0x0c: return "App. Parameters"; case 0x0d: return "Auth. Challenge"; case 0x0e: return "Auth. Response"; case 0x0f: return "Creator ID"; case 0x10: return "WAN UUID"; case 0x11: return "Object Class"; case 0x12: return "Session Parameters"; case 0x13: return "Session Sequence Number"; default: return "Unknown"; }}static void parse_headers(int level, struct frame *frm){ uint8_t hi, hv8; uint16_t len; uint32_t hv32; while (frm->len > 0) { hi = get_u8(frm); p_indent(level, frm); printf("%s (0x%02x)", hi2str(hi), hi); switch (hi & 0xc0) { case 0x00: /* Unicode */ len = get_u16(frm) - 3; printf(" = Unicode length %d\n", len); raw_ndump(level, frm, len); frm->ptr += len; frm->len -= len; break; case 0x40: /* Byte sequence */ len = get_u16(frm) - 3; printf(" = Sequence length %d\n", len); raw_ndump(level, frm, len); frm->ptr += len; frm->len -= len; break; case 0x80: /* One byte */ hv8 = get_u8(frm); printf(" = %d\n", hv8); break; case 0xc0: /* Four bytes */ hv32 = get_u32(frm); printf(" = %u\n", hv32); break; } }}void obex_dump(int level, struct frame *frm){ uint8_t last_opcode, opcode, status; uint8_t version, flags, constants; uint16_t length, pktlen; frm = add_frame(frm); while (frm->len > 0) { opcode = get_u8(frm); length = get_u16(frm); status = opcode & 0x7f; if (frm->len < length - 3) { frm->ptr -= 3; frm->len += 3; return; } p_indent(level, frm); last_opcode = get_opcode(frm->handle, frm->dlci); if (!(opcode & 0x70)) { printf("OBEX: %s cmd(%c): len %d", opcode2str(opcode), opcode & 0x80 ? 'f' : 'c', length); set_opcode(frm->handle, frm->dlci, opcode); } else { printf("OBEX: %s rsp(%c): status %x%02d len %d", opcode2str(last_opcode), opcode & 0x80 ? 'f' : 'c', status >> 4, status & 0xf, length); opcode = last_opcode; } if (get_status(frm->handle, frm->dlci) == 0x10) printf(" (continue)"); set_status(frm->handle, frm->dlci, status); switch (opcode & 0x7f) { case 0x00: /* Connect */ version = get_u8(frm); flags = get_u8(frm); pktlen = get_u16(frm); printf(" version %d.%d flags %d mtu %d\n", version >> 4, version & 0xf, flags, pktlen); break; case 0x05: /* SetPath */ if (length > 3) { flags = get_u8(frm); constants = get_u8(frm); printf(" flags %d constants %d\n", flags, constants); } else printf("\n"); break; default: printf("\n"); } if ((status & 0x70) && (parser.flags & DUMP_VERBOSE)) { p_indent(level, frm); printf("Status %x%02d = %s\n", status >> 4, status & 0xf, opcode2str(status)); } parse_headers(level, frm); }}void obex_clear(uint16_t handle, uint8_t dlci){ del_frame(handle, dlci);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -