fin_ack_lat.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <arpa/inet.h>
  3. #include <errno.h>
  4. #include <error.h>
  5. #include <netinet/in.h>
  6. #include <netinet/tcp.h>
  7. #include <signal.h>
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <sys/socket.h>
  11. #include <sys/time.h>
  12. #include <unistd.h>
  13. static int child_pid;
  14. static unsigned long timediff(struct timeval s, struct timeval e)
  15. {
  16. unsigned long s_us, e_us;
  17. s_us = s.tv_sec * 1000000 + s.tv_usec;
  18. e_us = e.tv_sec * 1000000 + e.tv_usec;
  19. if (s_us > e_us)
  20. return 0;
  21. return e_us - s_us;
  22. }
  23. static void client(int port)
  24. {
  25. int sock = 0;
  26. struct sockaddr_in addr, laddr;
  27. socklen_t len = sizeof(laddr);
  28. struct linger sl;
  29. int flag = 1;
  30. int buffer;
  31. struct timeval start, end;
  32. unsigned long lat, sum_lat = 0, nr_lat = 0;
  33. while (1) {
  34. gettimeofday(&start, NULL);
  35. sock = socket(AF_INET, SOCK_STREAM, 0);
  36. if (sock < 0)
  37. error(-1, errno, "socket creation");
  38. sl.l_onoff = 1;
  39. sl.l_linger = 0;
  40. if (setsockopt(sock, SOL_SOCKET, SO_LINGER, &sl, sizeof(sl)))
  41. error(-1, errno, "setsockopt(linger)");
  42. if (setsockopt(sock, IPPROTO_TCP, TCP_NODELAY,
  43. &flag, sizeof(flag)))
  44. error(-1, errno, "setsockopt(nodelay)");
  45. addr.sin_family = AF_INET;
  46. addr.sin_port = htons(port);
  47. if (inet_pton(AF_INET, "127.0.0.1", &addr.sin_addr) <= 0)
  48. error(-1, errno, "inet_pton");
  49. if (connect(sock, (struct sockaddr *)&addr, sizeof(addr)) < 0)
  50. error(-1, errno, "connect");
  51. send(sock, &buffer, sizeof(buffer), 0);
  52. if (read(sock, &buffer, sizeof(buffer)) == -1)
  53. error(-1, errno, "waiting read");
  54. gettimeofday(&end, NULL);
  55. lat = timediff(start, end);
  56. sum_lat += lat;
  57. nr_lat++;
  58. if (lat < 100000)
  59. goto close;
  60. if (getsockname(sock, (struct sockaddr *)&laddr, &len) == -1)
  61. error(-1, errno, "getsockname");
  62. printf("port: %d, lat: %lu, avg: %lu, nr: %lu\n",
  63. ntohs(laddr.sin_port), lat,
  64. sum_lat / nr_lat, nr_lat);
  65. close:
  66. fflush(stdout);
  67. close(sock);
  68. }
  69. }
  70. static void server(int sock, struct sockaddr_in address)
  71. {
  72. int accepted;
  73. int addrlen = sizeof(address);
  74. int buffer;
  75. while (1) {
  76. accepted = accept(sock, (struct sockaddr *)&address,
  77. (socklen_t *)&addrlen);
  78. if (accepted < 0)
  79. error(-1, errno, "accept");
  80. if (read(accepted, &buffer, sizeof(buffer)) == -1)
  81. error(-1, errno, "read");
  82. close(accepted);
  83. }
  84. }
  85. static void sig_handler(int signum)
  86. {
  87. kill(SIGTERM, child_pid);
  88. exit(0);
  89. }
  90. int main(int argc, char const *argv[])
  91. {
  92. int sock;
  93. int opt = 1;
  94. struct sockaddr_in address;
  95. struct sockaddr_in laddr;
  96. socklen_t len = sizeof(laddr);
  97. if (signal(SIGTERM, sig_handler) == SIG_ERR)
  98. error(-1, errno, "signal");
  99. sock = socket(AF_INET, SOCK_STREAM, 0);
  100. if (sock < 0)
  101. error(-1, errno, "socket");
  102. if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT,
  103. &opt, sizeof(opt)) == -1)
  104. error(-1, errno, "setsockopt");
  105. address.sin_family = AF_INET;
  106. address.sin_addr.s_addr = INADDR_ANY;
  107. /* dynamically allocate unused port */
  108. address.sin_port = 0;
  109. if (bind(sock, (struct sockaddr *)&address, sizeof(address)) < 0)
  110. error(-1, errno, "bind");
  111. if (listen(sock, 3) < 0)
  112. error(-1, errno, "listen");
  113. if (getsockname(sock, (struct sockaddr *)&laddr, &len) == -1)
  114. error(-1, errno, "getsockname");
  115. fprintf(stderr, "server port: %d\n", ntohs(laddr.sin_port));
  116. child_pid = fork();
  117. if (!child_pid)
  118. client(ntohs(laddr.sin_port));
  119. else
  120. server(sock, laddr);
  121. return 0;
  122. }