Commit 34596a86 authored by Vlad Yasevich's avatar Vlad Yasevich Committed by Jiri Slaby

macvtap: Fix race between device delete and open.

[ Upstream commit 40b8fe45 ]

In macvtap device delete and open calls can race and
this causes a list curruption of the vlan queue_list.

The race intself is triggered by the idr accessors
that located the vlan device.  The device is stored
into and removed from the idr under both an rtnl and
a mutex.  However, when attempting to locate the device
in idr, only a mutex is taken.  As a result, once cpu
perfoming a delete may take an rtnl and wait for the mutex,
while another cput doing an open() will take the idr
mutex first to fetch the device pointer and later take
an rtnl to add a queue for the device which may have
just gotten deleted.

With this patch, we now hold the rtnl for the duration
of the macvtap_open() call thus making sure that
open will not race with delete.

CC: Michael S. Tsirkin <mst@redhat.com>
CC: Jason Wang <jasowang@redhat.com>
Signed-off-by: default avatarVladislav Yasevich <vyasevic@redhat.com>
Acked-by: default avatarJason Wang <jasowang@redhat.com>
Acked-by: default avatarMichael S. Tsirkin <mst@redhat.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarJiri Slaby <jslaby@suse.cz>
parent 2a9e5d6b
...@@ -108,17 +108,15 @@ static int macvtap_enable_queue(struct net_device *dev, struct file *file, ...@@ -108,17 +108,15 @@ static int macvtap_enable_queue(struct net_device *dev, struct file *file,
return err; return err;
} }
/* Requires RTNL */
static int macvtap_set_queue(struct net_device *dev, struct file *file, static int macvtap_set_queue(struct net_device *dev, struct file *file,
struct macvtap_queue *q) struct macvtap_queue *q)
{ {
struct macvlan_dev *vlan = netdev_priv(dev); struct macvlan_dev *vlan = netdev_priv(dev);
int err = -EBUSY;
rtnl_lock();
if (vlan->numqueues == MAX_MACVTAP_QUEUES) if (vlan->numqueues == MAX_MACVTAP_QUEUES)
goto out; return -EBUSY;
err = 0;
rcu_assign_pointer(q->vlan, vlan); rcu_assign_pointer(q->vlan, vlan);
rcu_assign_pointer(vlan->taps[vlan->numvtaps], q); rcu_assign_pointer(vlan->taps[vlan->numvtaps], q);
sock_hold(&q->sk); sock_hold(&q->sk);
...@@ -132,9 +130,7 @@ static int macvtap_set_queue(struct net_device *dev, struct file *file, ...@@ -132,9 +130,7 @@ static int macvtap_set_queue(struct net_device *dev, struct file *file,
vlan->numvtaps++; vlan->numvtaps++;
vlan->numqueues++; vlan->numqueues++;
out: return 0;
rtnl_unlock();
return err;
} }
static int macvtap_disable_queue(struct macvtap_queue *q) static int macvtap_disable_queue(struct macvtap_queue *q)
...@@ -450,11 +446,12 @@ static void macvtap_sock_destruct(struct sock *sk) ...@@ -450,11 +446,12 @@ static void macvtap_sock_destruct(struct sock *sk)
static int macvtap_open(struct inode *inode, struct file *file) static int macvtap_open(struct inode *inode, struct file *file)
{ {
struct net *net = current->nsproxy->net_ns; struct net *net = current->nsproxy->net_ns;
struct net_device *dev = dev_get_by_macvtap_minor(iminor(inode)); struct net_device *dev;
struct macvtap_queue *q; struct macvtap_queue *q;
int err; int err = -ENODEV;
err = -ENODEV; rtnl_lock();
dev = dev_get_by_macvtap_minor(iminor(inode));
if (!dev) if (!dev)
goto out; goto out;
...@@ -494,6 +491,7 @@ static int macvtap_open(struct inode *inode, struct file *file) ...@@ -494,6 +491,7 @@ static int macvtap_open(struct inode *inode, struct file *file)
if (dev) if (dev)
dev_put(dev); dev_put(dev);
rtnl_unlock();
return err; return err;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment