The branch main has been updated by kevans:

URL: 
https://cgit.FreeBSD.org/src/commit/?id=a1174b3b1174754b1f69406bff4456d002e8f583

commit a1174b3b1174754b1f69406bff4456d002e8f583
Author:     Kyle Evans <kev...@freebsd.org>
AuthorDate: 2025-08-21 14:21:42 +0000
Commit:     Kyle Evans <kev...@freebsd.org>
CommitDate: 2025-08-21 14:22:44 +0000

    tun(4)/tap(4): allow devices to be configured as transient
    
    Transient tunnel devices are removed immediately after last close, so
    that an application that's created the tunnel could eliminate the need
    to manually destroy the tunnel whose lifetime it's already managing.
    
    Reviewed by:    zlei
    Differential Revision:  https://reviews.freebsd.org/D44200
---
 share/man/man4/tap.4             | 15 +++++++++++
 share/man/man4/tun.4             | 15 +++++++++++
 sys/net/if_clone.h               |  2 +-
 sys/net/if_tap.h                 |  2 ++
 sys/net/if_tun.h                 |  2 ++
 sys/net/if_tuntap.c              | 50 ++++++++++++++++++++++++++++++++++++-
 tests/sys/net/Makefile           |  1 +
 tests/sys/net/if_tun_test.sh     | 22 ++++++++++++++++
 tests/sys/net/transient_tuntap.c | 54 ++++++++++++++++++++++++++++++++++++++++
 9 files changed, 161 insertions(+), 2 deletions(-)

diff --git a/share/man/man4/tap.4 b/share/man/man4/tap.4
index 95a681a923d2..a4fe98cdfecf 100644
--- a/share/man/man4/tap.4
+++ b/share/man/man4/tap.4
@@ -203,6 +203,21 @@ The argument should be a pointer to a
 The interface name will be returned in the
 .Va ifr_name
 field.
+.It Dv TAPSTRANSIENT
+The argument should be a pointer to an
+.Va int ;
+this sets the transient flag on
+the
+.Nm
+device.
+A transient
+.Nm
+will be destroyed upon last close.
+.It Dv TAPGTRANSIENT
+The argument should be a pointer to an
+.Va int ;
+this stores the current state (enabled or disabled) of the transient flag into
+it.
 .It Dv FIONBIO
 Turn non-blocking I/O for reads off or on, according as the argument
 .Va int Ns 's
diff --git a/share/man/man4/tun.4 b/share/man/man4/tun.4
index 58f67cb20acb..1c5bd35f0ab8 100644
--- a/share/man/man4/tun.4
+++ b/share/man/man4/tun.4
@@ -282,6 +282,21 @@ The argument should be a pointer to an
 the ioctl sets the value to one if the device is in
 .Dq multi-af
 mode, and zero otherwise.
+.It Dv TUNSTRANSIENT
+The argument should be a pointer to an
+.Va int ;
+this sets the transient flag on
+the
+.Nm
+device.
+A transient
+.Nm
+will be destroyed upon last close.
+.It Dv TUNGTRANSIENT
+The argument should be a pointer to an
+.Va int ;
+this stores the current state (enabled or disabled) of the transient flag into
+it.
 .It Dv FIONBIO
 Turn non-blocking I/O for reads off or on, according as the argument
 .Vt int Ns 's
diff --git a/sys/net/if_clone.h b/sys/net/if_clone.h
index 5a74ffa1cc2f..d780e49af25f 100644
--- a/sys/net/if_clone.h
+++ b/sys/net/if_clone.h
@@ -153,7 +153,7 @@ int if_clone_destroy(const char *);
 int    if_clone_list(struct if_clonereq *);
 void   if_clone_restoregroup(struct ifnet *);
 
-/* The below interfaces are used only by epair(4). */
+/* The below interfaces are used only by epair(4) and tun(4)/tap(4). */
 void   if_clone_addif(struct if_clone *, struct ifnet *);
 int    if_clone_destroyif(struct if_clone *, struct ifnet *);
 
