Test.cc 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. #include <iostream>
  2. #include <string>
  3. #include <stdio.h>
  4. #include <errno.h>
  5. #include <unistd.h>
  6. #include <stdlib.h>
  7. #include <string.h>
  8. #include <getopt.h>
  9. #include <signal.h>
  10. #include <sys/time.h>
  11. #include <sys/poll.h>
  12. #include <sys/socket.h>
  13. #include <bluetooth/bluetooth.h>
  14. #include <bluetooth/hci.h>
  15. #include <bluetooth/hci_lib.h>
  16. #include <bluetooth/l2cap.h>
  17. #include "Util/Timer.h"
  18. /* Defaults */
  19. static bdaddr_t bdaddr;
  20. static int size = 44;
  21. static int ident = 200;
  22. static int delay = 1;
  23. static int count = -1;
  24. static int timeout = 10;
  25. static int reverse = 0;
  26. static int verify = 0;
  27. /* Stats */
  28. static int sent_pkt = 0;
  29. static int recv_pkt = 0;
  30. static float tv2fl(struct timeval tv)
  31. {
  32. return (float)(tv.tv_sec*1000.0) + (float)(tv.tv_usec/1000.0);
  33. }
  34. static void stat(int /*sig*/)
  35. {
  36. int loss = sent_pkt ? (float)((sent_pkt-recv_pkt)/(sent_pkt/100.0)) : 0;
  37. printf("%d sent, %d received, %d%% loss\n", sent_pkt, recv_pkt, loss);
  38. exit(0);
  39. }
  40. static void ping(const char* svr)
  41. {
  42. struct sigaction sa;
  43. struct sockaddr_l2 addr;
  44. socklen_t optlen;
  45. unsigned char *send_buf;
  46. unsigned char *recv_buf;
  47. char str[18];
  48. int i, sk, lost;
  49. uint8_t id;
  50. memset(&sa, 0, sizeof(sa));
  51. sa.sa_handler = stat;
  52. sigaction(SIGINT, &sa, NULL);
  53. send_buf = static_cast<unsigned char*>(malloc(L2CAP_CMD_HDR_SIZE + size));
  54. recv_buf = static_cast<unsigned char*>(malloc(L2CAP_CMD_HDR_SIZE + size));
  55. if (!send_buf || !recv_buf) {
  56. perror("Can't allocate buffer");
  57. exit(1);
  58. }
  59. /* Create socket */
  60. sk = socket(PF_BLUETOOTH, SOCK_RAW, BTPROTO_L2CAP);
  61. if (sk < 0) {
  62. perror("Can't create socket");
  63. goto error;
  64. }
  65. /* Bind to local address */
  66. memset(&addr, 0, sizeof(addr));
  67. addr.l2_family = AF_BLUETOOTH;
  68. bacpy(&addr.l2_bdaddr, &bdaddr);
  69. if (bind(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
  70. perror("Can't bind socket");
  71. goto error;
  72. }
  73. /* Connect to remote device */
  74. memset(&addr, 0, sizeof(addr));
  75. addr.l2_family = AF_BLUETOOTH;
  76. str2ba(svr, &addr.l2_bdaddr);
  77. if (connect(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
  78. perror("Can't connect");
  79. goto error;
  80. }
  81. /* Get local address */
  82. memset(&addr, 0, sizeof(addr));
  83. optlen = sizeof(addr);
  84. if (getsockname(sk, (struct sockaddr *) &addr, &optlen) < 0) {
  85. perror("Can't get local address");
  86. goto error;
  87. }
  88. ba2str(&addr.l2_bdaddr, str);
  89. printf("Ping: %s from %s (data size %d) ...\n", svr, str, size);
  90. /* Initialize send buffer */
  91. for (i = 0; i < size; i++)
  92. send_buf[L2CAP_CMD_HDR_SIZE + i] = (i % 40) + 'A';
  93. id = ident;
  94. while (count == -1 || count-- > 0) {
  95. struct timeval tv_send, tv_recv, tv_diff;
  96. l2cap_cmd_hdr *send_cmd = (l2cap_cmd_hdr *) send_buf;
  97. l2cap_cmd_hdr *recv_cmd = (l2cap_cmd_hdr *) recv_buf;
  98. /* Build command header */
  99. send_cmd->ident = id;
  100. send_cmd->len = htobs(size);
  101. if (reverse)
  102. send_cmd->code = L2CAP_ECHO_RSP;
  103. else
  104. send_cmd->code = L2CAP_ECHO_REQ;
  105. gettimeofday(&tv_send, NULL);
  106. /* Send Echo Command */
  107. if (send(sk, send_buf, L2CAP_CMD_HDR_SIZE + size, 0) <= 0) {
  108. perror("Send failed");
  109. goto error;
  110. }
  111. /* Wait for Echo Response */
  112. lost = 0;
  113. while (1) {
  114. struct pollfd pf[1];
  115. int err;
  116. pf[0].fd = sk;
  117. pf[0].events = POLLIN;
  118. if ((err = poll(pf, 1, timeout * 1000)) < 0) {
  119. perror("Poll failed");
  120. goto error;
  121. }
  122. if (!err) {
  123. lost = 1;
  124. break;
  125. }
  126. if ((err = recv(sk, recv_buf, L2CAP_CMD_HDR_SIZE + size, 0)) < 0) {
  127. perror("Recv failed");
  128. goto error;
  129. }
  130. if (!err){
  131. printf("Disconnected\n");
  132. goto error;
  133. }
  134. recv_cmd->len = btohs(recv_cmd->len);
  135. /* Check for our id */
  136. if (recv_cmd->ident != id)
  137. continue;
  138. /* Check type */
  139. if (!reverse && recv_cmd->code == L2CAP_ECHO_RSP)
  140. break;
  141. if (recv_cmd->code == L2CAP_COMMAND_REJ) {
  142. printf("Peer doesn't support Echo packets\n");
  143. goto error;
  144. }
  145. }
  146. sent_pkt++;
  147. if (!lost) {
  148. recv_pkt++;
  149. gettimeofday(&tv_recv, NULL);
  150. timersub(&tv_recv, &tv_send, &tv_diff);
  151. if (verify) {
  152. /* Check payload length */
  153. if (recv_cmd->len != size) {
  154. fprintf(stderr, "Received %d bytes, expected %d\n",
  155. recv_cmd->len, size);
  156. goto error;
  157. }
  158. /* Check payload */
  159. if (memcmp(&send_buf[L2CAP_CMD_HDR_SIZE],
  160. &recv_buf[L2CAP_CMD_HDR_SIZE], size)) {
  161. fprintf(stderr, "Response payload different.\n");
  162. goto error;
  163. }
  164. }
  165. printf("%d bytes from %s id %d time %.2fms\n", recv_cmd->len, svr,
  166. id - ident, tv2fl(tv_diff));
  167. if (delay)
  168. sleep(delay);
  169. } else {
  170. printf("no response from %s: id %d\n", svr, id - ident);
  171. }
  172. if (++id > 254)
  173. id = ident;
  174. }
  175. stat(0);
  176. free(send_buf);
  177. free(recv_buf);
  178. return;
  179. error:
  180. close(sk);
  181. free(send_buf);
  182. free(recv_buf);
  183. exit(1);
  184. }
  185. void print()
  186. {
  187. std::cout << "Print" << std::endl;
  188. }
  189. int main(int /*argc*/, char** /*argv[]*/)
  190. {
  191. //std::string address = "48:4B:AA:85:30:C5";
  192. //ping(address.c_str());
  193. //address = "d0:c5:f3:84:99:de";
  194. //ping(address.c_str());
  195. PresenceDetection::Util::Timer timer;
  196. std::function<void()> f_print = print;
  197. std::cout << "---" << std::endl;
  198. timer.StartContinuous(1000, f_print);
  199. timer.Stop();
  200. timer.Wait();
  201. std::cout << "---" << std::endl << std::endl;
  202. std::cout << "---" << std::endl;
  203. timer.StartContinuous(3000, f_print);
  204. timer.Abort();
  205. timer.Wait();
  206. std::cout << "---" << std::endl << std::endl;
  207. std::cout << "---" << std::endl;
  208. timer.StartContinuous(3000, f_print);
  209. sleep(1);
  210. timer.Abort();
  211. timer.Wait();
  212. std::cout << "---" << std::endl << std::endl;
  213. return 0;
  214. }