bnep.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. *
  4. * BlueZ - Bluetooth protocol stack for Linux
  5. *
  6. * Copyright (C) 2002-2003 Takashi Sasai <sasai@sm.sony.co.jp>
  7. * Copyright (C) 2003-2011 Marcel Holtmann <marcel@holtmann.org>
  8. *
  9. *
  10. */
  11. #ifdef HAVE_CONFIG_H
  12. #include <config.h>
  13. #endif
  14. #define _GNU_SOURCE
  15. #include <stdio.h>
  16. #include <errno.h>
  17. #include <unistd.h>
  18. #include <stdlib.h>
  19. #include <string.h>
  20. #include <net/ethernet.h>
  21. #include "parser.h"
  22. /* BNEP Type */
  23. #define BNEP_GENERAL_ETHERNET 0x00
  24. #define BNEP_CONTROL 0x01
  25. #define BNEP_COMPRESSED_ETHERNET 0x02
  26. #define BNEP_COMPRESSED_ETHERNET_SOURCE_ONLY 0x03
  27. #define BNEP_COMPRESSED_ETHERNET_DEST_ONLY 0x04
  28. /* BNEP Control Packet Type */
  29. #define BNEP_CONTROL_COMMAND_NOT_UNDERSTOOD 0x00
  30. #define BNEP_SETUP_CONNECTION_REQUEST_MSG 0x01
  31. #define BNEP_SETUP_CONNECTION_RESPONSE_MSG 0x02
  32. #define BNEP_FILTER_NET_TYPE_SET_MSG 0x03
  33. #define BNEP_FILTER_NET_TYPE_RESPONSE_MSG 0x04
  34. #define BNEP_FILTER_MULT_ADDR_SET_MSG 0x05
  35. #define BNEP_FILTER_MULT_ADDR_RESPONSE_MSG 0x06
  36. /* BNEP Extension Type */
  37. #define BNEP_EXTENSION_CONTROL 0x00
  38. #ifndef ETHERTYPE_IPV6
  39. #define ETHERTYPE_IPV6 ETH_P_IPV6
  40. #endif
  41. static char *get_macaddr(struct frame *frm)
  42. {
  43. static char str[20];
  44. unsigned char *buf = frm->ptr;
  45. sprintf(str, "%02x:%02x:%02x:%02x:%02x:%02x",
  46. buf[0], buf[1], buf[2], buf[3], buf[4], buf[5]);
  47. frm->ptr += 6;
  48. frm->len -= 6;
  49. return str;
  50. }
  51. static void bnep_control(int level, struct frame *frm, int header_length)
  52. {
  53. uint8_t uuid_size;
  54. int i, length;
  55. char *s;
  56. uint32_t uuid = 0;
  57. uint8_t type = p_get_u8(frm);
  58. p_indent(++level, frm);
  59. switch (type) {
  60. case BNEP_CONTROL_COMMAND_NOT_UNDERSTOOD:
  61. printf("Not Understood(0x%02x) type 0x%02x\n", type, p_get_u8(frm));
  62. break;
  63. case BNEP_SETUP_CONNECTION_REQUEST_MSG:
  64. uuid_size = p_get_u8(frm);
  65. printf("Setup Req(0x%02x) size 0x%02x ", type, uuid_size);
  66. switch (uuid_size) {
  67. case 2:
  68. uuid = p_get_u16(frm);
  69. printf("dst 0x%x", uuid);
  70. if ((s = get_uuid_name(uuid)) != 0)
  71. printf("(%s)", s);
  72. uuid = p_get_u16(frm);
  73. printf(" src 0x%x", uuid);
  74. if ((s = get_uuid_name(uuid)) != 0)
  75. printf("(%s)", s);
  76. printf("\n");
  77. break;
  78. case 4:
  79. uuid = p_get_u32(frm);
  80. printf("dst 0x%x", uuid);
  81. if ((s = get_uuid_name(uuid)) != 0)
  82. printf("(%s)", s);
  83. uuid = p_get_u32(frm);
  84. printf(" src 0x%x", uuid);
  85. if ((s = get_uuid_name(uuid)) != 0)
  86. printf("(%s)", s);
  87. printf("\n");
  88. break;
  89. case 16:
  90. uuid = p_get_u32(frm);
  91. printf("dst 0x%x", uuid);
  92. if ((s = get_uuid_name(uuid)) != 0)
  93. printf("(%s)", s);
  94. frm->ptr += 12;
  95. frm->len -= 12;
  96. uuid = p_get_u32(frm);
  97. printf(" src 0x%x", uuid);
  98. if ((s = get_uuid_name(uuid)) != 0)
  99. printf("(%s)", s);
  100. printf("\n");
  101. frm->ptr += 12;
  102. frm->len -= 12;
  103. break;
  104. default:
  105. frm->ptr += (uuid_size * 2);
  106. frm->len -= (uuid_size * 2);
  107. break;
  108. }
  109. break;
  110. case BNEP_SETUP_CONNECTION_RESPONSE_MSG:
  111. printf("Setup Rsp(0x%02x) res 0x%04x\n",
  112. type, p_get_u16(frm));
  113. break;
  114. case BNEP_FILTER_NET_TYPE_SET_MSG:
  115. length = p_get_u16(frm);
  116. printf("Filter NetType Set(0x%02x) len 0x%04x\n",
  117. type, length);
  118. for (i = 0; i < length / 4; i++) {
  119. p_indent(level + 1, frm);
  120. printf("0x%04x - ", p_get_u16(frm));
  121. printf("0x%04x\n", p_get_u16(frm));
  122. }
  123. break;
  124. case BNEP_FILTER_NET_TYPE_RESPONSE_MSG:
  125. printf("Filter NetType Rsp(0x%02x) res 0x%04x\n",
  126. type, p_get_u16(frm));
  127. break;
  128. case BNEP_FILTER_MULT_ADDR_SET_MSG:
  129. length = p_get_u16(frm);
  130. printf("Filter MultAddr Set(0x%02x) len 0x%04x\n",
  131. type, length);
  132. for (i = 0; i < length / 12; i++) {
  133. p_indent(level + 1, frm);
  134. printf("%s - ", get_macaddr(frm));
  135. printf("%s\n", get_macaddr(frm));
  136. }
  137. break;
  138. case BNEP_FILTER_MULT_ADDR_RESPONSE_MSG:
  139. printf("Filter MultAddr Rsp(0x%02x) res 0x%04x\n",
  140. type, p_get_u16(frm));
  141. break;
  142. default:
  143. printf("Unknown control type(0x%02x)\n", type);
  144. raw_ndump(level + 1, frm, header_length - 1);
  145. frm->ptr += header_length - 1;
  146. frm->len -= header_length - 1;
  147. return;
  148. }
  149. }
  150. static void bnep_eval_extension(int level, struct frame *frm)
  151. {
  152. uint8_t type = p_get_u8(frm);
  153. uint8_t length = p_get_u8(frm);
  154. int extension = type & 0x80;
  155. p_indent(level, frm);
  156. switch (type & 0x7f) {
  157. case BNEP_EXTENSION_CONTROL:
  158. printf("Ext Control(0x%02x|%s) len 0x%02x\n",
  159. type & 0x7f, extension ? "1" : "0", length);
  160. bnep_control(level, frm, length);
  161. break;
  162. default:
  163. printf("Ext Unknown(0x%02x|%s) len 0x%02x\n",
  164. type & 0x7f, extension ? "1" : "0", length);
  165. raw_ndump(level + 1, frm, length);
  166. frm->ptr += length;
  167. frm->len -= length;
  168. }
  169. if (extension)
  170. bnep_eval_extension(level, frm);
  171. }
  172. void bnep_dump(int level, struct frame *frm)
  173. {
  174. uint8_t type = p_get_u8(frm);
  175. uint16_t proto = 0x0000;
  176. int extension = type & 0x80;
  177. p_indent(level, frm);
  178. switch (type & 0x7f) {
  179. case BNEP_CONTROL:
  180. printf("BNEP: Control(0x%02x|%s)\n",
  181. type & 0x7f, extension ? "1" : "0");
  182. bnep_control(level, frm, -1);
  183. break;
  184. case BNEP_COMPRESSED_ETHERNET:
  185. printf("BNEP: Compressed(0x%02x|%s)\n",
  186. type & 0x7f, extension ? "1" : "0");
  187. p_indent(++level, frm);
  188. proto = p_get_u16(frm);
  189. printf("[proto 0x%04x]\n", proto);
  190. break;
  191. case BNEP_GENERAL_ETHERNET:
  192. printf("BNEP: General ethernet(0x%02x|%s)\n",
  193. type & 0x7f, extension ? "1" : "0");
  194. p_indent(++level, frm);
  195. printf("dst %s ", get_macaddr(frm));
  196. printf("src %s ", get_macaddr(frm));
  197. proto = p_get_u16(frm);
  198. printf("[proto 0x%04x]\n", proto);
  199. break;
  200. case BNEP_COMPRESSED_ETHERNET_DEST_ONLY:
  201. printf("BNEP: Compressed DestOnly(0x%02x|%s)\n",
  202. type & 0x7f, extension ? "1" : "0");
  203. p_indent(++level, frm);
  204. printf("dst %s ", get_macaddr(frm));
  205. proto = p_get_u16(frm);
  206. printf("[proto 0x%04x]\n", proto);
  207. break;
  208. case BNEP_COMPRESSED_ETHERNET_SOURCE_ONLY:
  209. printf("BNEP: Compressed SrcOnly(0x%02x|%s)\n",
  210. type & 0x7f, extension ? "1" : "0");
  211. p_indent(++level, frm);
  212. printf("src %s ", get_macaddr(frm));
  213. proto = p_get_u16(frm);
  214. printf("[proto 0x%04x]\n", proto);
  215. break;
  216. default:
  217. printf("(Unknown packet type)\n");
  218. return;
  219. }
  220. /* Extension info */
  221. if (extension)
  222. bnep_eval_extension(++level, frm);
  223. /* Control packet => No payload info */
  224. if ((type & 0x7f) == BNEP_CONTROL)
  225. return;
  226. /* 802.1p header */
  227. if (proto == 0x8100) {
  228. p_indent(level, frm);
  229. printf("802.1p Header: 0x%04x ", p_get_u16(frm));
  230. proto = p_get_u16(frm);
  231. printf("[proto 0x%04x]\n", proto);
  232. }
  233. if (!(parser.flags & DUMP_VERBOSE)) {
  234. raw_dump(level, frm);
  235. return;
  236. }
  237. switch (proto) {
  238. case ETHERTYPE_ARP:
  239. p_indent(++level, frm);
  240. printf("ARP: ");
  241. arp_dump(level, frm);
  242. break;
  243. case ETHERTYPE_REVARP:
  244. p_indent(++level, frm);
  245. printf("RARP: ");
  246. arp_dump(level, frm);
  247. break;
  248. case ETHERTYPE_IP:
  249. p_indent(++level, frm);
  250. printf("IP: ");
  251. ip_dump(level, frm);
  252. break;
  253. case ETHERTYPE_IPV6:
  254. p_indent(++level, frm);
  255. printf("IPV6: ");
  256. ip_dump(level, frm);
  257. break;
  258. default:
  259. raw_dump(level, frm);
  260. break;
  261. }
  262. }