| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236 |
- #include <string>
- #include <stdio.h>
- #include <errno.h>
- #include <unistd.h>
- #include <stdlib.h>
- #include <string.h>
- #include <getopt.h>
- #include <signal.h>
- #include <sys/time.h>
- #include <sys/poll.h>
- #include <sys/socket.h>
- #include <bluetooth/bluetooth.h>
- #include <bluetooth/hci.h>
- #include <bluetooth/hci_lib.h>
- #include <bluetooth/l2cap.h>
- /* Defaults */
- static bdaddr_t bdaddr;
- static int size = 44;
- static int ident = 200;
- static int delay = 1;
- static int count = -1;
- static int timeout = 10;
- static int reverse = 0;
- static int verify = 0;
- /* Stats */
- static int sent_pkt = 0;
- static int recv_pkt = 0;
- static float tv2fl(struct timeval tv)
- {
- return (float)(tv.tv_sec*1000.0) + (float)(tv.tv_usec/1000.0);
- }
- static void stat(int /*sig*/)
- {
- int loss = sent_pkt ? (float)((sent_pkt-recv_pkt)/(sent_pkt/100.0)) : 0;
- printf("%d sent, %d received, %d%% loss\n", sent_pkt, recv_pkt, loss);
- exit(0);
- }
- static void ping(const char* svr)
- {
- struct sigaction sa;
- struct sockaddr_l2 addr;
- socklen_t optlen;
- unsigned char *send_buf;
- unsigned char *recv_buf;
- char str[18];
- int i, sk, lost;
- uint8_t id;
- memset(&sa, 0, sizeof(sa));
- sa.sa_handler = stat;
- sigaction(SIGINT, &sa, NULL);
- send_buf = static_cast<unsigned char*>(malloc(L2CAP_CMD_HDR_SIZE + size));
- recv_buf = static_cast<unsigned char*>(malloc(L2CAP_CMD_HDR_SIZE + size));
- if (!send_buf || !recv_buf) {
- perror("Can't allocate buffer");
- exit(1);
- }
- /* Create socket */
- sk = socket(PF_BLUETOOTH, SOCK_RAW, BTPROTO_L2CAP);
- if (sk < 0) {
- perror("Can't create socket");
- goto error;
- }
- /* Bind to local address */
- memset(&addr, 0, sizeof(addr));
- addr.l2_family = AF_BLUETOOTH;
- bacpy(&addr.l2_bdaddr, &bdaddr);
- if (bind(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
- perror("Can't bind socket");
- goto error;
- }
- /* Connect to remote device */
- memset(&addr, 0, sizeof(addr));
- addr.l2_family = AF_BLUETOOTH;
- str2ba(svr, &addr.l2_bdaddr);
- if (connect(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
- perror("Can't connect");
- goto error;
- }
- /* Get local address */
- memset(&addr, 0, sizeof(addr));
- optlen = sizeof(addr);
- if (getsockname(sk, (struct sockaddr *) &addr, &optlen) < 0) {
- perror("Can't get local address");
- goto error;
- }
- ba2str(&addr.l2_bdaddr, str);
- printf("Ping: %s from %s (data size %d) ...\n", svr, str, size);
- /* Initialize send buffer */
- for (i = 0; i < size; i++)
- send_buf[L2CAP_CMD_HDR_SIZE + i] = (i % 40) + 'A';
- id = ident;
- while (count == -1 || count-- > 0) {
- struct timeval tv_send, tv_recv, tv_diff;
- l2cap_cmd_hdr *send_cmd = (l2cap_cmd_hdr *) send_buf;
- l2cap_cmd_hdr *recv_cmd = (l2cap_cmd_hdr *) recv_buf;
- /* Build command header */
- send_cmd->ident = id;
- send_cmd->len = htobs(size);
- if (reverse)
- send_cmd->code = L2CAP_ECHO_RSP;
- else
- send_cmd->code = L2CAP_ECHO_REQ;
- gettimeofday(&tv_send, NULL);
- /* Send Echo Command */
- if (send(sk, send_buf, L2CAP_CMD_HDR_SIZE + size, 0) <= 0) {
- perror("Send failed");
- goto error;
- }
- /* Wait for Echo Response */
- lost = 0;
- while (1) {
- struct pollfd pf[1];
- int err;
- pf[0].fd = sk;
- pf[0].events = POLLIN;
- if ((err = poll(pf, 1, timeout * 1000)) < 0) {
- perror("Poll failed");
- goto error;
- }
- if (!err) {
- lost = 1;
- break;
- }
- if ((err = recv(sk, recv_buf, L2CAP_CMD_HDR_SIZE + size, 0)) < 0) {
- perror("Recv failed");
- goto error;
- }
- if (!err){
- printf("Disconnected\n");
- goto error;
- }
- recv_cmd->len = btohs(recv_cmd->len);
- /* Check for our id */
- if (recv_cmd->ident != id)
- continue;
- /* Check type */
- if (!reverse && recv_cmd->code == L2CAP_ECHO_RSP)
- break;
- if (recv_cmd->code == L2CAP_COMMAND_REJ) {
- printf("Peer doesn't support Echo packets\n");
- goto error;
- }
- }
- sent_pkt++;
- if (!lost) {
- recv_pkt++;
- gettimeofday(&tv_recv, NULL);
- timersub(&tv_recv, &tv_send, &tv_diff);
- if (verify) {
- /* Check payload length */
- if (recv_cmd->len != size) {
- fprintf(stderr, "Received %d bytes, expected %d\n",
- recv_cmd->len, size);
- goto error;
- }
- /* Check payload */
- if (memcmp(&send_buf[L2CAP_CMD_HDR_SIZE],
- &recv_buf[L2CAP_CMD_HDR_SIZE], size)) {
- fprintf(stderr, "Response payload different.\n");
- goto error;
- }
- }
- printf("%d bytes from %s id %d time %.2fms\n", recv_cmd->len, svr,
- id - ident, tv2fl(tv_diff));
- if (delay)
- sleep(delay);
- } else {
- printf("no response from %s: id %d\n", svr, id - ident);
- }
- if (++id > 254)
- id = ident;
- }
- stat(0);
- free(send_buf);
- free(recv_buf);
- return;
- error:
- close(sk);
- free(send_buf);
- free(recv_buf);
- exit(1);
- }
- int main(int /*argc*/, char** /*argv[]*/)
- {
- std::string address = "48:4B:AA:85:30:C5";
- ping(address.c_str());
- //address = "d0:c5:f3:84:99:de";
- //ping(address.c_str());
- return 0;
- }
|