tun: Add ioctl() TUNGETDEVNETNS cmd to allow obtaining real net ns of tun device
authorKirill Tkhai <ktkhai@virtuozzo.com>
Wed, 20 Mar 2019 09:16:42 +0000 (12:16 +0300)
committerDavid S. Miller <davem@davemloft.net>
Thu, 21 Mar 2019 20:19:15 +0000 (13:19 -0700)
In commit f2780d6d7475 "tun: Add ioctl() SIOCGSKNS cmd to allow
obtaining net ns of tun device" it was missed that tun may change
its net ns, while net ns of socket remains the same as it was
created initially. SIOCGSKNS returns net ns of socket, so it is
not suitable for obtaining net ns of device.

We may have two tun devices with the same names in two net ns,
and in this case it's not possible to determ, which of them
fd refers to (TUNGETIFF will return the same name).

This patch adds new ioctl() cmd for obtaining net ns of a device.

Reported-by: Harald Albrecht <harald.albrecht@gmx.net>
Signed-off-by: Kirill Tkhai <ktkhai@virtuozzo.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/tun.c
include/uapi/linux/if_tun.h

index ef3b6551497663ed806bf097b7f832f21593a040..60d0a4e60ec5ef6e7b88fe33fd5d7722c74c35f3 100644 (file)
@@ -3102,6 +3102,7 @@ static long __tun_chr_ioctl(struct file *file, unsigned int cmd,
 
        tun_debug(KERN_INFO, tun, "tun_chr_ioctl cmd %u\n", cmd);
 
+       net = dev_net(tun->dev);
        ret = 0;
        switch (cmd) {
        case TUNGETIFF:
@@ -3327,6 +3328,13 @@ static long __tun_chr_ioctl(struct file *file, unsigned int cmd,
                ret = tun_net_change_carrier(tun->dev, (bool)carrier);
                break;
 
+       case TUNGETDEVNETNS:
+               ret = -EPERM;
+               if (!ns_capable(net->user_ns, CAP_NET_ADMIN))
+                       goto unlock;
+               ret = open_related_ns(&net->ns, get_net_ns);
+               break;
+
        default:
                ret = -EINVAL;
                break;
index 23a6753b37df3a0a241c2986aa932301bef02b1f..454ae31b93c7a4b4aeedddf7ceb5274377b39a58 100644 (file)
@@ -60,6 +60,7 @@
 #define TUNSETSTEERINGEBPF _IOR('T', 224, int)
 #define TUNSETFILTEREBPF _IOR('T', 225, int)
 #define TUNSETCARRIER _IOW('T', 226, int)
+#define TUNGETDEVNETNS _IO('T', 227)
 
 /* TUNSETIFF ifr flags */
 #define IFF_TUN                0x0001