l2ping.c 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. *
  4. * BlueZ - Bluetooth protocol stack for Linux
  5. *
  6. * Copyright (C) 2000-2001 Qualcomm Incorporated
  7. * Copyright (C) 2002-2003 Maxim Krasnyansky <maxk@qualcomm.com>
  8. * Copyright (C) 2002-2010 Marcel Holtmann <marcel@holtmann.org>
  9. *
  10. *
  11. */
  12. #ifdef HAVE_CONFIG_H
  13. #include <config.h>
  14. #endif
  15. #define _GNU_SOURCE
  16. #include <stdio.h>
  17. #include <errno.h>
  18. #include <unistd.h>
  19. #include <stdlib.h>
  20. #include <string.h>
  21. #include <getopt.h>
  22. #include <signal.h>
  23. #include <sys/time.h>
  24. #include <poll.h>
  25. #include <sys/socket.h>
  26. #include "lib/bluetooth.h"
  27. #include "lib/hci.h"
  28. #include "lib/hci_lib.h"
  29. #include "lib/l2cap.h"
  30. /* Defaults */
  31. static bdaddr_t bdaddr;
  32. static int size = 44;
  33. static int ident = 200;
  34. static int delay = 1;
  35. static int count = -1;
  36. static int timeout = 10;
  37. static int reverse = 0;
  38. static int verify = 0;
  39. /* Stats */
  40. static int sent_pkt = 0;
  41. static int recv_pkt = 0;
  42. static float tv2fl(struct timeval tv)
  43. {
  44. return (float)(tv.tv_sec*1000.0) + (float)(tv.tv_usec/1000.0);
  45. }
  46. static void stat(int sig)
  47. {
  48. int loss = sent_pkt ? (float)((sent_pkt-recv_pkt)/(sent_pkt/100.0)) : 0;
  49. printf("%d sent, %d received, %d%% loss\n", sent_pkt, recv_pkt, loss);
  50. exit(0);
  51. }
  52. static void ping(char *svr)
  53. {
  54. struct sigaction sa;
  55. struct sockaddr_l2 addr;
  56. socklen_t optlen;
  57. unsigned char *send_buf;
  58. unsigned char *recv_buf;
  59. char str[18];
  60. int i, sk, lost;
  61. uint8_t id;
  62. memset(&sa, 0, sizeof(sa));
  63. sa.sa_handler = stat;
  64. sigaction(SIGINT, &sa, NULL);
  65. send_buf = malloc(L2CAP_CMD_HDR_SIZE + size);
  66. recv_buf = malloc(L2CAP_CMD_HDR_SIZE + size);
  67. if (!send_buf || !recv_buf) {
  68. perror("Can't allocate buffer");
  69. exit(1);
  70. }
  71. /* Create socket */
  72. sk = socket(PF_BLUETOOTH, SOCK_RAW, BTPROTO_L2CAP);
  73. if (sk < 0) {
  74. perror("Can't create socket");
  75. goto error;
  76. }
  77. /* Bind to local address */
  78. memset(&addr, 0, sizeof(addr));
  79. addr.l2_family = AF_BLUETOOTH;
  80. bacpy(&addr.l2_bdaddr, &bdaddr);
  81. if (bind(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
  82. perror("Can't bind socket");
  83. goto error;
  84. }
  85. /* Connect to remote device */
  86. memset(&addr, 0, sizeof(addr));
  87. addr.l2_family = AF_BLUETOOTH;
  88. str2ba(svr, &addr.l2_bdaddr);
  89. if (connect(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
  90. perror("Can't connect");
  91. goto error;
  92. }
  93. /* Get local address */
  94. memset(&addr, 0, sizeof(addr));
  95. optlen = sizeof(addr);
  96. if (getsockname(sk, (struct sockaddr *) &addr, &optlen) < 0) {
  97. perror("Can't get local address");
  98. goto error;
  99. }
  100. ba2str(&addr.l2_bdaddr, str);
  101. printf("Ping: %s from %s (data size %d) ...\n", svr, str, size);
  102. /* Initialize send buffer */
  103. for (i = 0; i < size; i++)
  104. send_buf[L2CAP_CMD_HDR_SIZE + i] = (i % 40) + 'A';
  105. id = ident;
  106. while (count == -1 || count-- > 0) {
  107. struct timeval tv_send, tv_recv, tv_diff;
  108. l2cap_cmd_hdr *send_cmd = (l2cap_cmd_hdr *) send_buf;
  109. l2cap_cmd_hdr *recv_cmd = (l2cap_cmd_hdr *) recv_buf;
  110. /* Build command header */
  111. send_cmd->ident = id;
  112. send_cmd->len = htobs(size);
  113. if (reverse)
  114. send_cmd->code = L2CAP_ECHO_RSP;
  115. else
  116. send_cmd->code = L2CAP_ECHO_REQ;
  117. gettimeofday(&tv_send, NULL);
  118. /* Send Echo Command */
  119. if (send(sk, send_buf, L2CAP_CMD_HDR_SIZE + size, 0) <= 0) {
  120. perror("Send failed");
  121. goto error;
  122. }
  123. /* Wait for Echo Response */
  124. lost = 0;
  125. while (1) {
  126. struct pollfd pf[1];
  127. int err;
  128. pf[0].fd = sk;
  129. pf[0].events = POLLIN;
  130. if ((err = poll(pf, 1, timeout * 1000)) < 0) {
  131. perror("Poll failed");
  132. goto error;
  133. }
  134. if (!err) {
  135. lost = 1;
  136. break;
  137. }
  138. if ((err = recv(sk, recv_buf, L2CAP_CMD_HDR_SIZE + size, 0)) < 0) {
  139. perror("Recv failed");
  140. goto error;
  141. }
  142. if (!err){
  143. printf("Disconnected\n");
  144. goto error;
  145. }
  146. recv_cmd->len = btohs(recv_cmd->len);
  147. /* Check for our id */
  148. if (recv_cmd->ident != id)
  149. continue;
  150. /* Check type */
  151. if (!reverse && recv_cmd->code == L2CAP_ECHO_RSP)
  152. break;
  153. if (recv_cmd->code == L2CAP_COMMAND_REJ) {
  154. printf("Peer doesn't support Echo packets\n");
  155. goto error;
  156. }
  157. }
  158. sent_pkt++;
  159. if (!lost) {
  160. recv_pkt++;
  161. gettimeofday(&tv_recv, NULL);
  162. timersub(&tv_recv, &tv_send, &tv_diff);
  163. if (verify) {
  164. /* Check payload length */
  165. if (recv_cmd->len != size) {
  166. fprintf(stderr, "Received %d bytes, expected %d\n",
  167. recv_cmd->len, size);
  168. goto error;
  169. }
  170. /* Check payload */
  171. if (memcmp(&send_buf[L2CAP_CMD_HDR_SIZE],
  172. &recv_buf[L2CAP_CMD_HDR_SIZE], size)) {
  173. fprintf(stderr, "Response payload different.\n");
  174. goto error;
  175. }
  176. }
  177. printf("%d bytes from %s id %d time %.2fms\n", recv_cmd->len, svr,
  178. id - ident, tv2fl(tv_diff));
  179. if (delay)
  180. sleep(delay);
  181. } else {
  182. printf("no response from %s: id %d\n", svr, id - ident);
  183. }
  184. if (++id > 254)
  185. id = ident;
  186. }
  187. stat(0);
  188. free(send_buf);
  189. free(recv_buf);
  190. return;
  191. error:
  192. close(sk);
  193. free(send_buf);
  194. free(recv_buf);
  195. exit(1);
  196. }
  197. static void usage(void)
  198. {
  199. printf("l2ping - L2CAP ping\n");
  200. printf("Usage:\n");
  201. printf("\tl2ping [-i device] [-s size] [-c count] [-t timeout] [-d delay] [-f] [-r] [-v] <bdaddr>\n");
  202. printf("\t-f Flood ping (delay = 0)\n");
  203. printf("\t-r Reverse ping\n");
  204. printf("\t-v Verify request and response payload\n");
  205. }
  206. int main(int argc, char *argv[])
  207. {
  208. int opt;
  209. /* Default options */
  210. bacpy(&bdaddr, BDADDR_ANY);
  211. while ((opt=getopt(argc,argv,"i:d:s:c:t:frv")) != EOF) {
  212. switch(opt) {
  213. case 'i':
  214. if (!strncasecmp(optarg, "hci", 3))
  215. hci_devba(atoi(optarg + 3), &bdaddr);
  216. else
  217. str2ba(optarg, &bdaddr);
  218. break;
  219. case 'd':
  220. delay = atoi(optarg);
  221. break;
  222. case 'f':
  223. /* Kinda flood ping */
  224. delay = 0;
  225. break;
  226. case 'r':
  227. /* Use responses instead of requests */
  228. reverse = 1;
  229. break;
  230. case 'v':
  231. verify = 1;
  232. break;
  233. case 'c':
  234. count = atoi(optarg);
  235. break;
  236. case 't':
  237. timeout = atoi(optarg);
  238. break;
  239. case 's':
  240. size = atoi(optarg);
  241. break;
  242. default:
  243. usage();
  244. exit(1);
  245. }
  246. }
  247. if (!(argc - optind)) {
  248. usage();
  249. exit(1);
  250. }
  251. ping(argv[optind]);
  252. return 0;
  253. }