diff --git a/sys/net/if_tap.h b/sys/net/if_tap.h
index d84cd2eba6f3..8297b8d9e3d2 100644
--- a/sys/net/if_tap.h
+++ b/sys/net/if_tap.h
@@ -57,6 +57,8 @@
 #define        TAPGIFNAME              TUNGIFNAME
 #define        TAPSVNETHDR             _IOW('t', 91, int)
 #define        TAPGVNETHDR             _IOR('t', 94, int)
+#define        TAPSTRANSIENT           TUNSTRANSIENT
+#define        TAPGTRANSIENT           TUNGTRANSIENT
 
 /* VMware ioctl's */
 #define VMIO_SIOCSIFFLAGS      _IOWINT('V', 0)
diff --git a/sys/net/if_tun.h b/sys/net/if_tun.h
index a8fb61db45a2..ccdc25944823 100644
--- a/sys/net/if_tun.h
+++ b/sys/net/if_tun.h
@@ -43,5 +43,7 @@ struct tuninfo {
 #define        TUNSIFPID       _IO('t', 95)
 #define        TUNSIFHEAD      _IOW('t', 96, int)
 #define        TUNGIFHEAD      _IOR('t', 97, int)
+#define        TUNSTRANSIENT   _IOW('t', 98, int)
+#define        TUNGTRANSIENT   _IOR('t', 99, int)
 
 #endif /* !_NET_IF_TUN_H_ */
diff --git a/sys/net/if_tuntap.c b/sys/net/if_tuntap.c
index 275581ea2d4e..c8dbb6aa8893 100644
--- a/sys/net/if_tuntap.c
+++ b/sys/net/if_tuntap.c
@@ -132,6 +132,7 @@ struct tuntap_softc {
 #define        TUN_DYING       0x0200
 #define        TUN_L2          0x0400
 #define        TUN_VMNET       0x0800
+#define        TUN_TRANSIENT   0x1000
 
 #define        TUN_DRIVER_IDENT_MASK   (TUN_L2 | TUN_VMNET)
 #define        TUN_READY               (TUN_OPEN | TUN_INITED)
@@ -443,6 +444,18 @@ tuntap_name2info(const char *name, int *outunit, int 
*outflags)
        return (0);
 }
 
+static struct if_clone *
+tuntap_cloner_from_flags(int tun_flags)
+{
+
+       for (u_int i = 0; i < NDRV; i++)
+               if ((tun_flags & TUN_DRIVER_IDENT_MASK) ==
+                   tuntap_drivers[i].ident_flags)
+                       return (V_tuntap_driver_cloners[i]);
+
+       return (NULL);
+}
+
 /*
  * Get driver information from a set of flags specified.  Masks the identifying
  * part of the flags and compares it against all of the available
@@ -621,7 +634,12 @@ tun_destroy(struct tuntap_softc *tp, bool may_intr)
        int error;
 
        TUN_LOCK(tp);
-       MPASS((tp->tun_flags & TUN_DYING) == 0);
+
+       /*
+        * Transient tunnels may have set TUN_DYING if we're being destroyed as
+        * a result of the last close, which we'll allow.
+        */
+       MPASS((tp->tun_flags & (TUN_DYING | TUN_TRANSIENT)) != TUN_DYING);
        tp->tun_flags |= TUN_DYING;
        error = 0;
        while (tp->tun_busy != 0) {
@@ -1229,6 +1247,23 @@ out:
        tun_vnethdr_set(ifp, 0);
 
        tun_unbusy_locked(tp);
+       if ((tp->tun_flags & TUN_TRANSIENT) != 0) {
+               struct if_clone *cloner;
+               int error __diagused;
+
+               /* Mark it busy so that nothing can re-open it. */
+               tp->tun_flags |= TUN_DYING;
+               TUN_UNLOCK(tp);
+
+               CURVNET_SET_QUIET(ifp->if_home_vnet);
+               cloner = tuntap_cloner_from_flags(tp->tun_flags);
+               CURVNET_RESTORE();
+
+               error = if_clone_destroyif(cloner, ifp);
+               MPASS(error == 0 || error == EINTR || error == ERESTART);
+               return;
+       }
+
        TUN_UNLOCK(tp);
 }
 
@@ -1680,6 +1715,19 @@ tunioctl(struct cdev *dev, u_long cmd, caddr_t data, int 
flag,
        case TUNGDEBUG:
                *(int *)data = tundebug;
                break;
+       case TUNSTRANSIENT:
+               TUN_LOCK(tp);
+               if (*(int *)data)
+                       tp->tun_flags |= TUN_TRANSIENT;
+               else
+                       tp->tun_flags &= ~TUN_TRANSIENT;
+               TUN_UNLOCK(tp);
+               break;
+       case TUNGTRANSIENT:
+               TUN_LOCK(tp);
+               *(int *)data = (tp->tun_flags & TUN_TRANSIENT) != 0;
+               TUN_UNLOCK(tp);
+               break;
        case FIONBIO:
                break;
        case FIOASYNC:
diff --git a/tests/sys/net/Makefile b/tests/sys/net/Makefile
index 65cc99a3e932..e390c6e8059d 100644
--- a/tests/sys/net/Makefile
+++ b/tests/sys/net/Makefile
@@ -40,6 +40,7 @@ ${PACKAGE}FILESMODE_stp.py=           0555
 
 MAN=
 PROGS+=                randsleep
+PROGS+=                transient_tuntap
 
 CFLAGS+=        -I${.CURDIR:H:H}
 
diff --git a/tests/sys/net/if_tun_test.sh b/tests/sys/net/if_tun_test.sh
index a4ffe66e04ce..f4ce7800272e 100755
--- a/tests/sys/net/if_tun_test.sh
+++ b/tests/sys/net/if_tun_test.sh
@@ -56,8 +56,30 @@ basic_cleanup()
        vnet_cleanup
 }
 
+atf_test_case "transient" "cleanup"
+transient_head()
+{
+       atf_set descr "Test transient tunnel support"
+       atf_set require.user root
+}
+transient_body()
+{
+       vnet_init
+       vnet_mkjail one
+
+       tun=$(jexec one ifconfig tun create)
+       atf_check -s exit:0 -o not-empty jexec one ifconfig ${tun}
+       jexec one $(atf_get_srcdir)/transient_tuntap /dev/${tun}
+       atf_check -s not-exit:0 -e not-empty jexec one ifconfig ${tun}
+}
+transient_cleanup()
+{
+       vnet_cleanup
+}
+
 atf_init_test_cases()
 {
        atf_add_test_case "235704"
        atf_add_test_case "basic"
+       atf_add_test_case "transient"
 }
diff --git a/tests/sys/net/transient_tuntap.c b/tests/sys/net/transient_tuntap.c
new file mode 100644
index 000000000000..b0cf43064317
--- /dev/null
+++ b/tests/sys/net/transient_tuntap.c
@@ -0,0 +1,54 @@
+/*-
+ * Copyright (c) 2024 Kyle Evans <kev...@freebsd.org>
+ *
+ * SPDX-License-Identifier: BSD-2-Clause
+ */
+
+/*
+ * This test simply configures the tunnel as transient and exits.  By the time
+ * we return, the tunnel should be gone because the last reference disappears.
+ */
+
+#include <sys/types.h>
+#include <sys/ioctl.h>
+#include <net/if_tun.h>
+#include <net/if_tap.h>
+
+#include <assert.h>
+#include <err.h>
+#include <fcntl.h>
+#include <string.h>
+#include <unistd.h>
+
+int
+main(int argc, char *argv[])
+{
+       unsigned long tunreq;
+       const char *tundev;
+       int one = 1, tunfd;
+
+       assert(argc > 1);
+       tundev = argv[1];
+
+       tunfd = open(tundev, O_RDWR);
+       assert(tunfd >= 0);
+
+       /*
+        * These are technically the same request, but we'll use the technically
+        * correct one just in case.
+        */
+       if (strstr(tundev, "tun") != NULL) {
+               tunreq = TUNSTRANSIENT;
+       } else {
+               assert(strstr(tundev, "tap") != NULL);
+               tunreq = TAPSTRANSIENT;
+       }
+
+       if (ioctl(tunfd, tunreq, &one) == -1)
+               err(1, "ioctl");
+
+       /* Final close should destroy the tunnel automagically. */
+       close(tunfd);
+
+       return (0);
+}

Reply via email to