tcp_inq.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. /*
  2. * Copyright 2018 Google Inc.
  3. * Author: Soheil Hassas Yeganeh (soheil@google.com)
  4. *
  5. * Simple example on how to use TCP_INQ and TCP_CM_INQ.
  6. *
  7. * License (GPLv2):
  8. *
  9. * This program is free software; you can redistribute it and/or modify it
  10. * under the terms and conditions of the GNU General Public License,
  11. * version 2, as published by the Free Software Foundation.
  12. *
  13. * This program is distributed in the hope it will be useful, but WITHOUT
  14. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  15. * FITNESS FOR A PARTICULAR PURPOSE. * See the GNU General Public License for
  16. * more details.
  17. */
  18. #define _GNU_SOURCE
  19. #include <error.h>
  20. #include <netinet/in.h>
  21. #include <netinet/tcp.h>
  22. #include <pthread.h>
  23. #include <stdio.h>
  24. #include <errno.h>
  25. #include <stdlib.h>
  26. #include <string.h>
  27. #include <sys/socket.h>
  28. #include <unistd.h>
  29. #ifndef TCP_INQ
  30. #define TCP_INQ 36
  31. #endif
  32. #ifndef TCP_CM_INQ
  33. #define TCP_CM_INQ TCP_INQ
  34. #endif
  35. #define BUF_SIZE 8192
  36. #define CMSG_SIZE 32
  37. static int family = AF_INET6;
  38. static socklen_t addr_len = sizeof(struct sockaddr_in6);
  39. static int port = 4974;
  40. static void setup_loopback_addr(int family, struct sockaddr_storage *sockaddr)
  41. {
  42. struct sockaddr_in6 *addr6 = (void *) sockaddr;
  43. struct sockaddr_in *addr4 = (void *) sockaddr;
  44. switch (family) {
  45. case PF_INET:
  46. memset(addr4, 0, sizeof(*addr4));
  47. addr4->sin_family = AF_INET;
  48. addr4->sin_addr.s_addr = htonl(INADDR_LOOPBACK);
  49. addr4->sin_port = htons(port);
  50. break;
  51. case PF_INET6:
  52. memset(addr6, 0, sizeof(*addr6));
  53. addr6->sin6_family = AF_INET6;
  54. addr6->sin6_addr = in6addr_loopback;
  55. addr6->sin6_port = htons(port);
  56. break;
  57. default:
  58. error(1, 0, "illegal family");
  59. }
  60. }
  61. void *start_server(void *arg)
  62. {
  63. int server_fd = (int)(unsigned long)arg;
  64. struct sockaddr_in addr;
  65. socklen_t addrlen = sizeof(addr);
  66. char *buf;
  67. int fd;
  68. int r;
  69. buf = malloc(BUF_SIZE);
  70. for (;;) {
  71. fd = accept(server_fd, (struct sockaddr *)&addr, &addrlen);
  72. if (fd == -1) {
  73. perror("accept");
  74. break;
  75. }
  76. do {
  77. r = send(fd, buf, BUF_SIZE, 0);
  78. } while (r < 0 && errno == EINTR);
  79. if (r < 0)
  80. perror("send");
  81. if (r != BUF_SIZE)
  82. fprintf(stderr, "can only send %d bytes\n", r);
  83. /* TCP_INQ can overestimate in-queue by one byte if we send
  84. * the FIN packet. Sleep for 1 second, so that the client
  85. * likely invoked recvmsg().
  86. */
  87. sleep(1);
  88. close(fd);
  89. }
  90. free(buf);
  91. close(server_fd);
  92. pthread_exit(0);
  93. }
  94. int main(int argc, char *argv[])
  95. {
  96. struct sockaddr_storage listen_addr, addr;
  97. int c, one = 1, inq = -1;
  98. pthread_t server_thread;
  99. char cmsgbuf[CMSG_SIZE];
  100. struct iovec iov[1];
  101. struct cmsghdr *cm;
  102. struct msghdr msg;
  103. int server_fd, fd;
  104. char *buf;
  105. while ((c = getopt(argc, argv, "46p:")) != -1) {
  106. switch (c) {
  107. case '4':
  108. family = PF_INET;
  109. addr_len = sizeof(struct sockaddr_in);
  110. break;
  111. case '6':
  112. family = PF_INET6;
  113. addr_len = sizeof(struct sockaddr_in6);
  114. break;
  115. case 'p':
  116. port = atoi(optarg);
  117. break;
  118. }
  119. }
  120. server_fd = socket(family, SOCK_STREAM, 0);
  121. if (server_fd < 0)
  122. error(1, errno, "server socket");
  123. setup_loopback_addr(family, &listen_addr);
  124. if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR,
  125. &one, sizeof(one)) != 0)
  126. error(1, errno, "setsockopt(SO_REUSEADDR)");
  127. if (bind(server_fd, (const struct sockaddr *)&listen_addr,
  128. addr_len) == -1)
  129. error(1, errno, "bind");
  130. if (listen(server_fd, 128) == -1)
  131. error(1, errno, "listen");
  132. if (pthread_create(&server_thread, NULL, start_server,
  133. (void *)(unsigned long)server_fd) != 0)
  134. error(1, errno, "pthread_create");
  135. fd = socket(family, SOCK_STREAM, 0);
  136. if (fd < 0)
  137. error(1, errno, "client socket");
  138. setup_loopback_addr(family, &addr);
  139. if (connect(fd, (const struct sockaddr *)&addr, addr_len) == -1)
  140. error(1, errno, "connect");
  141. if (setsockopt(fd, SOL_TCP, TCP_INQ, &one, sizeof(one)) != 0)
  142. error(1, errno, "setsockopt(TCP_INQ)");
  143. msg.msg_name = NULL;
  144. msg.msg_namelen = 0;
  145. msg.msg_iov = iov;
  146. msg.msg_iovlen = 1;
  147. msg.msg_control = cmsgbuf;
  148. msg.msg_controllen = sizeof(cmsgbuf);
  149. msg.msg_flags = 0;
  150. buf = malloc(BUF_SIZE);
  151. iov[0].iov_base = buf;
  152. iov[0].iov_len = BUF_SIZE / 2;
  153. if (recvmsg(fd, &msg, 0) != iov[0].iov_len)
  154. error(1, errno, "recvmsg");
  155. if (msg.msg_flags & MSG_CTRUNC)
  156. error(1, 0, "control message is truncated");
  157. for (cm = CMSG_FIRSTHDR(&msg); cm; cm = CMSG_NXTHDR(&msg, cm))
  158. if (cm->cmsg_level == SOL_TCP && cm->cmsg_type == TCP_CM_INQ)
  159. inq = *((int *) CMSG_DATA(cm));
  160. if (inq != BUF_SIZE - iov[0].iov_len) {
  161. fprintf(stderr, "unexpected inq: %d\n", inq);
  162. exit(1);
  163. }
  164. printf("PASSED\n");
  165. free(buf);
  166. close(fd);
  167. return 0;
  168. }