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