18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright 2018 Google Inc. 48c2ecf20Sopenharmony_ci * Author: Soheil Hassas Yeganeh (soheil@google.com) 58c2ecf20Sopenharmony_ci * 68c2ecf20Sopenharmony_ci * Simple example on how to use TCP_INQ and TCP_CM_INQ. 78c2ecf20Sopenharmony_ci */ 88c2ecf20Sopenharmony_ci#define _GNU_SOURCE 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <error.h> 118c2ecf20Sopenharmony_ci#include <netinet/in.h> 128c2ecf20Sopenharmony_ci#include <netinet/tcp.h> 138c2ecf20Sopenharmony_ci#include <pthread.h> 148c2ecf20Sopenharmony_ci#include <stdio.h> 158c2ecf20Sopenharmony_ci#include <errno.h> 168c2ecf20Sopenharmony_ci#include <stdlib.h> 178c2ecf20Sopenharmony_ci#include <string.h> 188c2ecf20Sopenharmony_ci#include <sys/socket.h> 198c2ecf20Sopenharmony_ci#include <unistd.h> 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_ci#ifndef TCP_INQ 228c2ecf20Sopenharmony_ci#define TCP_INQ 36 238c2ecf20Sopenharmony_ci#endif 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_ci#ifndef TCP_CM_INQ 268c2ecf20Sopenharmony_ci#define TCP_CM_INQ TCP_INQ 278c2ecf20Sopenharmony_ci#endif 288c2ecf20Sopenharmony_ci 298c2ecf20Sopenharmony_ci#define BUF_SIZE 8192 308c2ecf20Sopenharmony_ci#define CMSG_SIZE 32 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_cistatic int family = AF_INET6; 338c2ecf20Sopenharmony_cistatic socklen_t addr_len = sizeof(struct sockaddr_in6); 348c2ecf20Sopenharmony_cistatic int port = 4974; 358c2ecf20Sopenharmony_ci 368c2ecf20Sopenharmony_cistatic void setup_loopback_addr(int family, struct sockaddr_storage *sockaddr) 378c2ecf20Sopenharmony_ci{ 388c2ecf20Sopenharmony_ci struct sockaddr_in6 *addr6 = (void *) sockaddr; 398c2ecf20Sopenharmony_ci struct sockaddr_in *addr4 = (void *) sockaddr; 408c2ecf20Sopenharmony_ci 418c2ecf20Sopenharmony_ci switch (family) { 428c2ecf20Sopenharmony_ci case PF_INET: 438c2ecf20Sopenharmony_ci memset(addr4, 0, sizeof(*addr4)); 448c2ecf20Sopenharmony_ci addr4->sin_family = AF_INET; 458c2ecf20Sopenharmony_ci addr4->sin_addr.s_addr = htonl(INADDR_LOOPBACK); 468c2ecf20Sopenharmony_ci addr4->sin_port = htons(port); 478c2ecf20Sopenharmony_ci break; 488c2ecf20Sopenharmony_ci case PF_INET6: 498c2ecf20Sopenharmony_ci memset(addr6, 0, sizeof(*addr6)); 508c2ecf20Sopenharmony_ci addr6->sin6_family = AF_INET6; 518c2ecf20Sopenharmony_ci addr6->sin6_addr = in6addr_loopback; 528c2ecf20Sopenharmony_ci addr6->sin6_port = htons(port); 538c2ecf20Sopenharmony_ci break; 548c2ecf20Sopenharmony_ci default: 558c2ecf20Sopenharmony_ci error(1, 0, "illegal family"); 568c2ecf20Sopenharmony_ci } 578c2ecf20Sopenharmony_ci} 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_civoid *start_server(void *arg) 608c2ecf20Sopenharmony_ci{ 618c2ecf20Sopenharmony_ci int server_fd = (int)(unsigned long)arg; 628c2ecf20Sopenharmony_ci struct sockaddr_in addr; 638c2ecf20Sopenharmony_ci socklen_t addrlen = sizeof(addr); 648c2ecf20Sopenharmony_ci char *buf; 658c2ecf20Sopenharmony_ci int fd; 668c2ecf20Sopenharmony_ci int r; 678c2ecf20Sopenharmony_ci 688c2ecf20Sopenharmony_ci buf = malloc(BUF_SIZE); 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_ci for (;;) { 718c2ecf20Sopenharmony_ci fd = accept(server_fd, (struct sockaddr *)&addr, &addrlen); 728c2ecf20Sopenharmony_ci if (fd == -1) { 738c2ecf20Sopenharmony_ci perror("accept"); 748c2ecf20Sopenharmony_ci break; 758c2ecf20Sopenharmony_ci } 768c2ecf20Sopenharmony_ci do { 778c2ecf20Sopenharmony_ci r = send(fd, buf, BUF_SIZE, 0); 788c2ecf20Sopenharmony_ci } while (r < 0 && errno == EINTR); 798c2ecf20Sopenharmony_ci if (r < 0) 808c2ecf20Sopenharmony_ci perror("send"); 818c2ecf20Sopenharmony_ci if (r != BUF_SIZE) 828c2ecf20Sopenharmony_ci fprintf(stderr, "can only send %d bytes\n", r); 838c2ecf20Sopenharmony_ci /* TCP_INQ can overestimate in-queue by one byte if we send 848c2ecf20Sopenharmony_ci * the FIN packet. Sleep for 1 second, so that the client 858c2ecf20Sopenharmony_ci * likely invoked recvmsg(). 868c2ecf20Sopenharmony_ci */ 878c2ecf20Sopenharmony_ci sleep(1); 888c2ecf20Sopenharmony_ci close(fd); 898c2ecf20Sopenharmony_ci } 908c2ecf20Sopenharmony_ci 918c2ecf20Sopenharmony_ci free(buf); 928c2ecf20Sopenharmony_ci close(server_fd); 938c2ecf20Sopenharmony_ci pthread_exit(0); 948c2ecf20Sopenharmony_ci} 958c2ecf20Sopenharmony_ci 968c2ecf20Sopenharmony_ciint main(int argc, char *argv[]) 978c2ecf20Sopenharmony_ci{ 988c2ecf20Sopenharmony_ci struct sockaddr_storage listen_addr, addr; 998c2ecf20Sopenharmony_ci int c, one = 1, inq = -1; 1008c2ecf20Sopenharmony_ci pthread_t server_thread; 1018c2ecf20Sopenharmony_ci char cmsgbuf[CMSG_SIZE]; 1028c2ecf20Sopenharmony_ci struct iovec iov[1]; 1038c2ecf20Sopenharmony_ci struct cmsghdr *cm; 1048c2ecf20Sopenharmony_ci struct msghdr msg; 1058c2ecf20Sopenharmony_ci int server_fd, fd; 1068c2ecf20Sopenharmony_ci char *buf; 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci while ((c = getopt(argc, argv, "46p:")) != -1) { 1098c2ecf20Sopenharmony_ci switch (c) { 1108c2ecf20Sopenharmony_ci case '4': 1118c2ecf20Sopenharmony_ci family = PF_INET; 1128c2ecf20Sopenharmony_ci addr_len = sizeof(struct sockaddr_in); 1138c2ecf20Sopenharmony_ci break; 1148c2ecf20Sopenharmony_ci case '6': 1158c2ecf20Sopenharmony_ci family = PF_INET6; 1168c2ecf20Sopenharmony_ci addr_len = sizeof(struct sockaddr_in6); 1178c2ecf20Sopenharmony_ci break; 1188c2ecf20Sopenharmony_ci case 'p': 1198c2ecf20Sopenharmony_ci port = atoi(optarg); 1208c2ecf20Sopenharmony_ci break; 1218c2ecf20Sopenharmony_ci } 1228c2ecf20Sopenharmony_ci } 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_ci server_fd = socket(family, SOCK_STREAM, 0); 1258c2ecf20Sopenharmony_ci if (server_fd < 0) 1268c2ecf20Sopenharmony_ci error(1, errno, "server socket"); 1278c2ecf20Sopenharmony_ci setup_loopback_addr(family, &listen_addr); 1288c2ecf20Sopenharmony_ci if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, 1298c2ecf20Sopenharmony_ci &one, sizeof(one)) != 0) 1308c2ecf20Sopenharmony_ci error(1, errno, "setsockopt(SO_REUSEADDR)"); 1318c2ecf20Sopenharmony_ci if (bind(server_fd, (const struct sockaddr *)&listen_addr, 1328c2ecf20Sopenharmony_ci addr_len) == -1) 1338c2ecf20Sopenharmony_ci error(1, errno, "bind"); 1348c2ecf20Sopenharmony_ci if (listen(server_fd, 128) == -1) 1358c2ecf20Sopenharmony_ci error(1, errno, "listen"); 1368c2ecf20Sopenharmony_ci if (pthread_create(&server_thread, NULL, start_server, 1378c2ecf20Sopenharmony_ci (void *)(unsigned long)server_fd) != 0) 1388c2ecf20Sopenharmony_ci error(1, errno, "pthread_create"); 1398c2ecf20Sopenharmony_ci 1408c2ecf20Sopenharmony_ci fd = socket(family, SOCK_STREAM, 0); 1418c2ecf20Sopenharmony_ci if (fd < 0) 1428c2ecf20Sopenharmony_ci error(1, errno, "client socket"); 1438c2ecf20Sopenharmony_ci setup_loopback_addr(family, &addr); 1448c2ecf20Sopenharmony_ci if (connect(fd, (const struct sockaddr *)&addr, addr_len) == -1) 1458c2ecf20Sopenharmony_ci error(1, errno, "connect"); 1468c2ecf20Sopenharmony_ci if (setsockopt(fd, SOL_TCP, TCP_INQ, &one, sizeof(one)) != 0) 1478c2ecf20Sopenharmony_ci error(1, errno, "setsockopt(TCP_INQ)"); 1488c2ecf20Sopenharmony_ci 1498c2ecf20Sopenharmony_ci msg.msg_name = NULL; 1508c2ecf20Sopenharmony_ci msg.msg_namelen = 0; 1518c2ecf20Sopenharmony_ci msg.msg_iov = iov; 1528c2ecf20Sopenharmony_ci msg.msg_iovlen = 1; 1538c2ecf20Sopenharmony_ci msg.msg_control = cmsgbuf; 1548c2ecf20Sopenharmony_ci msg.msg_controllen = sizeof(cmsgbuf); 1558c2ecf20Sopenharmony_ci msg.msg_flags = 0; 1568c2ecf20Sopenharmony_ci 1578c2ecf20Sopenharmony_ci buf = malloc(BUF_SIZE); 1588c2ecf20Sopenharmony_ci iov[0].iov_base = buf; 1598c2ecf20Sopenharmony_ci iov[0].iov_len = BUF_SIZE / 2; 1608c2ecf20Sopenharmony_ci 1618c2ecf20Sopenharmony_ci if (recvmsg(fd, &msg, 0) != iov[0].iov_len) 1628c2ecf20Sopenharmony_ci error(1, errno, "recvmsg"); 1638c2ecf20Sopenharmony_ci if (msg.msg_flags & MSG_CTRUNC) 1648c2ecf20Sopenharmony_ci error(1, 0, "control message is truncated"); 1658c2ecf20Sopenharmony_ci 1668c2ecf20Sopenharmony_ci for (cm = CMSG_FIRSTHDR(&msg); cm; cm = CMSG_NXTHDR(&msg, cm)) 1678c2ecf20Sopenharmony_ci if (cm->cmsg_level == SOL_TCP && cm->cmsg_type == TCP_CM_INQ) 1688c2ecf20Sopenharmony_ci inq = *((int *) CMSG_DATA(cm)); 1698c2ecf20Sopenharmony_ci 1708c2ecf20Sopenharmony_ci if (inq != BUF_SIZE - iov[0].iov_len) { 1718c2ecf20Sopenharmony_ci fprintf(stderr, "unexpected inq: %d\n", inq); 1728c2ecf20Sopenharmony_ci exit(1); 1738c2ecf20Sopenharmony_ci } 1748c2ecf20Sopenharmony_ci 1758c2ecf20Sopenharmony_ci printf("PASSED\n"); 1768c2ecf20Sopenharmony_ci free(buf); 1778c2ecf20Sopenharmony_ci close(fd); 1788c2ecf20Sopenharmony_ci return 0; 1798c2ecf20Sopenharmony_ci} 180