// SPDX-License-Identifier: GPL-2.0 #include #include #include #include #include #include #include #include #include #include #include #include #include static int cfg_server; static int cfg_client; static int cfg_port = 8000; static struct sockaddr_in6 cfg_addr; static char *cfg_outfile; static int parse_address(const char *str, int port, struct sockaddr_in6 *sin6) { int ret; sin6->sin6_family = AF_INET6; sin6->sin6_port = htons(port); ret = inet_pton(sin6->sin6_family, str, &sin6->sin6_addr); if (ret != 1) { /* fallback to plain IPv4 */ ret = inet_pton(AF_INET, str, &sin6->sin6_addr.s6_addr32[3]); if (ret != 1) return -1; /* add ::ffff prefix */ sin6->sin6_addr.s6_addr32[0] = 0; sin6->sin6_addr.s6_addr32[1] = 0; sin6->sin6_addr.s6_addr16[4] = 0; sin6->sin6_addr.s6_addr16[5] = 0xffff; } return 0; } static void run_server(void) { unsigned long qlen = 32; int fd, opt, connfd; socklen_t len; char buf[64]; FILE *outfile; outfile = fopen(cfg_outfile, "w"); if (!outfile) error(1, errno, "fopen() outfile"); fd = socket(AF_INET6, SOCK_STREAM, 0); if (fd == -1) error(1, errno, "socket()"); opt = 1; if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) < 0) error(1, errno, "setsockopt(SO_REUSEADDR)"); if (setsockopt(fd, SOL_TCP, TCP_FASTOPEN, &qlen, sizeof(qlen)) < 0) error(1, errno, "setsockopt(TCP_FASTOPEN)"); if (bind(fd, (struct sockaddr *)&cfg_addr, sizeof(cfg_addr)) < 0) error(1, errno, "bind()"); if (listen(fd, 5) < 0) error(1, errno, "listen()"); len = sizeof(cfg_addr); connfd = accept(fd, (struct sockaddr *)&cfg_addr, &len); if (connfd < 0) error(1, errno, "accept()"); len = sizeof(opt); if (getsockopt(connfd, SOL_SOCKET, SO_INCOMING_NAPI_ID, &opt, &len) < 0) error(1, errno, "getsockopt(SO_INCOMING_NAPI_ID)"); read(connfd, buf, 64); fprintf(outfile, "%d\n", opt); fclose(outfile); close(connfd); close(fd); } static void run_client(void) { int fd; char *msg = "Hello, world!"; fd = socket(AF_INET6, SOCK_STREAM, 0); if (fd == -1) error(1, errno, "socket()"); sendto(fd, msg, strlen(msg), MSG_FASTOPEN, (struct sockaddr *)&cfg_addr, sizeof(cfg_addr)); close(fd); } static void usage(const char *filepath) { error(1, 0, "Usage: %s (-s|-c) -h -p -o ", filepath); } static void parse_opts(int argc, char **argv) { struct sockaddr_in6 *addr6 = (void *) &cfg_addr; char *addr = NULL; int ret; int c; if (argc <= 1) usage(argv[0]); while ((c = getopt(argc, argv, "sch:p:o:")) != -1) { switch (c) { case 's': if (cfg_client) error(1, 0, "Pass one of -s or -c"); cfg_server = 1; break; case 'c': if (cfg_server) error(1, 0, "Pass one of -s or -c"); cfg_client = 1; break; case 'h': addr = optarg; break; case 'p': cfg_port = strtoul(optarg, NULL, 0); break; case 'o': cfg_outfile = strdup(optarg); if (!cfg_outfile) error(1, 0, "outfile invalid"); break; } } if (cfg_server && addr) error(1, 0, "Server cannot have -h specified"); memset(addr6, 0, sizeof(*addr6)); addr6->sin6_family = AF_INET6; addr6->sin6_port = htons(cfg_port); addr6->sin6_addr = in6addr_any; if (addr) { ret = parse_address(addr, cfg_port, addr6); if (ret) error(1, 0, "Client address parse error: %s", addr); } } int main(int argc, char **argv) { parse_opts(argc, argv); if (cfg_server) run_server(); else if (cfg_client) run_client(); return 0; }