// SPDX-License-Identifier: GPL-2.0 #define _GNU_SOURCE #include <errno.h> #include <fcntl.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <linux/if.h> #include <linux/if_tun.h> #include <linux/netlink.h> #include <linux/rtnetlink.h> #include <sys/ioctl.h> #include <sys/socket.h> #include "../kselftest_harness.h" static int tun_attach(int fd, char *dev) { struct ifreq ifr; memset(&ifr, 0, sizeof(ifr)); strcpy(ifr.ifr_name, dev); ifr.ifr_flags = IFF_ATTACH_QUEUE; return ioctl(fd, TUNSETQUEUE, (void *) &ifr); } static int tun_detach(int fd, char *dev) { struct ifreq ifr; memset(&ifr, 0, sizeof(ifr)); strcpy(ifr.ifr_name, dev); ifr.ifr_flags = IFF_DETACH_QUEUE; return ioctl(fd, TUNSETQUEUE, (void *) &ifr); } static int tun_alloc(char *dev) { struct ifreq ifr; int fd, err; fd = open("/dev/net/tun", O_RDWR); if (fd < 0) { fprintf(stderr, "can't open tun: %s\n", strerror(errno)); return fd; } memset(&ifr, 0, sizeof(ifr)); strcpy(ifr.ifr_name, dev); ifr.ifr_flags = IFF_TAP | IFF_NAPI | IFF_MULTI_QUEUE; err = ioctl(fd, TUNSETIFF, (void *) &ifr); if (err < 0) { fprintf(stderr, "can't TUNSETIFF: %s\n", strerror(errno)); close(fd); return err; } strcpy(dev, ifr.ifr_name); return fd; } static int tun_delete(char *dev) { struct { struct nlmsghdr nh; struct ifinfomsg ifm; unsigned char data[64]; } req; struct rtattr *rta; int ret, rtnl; rtnl = socket(AF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE); if (rtnl < 0) { fprintf(stderr, "can't open rtnl: %s\n", strerror(errno)); return 1; } memset(&req, 0, sizeof(req)); req.nh.nlmsg_len = NLMSG_ALIGN(NLMSG_LENGTH(sizeof(req.ifm))); req.nh.nlmsg_flags = NLM_F_REQUEST; req.nh.nlmsg_type = RTM_DELLINK; req.ifm.ifi_family = AF_UNSPEC; rta = (struct rtattr *)(((char *)&req) + NLMSG_ALIGN(req.nh.nlmsg_len)); rta->rta_type = IFLA_IFNAME; rta->rta_len = RTA_LENGTH(IFNAMSIZ); req.nh.nlmsg_len += rta->rta_len; memcpy(RTA_DATA(rta), dev, IFNAMSIZ); ret = send(rtnl, &req, req.nh.nlmsg_len, 0); if (ret < 0) fprintf(stderr, "can't send: %s\n", strerror(errno)); ret = (unsigned int)ret != req.nh.nlmsg_len; close(rtnl); return ret; } FIXTURE(tun) { char ifname[IFNAMSIZ]; int fd, fd2; }; FIXTURE_SETUP(tun) { memset(self->ifname, 0, sizeof(self->ifname)); self->fd = tun_alloc(self->ifname); ASSERT_GE(self->fd, 0); self->fd2 = tun_alloc(self->ifname); ASSERT_GE(self->fd2, 0); } FIXTURE_TEARDOWN(tun) { if (self->fd >= 0) close(self->fd); if (self->fd2 >= 0) close(self->fd2); } TEST_F(tun, delete_detach_close) { EXPECT_EQ(tun_delete(self->ifname), 0); EXPECT_EQ(tun_detach(self->fd, self->ifname), -1); EXPECT_EQ(errno, 22); } TEST_F(tun, detach_delete_close) { EXPECT_EQ(tun_detach(self->fd, self->ifname), 0); EXPECT_EQ(tun_delete(self->ifname), 0); } TEST_F(tun, detach_close_delete) { EXPECT_EQ(tun_detach(self->fd, self->ifname), 0); close(self->fd); self->fd = -1; EXPECT_EQ(tun_delete(self->ifname), 0); } TEST_F(tun, reattach_delete_close) { EXPECT_EQ(tun_detach(self->fd, self->ifname), 0); EXPECT_EQ(tun_attach(self->fd, self->ifname), 0); EXPECT_EQ(tun_delete(self->ifname), 0); } TEST_F(tun, reattach_close_delete) { EXPECT_EQ(tun_detach(self->fd, self->ifname), 0); EXPECT_EQ(tun_attach(self->fd, self->ifname), 0); close(self->fd); self->fd = -1; EXPECT_EQ(tun_delete(self->ifname), 0); } TEST_HARNESS_MAIN