tcp_inq.c 4.1 KB

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