xfrm interface: fix management of phydev

With the current implementation, phydev cannot be removed:

$ ip link add dummy type dummy
$ ip link add xfrm1 type xfrm dev dummy if_id 1
$ ip l d dummy
 kernel:[77938.465445] unregister_netdevice: waiting for dummy to become free. Usage count = 1

Manage it like in ip tunnels, ie just keep the ifindex. Not that the side
effect, is that the phydev is now optional.

Fixes: f203b76d78 ("xfrm: Add virtual xfrm interfaces")
Signed-off-by: Nicolas Dichtel <nicolas.dichtel@6wind.com>
Tested-by: Julien Floret <julien.floret@6wind.com>
Signed-off-by: Steffen Klassert <steffen.klassert@secunet.com>
This commit is contained in:
Nicolas Dichtel 2019-07-15 12:00:23 +02:00 committed by Steffen Klassert
parent c5d1030f23
commit 22d6552f82
2 changed files with 17 additions and 16 deletions

View File

@ -990,7 +990,6 @@ struct xfrm_if_parms {
struct xfrm_if { struct xfrm_if {
struct xfrm_if __rcu *next; /* next interface in list */ struct xfrm_if __rcu *next; /* next interface in list */
struct net_device *dev; /* virtual device associated with interface */ struct net_device *dev; /* virtual device associated with interface */
struct net_device *phydev; /* physical device */
struct net *net; /* netns for packet i/o */ struct net *net; /* netns for packet i/o */
struct xfrm_if_parms p; /* interface parms */ struct xfrm_if_parms p; /* interface parms */

View File

@ -175,7 +175,6 @@ static void xfrmi_dev_uninit(struct net_device *dev)
struct xfrmi_net *xfrmn = net_generic(xi->net, xfrmi_net_id); struct xfrmi_net *xfrmn = net_generic(xi->net, xfrmi_net_id);
xfrmi_unlink(xfrmn, xi); xfrmi_unlink(xfrmn, xi);
dev_put(xi->phydev);
dev_put(dev); dev_put(dev);
} }
@ -362,7 +361,7 @@ static netdev_tx_t xfrmi_xmit(struct sk_buff *skb, struct net_device *dev)
goto tx_err; goto tx_err;
} }
fl.flowi_oif = xi->phydev->ifindex; fl.flowi_oif = xi->p.link;
ret = xfrmi_xmit2(skb, dev, &fl); ret = xfrmi_xmit2(skb, dev, &fl);
if (ret < 0) if (ret < 0)
@ -548,7 +547,7 @@ static int xfrmi_get_iflink(const struct net_device *dev)
{ {
struct xfrm_if *xi = netdev_priv(dev); struct xfrm_if *xi = netdev_priv(dev);
return xi->phydev->ifindex; return xi->p.link;
} }
@ -574,12 +573,14 @@ static void xfrmi_dev_setup(struct net_device *dev)
dev->needs_free_netdev = true; dev->needs_free_netdev = true;
dev->priv_destructor = xfrmi_dev_free; dev->priv_destructor = xfrmi_dev_free;
netif_keep_dst(dev); netif_keep_dst(dev);
eth_broadcast_addr(dev->broadcast);
} }
static int xfrmi_dev_init(struct net_device *dev) static int xfrmi_dev_init(struct net_device *dev)
{ {
struct xfrm_if *xi = netdev_priv(dev); struct xfrm_if *xi = netdev_priv(dev);
struct net_device *phydev = xi->phydev; struct net_device *phydev = __dev_get_by_index(xi->net, xi->p.link);
int err; int err;
dev->tstats = netdev_alloc_pcpu_stats(struct pcpu_sw_netstats); dev->tstats = netdev_alloc_pcpu_stats(struct pcpu_sw_netstats);
@ -594,13 +595,19 @@ static int xfrmi_dev_init(struct net_device *dev)
dev->features |= NETIF_F_LLTX; dev->features |= NETIF_F_LLTX;
dev->needed_headroom = phydev->needed_headroom; if (phydev) {
dev->needed_tailroom = phydev->needed_tailroom; dev->needed_headroom = phydev->needed_headroom;
dev->needed_tailroom = phydev->needed_tailroom;
if (is_zero_ether_addr(dev->dev_addr)) if (is_zero_ether_addr(dev->dev_addr))
eth_hw_addr_inherit(dev, phydev); eth_hw_addr_inherit(dev, phydev);
if (is_zero_ether_addr(dev->broadcast)) if (is_zero_ether_addr(dev->broadcast))
memcpy(dev->broadcast, phydev->broadcast, dev->addr_len); memcpy(dev->broadcast, phydev->broadcast,
dev->addr_len);
} else {
eth_hw_addr_random(dev);
eth_broadcast_addr(dev->broadcast);
}
return 0; return 0;
} }
@ -644,13 +651,8 @@ static int xfrmi_newlink(struct net *src_net, struct net_device *dev,
xi->p = p; xi->p = p;
xi->net = net; xi->net = net;
xi->dev = dev; xi->dev = dev;
xi->phydev = dev_get_by_index(net, p.link);
if (!xi->phydev)
return -ENODEV;
err = xfrmi_create(dev); err = xfrmi_create(dev);
if (err < 0)
dev_put(xi->phydev);
return err; return err;
} }