| /* |
| * Copyright 2018 Google Inc. |
| * Author: Soheil Hassas Yeganeh (soheil@google.com) |
| * |
| * Simple example on how to use TCP_INQ and TCP_CM_INQ. |
| * |
| * License (GPLv2): |
| * |
| * This program is free software; you can redistribute it and/or modify it |
| * under the terms and conditions of the GNU General Public License, |
| * version 2, as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. * See the GNU General Public License for |
| * more details. |
| */ |
| #define _GNU_SOURCE |
| |
| #include <error.h> |
| #include <netinet/in.h> |
| #include <netinet/tcp.h> |
| #include <pthread.h> |
| #include <stdio.h> |
| #include <errno.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <sys/socket.h> |
| #include <unistd.h> |
| |
| #ifndef TCP_INQ |
| #define TCP_INQ 36 |
| #endif |
| |
| #ifndef TCP_CM_INQ |
| #define TCP_CM_INQ TCP_INQ |
| #endif |
| |
| #define BUF_SIZE 8192 |
| #define CMSG_SIZE 32 |
| |
| static int family = AF_INET6; |
| static socklen_t addr_len = sizeof(struct sockaddr_in6); |
| static int port = 4974; |
| |
| static void setup_loopback_addr(int family, struct sockaddr_storage *sockaddr) |
| { |
| struct sockaddr_in6 *addr6 = (void *) sockaddr; |
| struct sockaddr_in *addr4 = (void *) sockaddr; |
| |
| switch (family) { |
| case PF_INET: |
| memset(addr4, 0, sizeof(*addr4)); |
| addr4->sin_family = AF_INET; |
| addr4->sin_addr.s_addr = htonl(INADDR_LOOPBACK); |
| addr4->sin_port = htons(port); |
| break; |
| case PF_INET6: |
| memset(addr6, 0, sizeof(*addr6)); |
| addr6->sin6_family = AF_INET6; |
| addr6->sin6_addr = in6addr_loopback; |
| addr6->sin6_port = htons(port); |
| break; |
| default: |
| error(1, 0, "illegal family"); |
| } |
| } |
| |
| void *start_server(void *arg) |
| { |
| int server_fd = (int)(unsigned long)arg; |
| struct sockaddr_in addr; |
| socklen_t addrlen = sizeof(addr); |
| char *buf; |
| int fd; |
| int r; |
| |
| buf = malloc(BUF_SIZE); |
| |
| for (;;) { |
| fd = accept(server_fd, (struct sockaddr *)&addr, &addrlen); |
| if (fd == -1) { |
| perror("accept"); |
| break; |
| } |
| do { |
| r = send(fd, buf, BUF_SIZE, 0); |
| } while (r < 0 && errno == EINTR); |
| if (r < 0) |
| perror("send"); |
| if (r != BUF_SIZE) |
| fprintf(stderr, "can only send %d bytes\n", r); |
| /* TCP_INQ can overestimate in-queue by one byte if we send |
| * the FIN packet. Sleep for 1 second, so that the client |
| * likely invoked recvmsg(). |
| */ |
| sleep(1); |
| close(fd); |
| } |
| |
| free(buf); |
| close(server_fd); |
| pthread_exit(0); |
| } |
| |
| int main(int argc, char *argv[]) |
| { |
| struct sockaddr_storage listen_addr, addr; |
| int c, one = 1, inq = -1; |
| pthread_t server_thread; |
| char cmsgbuf[CMSG_SIZE]; |
| struct iovec iov[1]; |
| struct cmsghdr *cm; |
| struct msghdr msg; |
| int server_fd, fd; |
| char *buf; |
| |
| while ((c = getopt(argc, argv, "46p:")) != -1) { |
| switch (c) { |
| case '4': |
| family = PF_INET; |
| addr_len = sizeof(struct sockaddr_in); |
| break; |
| case '6': |
| family = PF_INET6; |
| addr_len = sizeof(struct sockaddr_in6); |
| break; |
| case 'p': |
| port = atoi(optarg); |
| break; |
| } |
| } |
| |
| server_fd = socket(family, SOCK_STREAM, 0); |
| if (server_fd < 0) |
| error(1, errno, "server socket"); |
| setup_loopback_addr(family, &listen_addr); |
| if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR, |
| &one, sizeof(one)) != 0) |
| error(1, errno, "setsockopt(SO_REUSEADDR)"); |
| if (bind(server_fd, (const struct sockaddr *)&listen_addr, |
| addr_len) == -1) |
| error(1, errno, "bind"); |
| if (listen(server_fd, 128) == -1) |
| error(1, errno, "listen"); |
| if (pthread_create(&server_thread, NULL, start_server, |
| (void *)(unsigned long)server_fd) != 0) |
| error(1, errno, "pthread_create"); |
| |
| fd = socket(family, SOCK_STREAM, 0); |
| if (fd < 0) |
| error(1, errno, "client socket"); |
| setup_loopback_addr(family, &addr); |
| if (connect(fd, (const struct sockaddr *)&addr, addr_len) == -1) |
| error(1, errno, "connect"); |
| if (setsockopt(fd, SOL_TCP, TCP_INQ, &one, sizeof(one)) != 0) |
| error(1, errno, "setsockopt(TCP_INQ)"); |
| |
| msg.msg_name = NULL; |
| msg.msg_namelen = 0; |
| msg.msg_iov = iov; |
| msg.msg_iovlen = 1; |
| msg.msg_control = cmsgbuf; |
| msg.msg_controllen = sizeof(cmsgbuf); |
| msg.msg_flags = 0; |
| |
| buf = malloc(BUF_SIZE); |
| iov[0].iov_base = buf; |
| iov[0].iov_len = BUF_SIZE / 2; |
| |
| if (recvmsg(fd, &msg, 0) != iov[0].iov_len) |
| error(1, errno, "recvmsg"); |
| if (msg.msg_flags & MSG_CTRUNC) |
| error(1, 0, "control message is truncated"); |
| |
| for (cm = CMSG_FIRSTHDR(&msg); cm; cm = CMSG_NXTHDR(&msg, cm)) |
| if (cm->cmsg_level == SOL_TCP && cm->cmsg_type == TCP_CM_INQ) |
| inq = *((int *) CMSG_DATA(cm)); |
| |
| if (inq != BUF_SIZE - iov[0].iov_len) { |
| fprintf(stderr, "unexpected inq: %d\n", inq); |
| exit(1); |
| } |
| |
| printf("PASSED\n"); |
| free(buf); |
| close(fd); |
| return 0; |
| } |