Skip to content

Commit

Permalink
Merge branch 'bridge-ioctl-fixes'
Browse files Browse the repository at this point in the history
Nikolay Aleksandrov says:

====================
net: bridge: fix recent ioctl changes

These are three fixes for the recent bridge removal of ndo_do_ioctl
done by commit ad2f99a ("net: bridge: move bridge ioctls out of
.ndo_do_ioctl"). Patch 01 fixes a deadlock of the new bridge ioctl
hook lock and rtnl by taking a netdev reference and always taking the
bridge ioctl lock first then rtnl from within the bridge hook.
Patch 02 fixes old_deviceless() bridge calls device name argument, and
patch 03 checks in dev_ifsioc()'s SIOCBRADD/DELIF cases if the netdevice is
actually a bridge before interpreting its private ptr as net_bridge.

Patch 01 was tested by running old bridge-utils commands with lockdep
enabled. Patch 02 was tested again by using bridge-utils and using the
respective ioctl calls on a "up" bridge device. Patch 03 was tested by
using the addif ioctl on a non-bridge device (e.g. loopback).
====================

Signed-off-by: David S. Miller <[email protected]>
  • Loading branch information
davem330 committed Aug 5, 2021
2 parents 4167a96 + 9384eac commit d15040a
Show file tree
Hide file tree
Showing 3 changed files with 34 additions and 18 deletions.
4 changes: 1 addition & 3 deletions net/bridge/br_if.c
Original file line number Diff line number Diff line change
Expand Up @@ -456,7 +456,7 @@ int br_add_bridge(struct net *net, const char *name)
dev_net_set(dev, net);
dev->rtnl_link_ops = &br_link_ops;

res = register_netdev(dev);
res = register_netdevice(dev);
if (res)
free_netdev(dev);
return res;
Expand All @@ -467,7 +467,6 @@ int br_del_bridge(struct net *net, const char *name)
struct net_device *dev;
int ret = 0;

rtnl_lock();
dev = __dev_get_by_name(net, name);
if (dev == NULL)
ret = -ENXIO; /* Could not find device */
Expand All @@ -485,7 +484,6 @@ int br_del_bridge(struct net *net, const char *name)
else
br_dev_delete(dev, NULL);

rtnl_unlock();
return ret;
}

Expand Down
39 changes: 25 additions & 14 deletions net/bridge/br_ioctl.c
Original file line number Diff line number Diff line change
Expand Up @@ -351,7 +351,7 @@ static int old_deviceless(struct net *net, void __user *uarg)
if (!ns_capable(net->user_ns, CAP_NET_ADMIN))
return -EPERM;

if (copy_from_user(buf, uarg, IFNAMSIZ))
if (copy_from_user(buf, (void __user *)args[1], IFNAMSIZ))
return -EFAULT;

buf[IFNAMSIZ-1] = 0;
Expand All @@ -369,33 +369,44 @@ static int old_deviceless(struct net *net, void __user *uarg)
int br_ioctl_stub(struct net *net, struct net_bridge *br, unsigned int cmd,
struct ifreq *ifr, void __user *uarg)
{
int ret = -EOPNOTSUPP;

rtnl_lock();

switch (cmd) {
case SIOCGIFBR:
case SIOCSIFBR:
return old_deviceless(net, uarg);

ret = old_deviceless(net, uarg);
break;
case SIOCBRADDBR:
case SIOCBRDELBR:
{
char buf[IFNAMSIZ];

if (!ns_capable(net->user_ns, CAP_NET_ADMIN))
return -EPERM;
if (!ns_capable(net->user_ns, CAP_NET_ADMIN)) {
ret = -EPERM;
break;
}

if (copy_from_user(buf, uarg, IFNAMSIZ))
return -EFAULT;
if (copy_from_user(buf, uarg, IFNAMSIZ)) {
ret = -EFAULT;
break;
}

buf[IFNAMSIZ-1] = 0;
if (cmd == SIOCBRADDBR)
return br_add_bridge(net, buf);

return br_del_bridge(net, buf);
ret = br_add_bridge(net, buf);
else
ret = br_del_bridge(net, buf);
}

break;
case SIOCBRADDIF:
case SIOCBRDELIF:
return add_del_if(br, ifr->ifr_ifindex, cmd == SIOCBRADDIF);

ret = add_del_if(br, ifr->ifr_ifindex, cmd == SIOCBRADDIF);
break;
}
return -EOPNOTSUPP;

rtnl_unlock();

return ret;
}
9 changes: 8 additions & 1 deletion net/core/dev_ioctl.c
Original file line number Diff line number Diff line change
Expand Up @@ -379,7 +379,14 @@ static int dev_ifsioc(struct net *net, struct ifreq *ifr, void __user *data,
case SIOCBRDELIF:
if (!netif_device_present(dev))
return -ENODEV;
return br_ioctl_call(net, netdev_priv(dev), cmd, ifr, NULL);
if (!netif_is_bridge_master(dev))
return -EOPNOTSUPP;
dev_hold(dev);
rtnl_unlock();
err = br_ioctl_call(net, netdev_priv(dev), cmd, ifr, NULL);
dev_put(dev);
rtnl_lock();
return err;

case SIOCSHWTSTAMP:
err = net_hwtstamp_validate(ifr);
Expand Down

0 comments on commit d15040a

Please sign in to comment.