Commit 9ab179d8 authored by David Ahern's avatar David Ahern Committed by David S. Miller

net: vrf: Fix dst reference counting

Vivek reported a kernel exception deleting a VRF with an active
connection through it. The root cause is that the socket has a cached
reference to a dst that is destroyed. Converting the dst_destroy to
dst_release and letting proper reference counting kick in does not
work as the dst has a reference to the device which needs to be released
as well.

I talked to Hannes about this at netdev and he pointed out the ipv4 and
ipv6 dst handling has dst_ifdown for just this scenario. Rather than
continuing with the reinvented dst wheel in VRF just remove it and
leverage the ipv4 and ipv6 versions.

Fixes: 193125db ("net: Introduce VRF device driver")
Fixes: 35402e31 ("net: Add IPv6 support to VRF device")
Signed-off-by: default avatarDavid Ahern <dsa@cumulusnetworks.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 10c3c022
...@@ -60,41 +60,6 @@ struct pcpu_dstats { ...@@ -60,41 +60,6 @@ struct pcpu_dstats {
struct u64_stats_sync syncp; struct u64_stats_sync syncp;
}; };
static struct dst_entry *vrf_ip_check(struct dst_entry *dst, u32 cookie)
{
return dst;
}
static int vrf_ip_local_out(struct net *net, struct sock *sk, struct sk_buff *skb)
{
return ip_local_out(net, sk, skb);
}
static unsigned int vrf_v4_mtu(const struct dst_entry *dst)
{
/* TO-DO: return max ethernet size? */
return dst->dev->mtu;
}
static void vrf_dst_destroy(struct dst_entry *dst)
{
/* our dst lives forever - or until the device is closed */
}
static unsigned int vrf_default_advmss(const struct dst_entry *dst)
{
return 65535 - 40;
}
static struct dst_ops vrf_dst_ops = {
.family = AF_INET,
.local_out = vrf_ip_local_out,
.check = vrf_ip_check,
.mtu = vrf_v4_mtu,
.destroy = vrf_dst_destroy,
.default_advmss = vrf_default_advmss,
};
/* neighbor handling is done with actual device; do not want /* neighbor handling is done with actual device; do not want
* to flip skb->dev for those ndisc packets. This really fails * to flip skb->dev for those ndisc packets. This really fails
* for multiple next protocols (e.g., NEXTHDR_HOP). But it is * for multiple next protocols (e.g., NEXTHDR_HOP). But it is
...@@ -349,46 +314,6 @@ static netdev_tx_t vrf_xmit(struct sk_buff *skb, struct net_device *dev) ...@@ -349,46 +314,6 @@ static netdev_tx_t vrf_xmit(struct sk_buff *skb, struct net_device *dev)
} }
#if IS_ENABLED(CONFIG_IPV6) #if IS_ENABLED(CONFIG_IPV6)
static struct dst_entry *vrf_ip6_check(struct dst_entry *dst, u32 cookie)
{
return dst;
}
static struct dst_ops vrf_dst_ops6 = {
.family = AF_INET6,
.local_out = ip6_local_out,
.check = vrf_ip6_check,
.mtu = vrf_v4_mtu,
.destroy = vrf_dst_destroy,
.default_advmss = vrf_default_advmss,
};
static int init_dst_ops6_kmem_cachep(void)
{
vrf_dst_ops6.kmem_cachep = kmem_cache_create("vrf_ip6_dst_cache",
sizeof(struct rt6_info),
0,
SLAB_HWCACHE_ALIGN,
NULL);
if (!vrf_dst_ops6.kmem_cachep)
return -ENOMEM;
return 0;
}
static void free_dst_ops6_kmem_cachep(void)
{
kmem_cache_destroy(vrf_dst_ops6.kmem_cachep);
}
static int vrf_input6(struct sk_buff *skb)
{
skb->dev->stats.rx_errors++;
kfree_skb(skb);
return 0;
}
/* modelled after ip6_finish_output2 */ /* modelled after ip6_finish_output2 */
static int vrf_finish_output6(struct net *net, struct sock *sk, static int vrf_finish_output6(struct net *net, struct sock *sk,
struct sk_buff *skb) struct sk_buff *skb)
...@@ -429,67 +354,34 @@ static int vrf_output6(struct net *net, struct sock *sk, struct sk_buff *skb) ...@@ -429,67 +354,34 @@ static int vrf_output6(struct net *net, struct sock *sk, struct sk_buff *skb)
!(IP6CB(skb)->flags & IP6SKB_REROUTED)); !(IP6CB(skb)->flags & IP6SKB_REROUTED));
} }
static void vrf_rt6_destroy(struct net_vrf *vrf) static void vrf_rt6_release(struct net_vrf *vrf)
{ {
dst_destroy(&vrf->rt6->dst); dst_release(&vrf->rt6->dst);
free_percpu(vrf->rt6->rt6i_pcpu);
vrf->rt6 = NULL; vrf->rt6 = NULL;
} }
static int vrf_rt6_create(struct net_device *dev) static int vrf_rt6_create(struct net_device *dev)
{ {
struct net_vrf *vrf = netdev_priv(dev); struct net_vrf *vrf = netdev_priv(dev);
struct dst_entry *dst; struct net *net = dev_net(dev);
struct rt6_info *rt6; struct rt6_info *rt6;
int cpu;
int rc = -ENOMEM; int rc = -ENOMEM;
rt6 = dst_alloc(&vrf_dst_ops6, dev, 0, rt6 = ip6_dst_alloc(net, dev,
DST_OBSOLETE_NONE, DST_HOST | DST_NOPOLICY | DST_NOXFRM | DST_NOCACHE);
(DST_HOST | DST_NOPOLICY | DST_NOXFRM));
if (!rt6) if (!rt6)
goto out; goto out;
dst = &rt6->dst;
rt6->rt6i_pcpu = alloc_percpu_gfp(struct rt6_info *, GFP_KERNEL);
if (!rt6->rt6i_pcpu) {
dst_destroy(dst);
goto out;
}
for_each_possible_cpu(cpu) {
struct rt6_info **p = per_cpu_ptr(rt6->rt6i_pcpu, cpu);
*p = NULL;
}
memset(dst + 1, 0, sizeof(*rt6) - sizeof(*dst));
INIT_LIST_HEAD(&rt6->rt6i_siblings);
INIT_LIST_HEAD(&rt6->rt6i_uncached);
rt6->dst.input = vrf_input6;
rt6->dst.output = vrf_output6; rt6->dst.output = vrf_output6;
rt6->rt6i_table = fib6_get_table(net, vrf->tb_id);
rt6->rt6i_table = fib6_get_table(dev_net(dev), vrf->tb_id); dst_hold(&rt6->dst);
atomic_set(&rt6->dst.__refcnt, 2);
vrf->rt6 = rt6; vrf->rt6 = rt6;
rc = 0; rc = 0;
out: out:
return rc; return rc;
} }
#else #else
static int init_dst_ops6_kmem_cachep(void) static void vrf_rt6_release(struct net_vrf *vrf)
{
return 0;
}
static void free_dst_ops6_kmem_cachep(void)
{
}
static void vrf_rt6_destroy(struct net_vrf *vrf)
{ {
} }
...@@ -557,11 +449,11 @@ static int vrf_output(struct net *net, struct sock *sk, struct sk_buff *skb) ...@@ -557,11 +449,11 @@ static int vrf_output(struct net *net, struct sock *sk, struct sk_buff *skb)
!(IPCB(skb)->flags & IPSKB_REROUTED)); !(IPCB(skb)->flags & IPSKB_REROUTED));
} }
static void vrf_rtable_destroy(struct net_vrf *vrf) static void vrf_rtable_release(struct net_vrf *vrf)
{ {
struct dst_entry *dst = (struct dst_entry *)vrf->rth; struct dst_entry *dst = (struct dst_entry *)vrf->rth;
dst_destroy(dst); dst_release(dst);
vrf->rth = NULL; vrf->rth = NULL;
} }
...@@ -570,22 +462,10 @@ static struct rtable *vrf_rtable_create(struct net_device *dev) ...@@ -570,22 +462,10 @@ static struct rtable *vrf_rtable_create(struct net_device *dev)
struct net_vrf *vrf = netdev_priv(dev); struct net_vrf *vrf = netdev_priv(dev);
struct rtable *rth; struct rtable *rth;
rth = dst_alloc(&vrf_dst_ops, dev, 2, rth = rt_dst_alloc(dev, 0, RTN_UNICAST, 1, 1, 0);
DST_OBSOLETE_NONE,
(DST_HOST | DST_NOPOLICY | DST_NOXFRM));
if (rth) { if (rth) {
rth->dst.output = vrf_output; rth->dst.output = vrf_output;
rth->rt_genid = rt_genid_ipv4(dev_net(dev));
rth->rt_flags = 0;
rth->rt_type = RTN_UNICAST;
rth->rt_is_input = 0;
rth->rt_iif = 0;
rth->rt_pmtu = 0;
rth->rt_gateway = 0;
rth->rt_uses_gateway = 0;
rth->rt_table_id = vrf->tb_id; rth->rt_table_id = vrf->tb_id;
INIT_LIST_HEAD(&rth->rt_uncached);
rth->rt_uncached_list = NULL;
} }
return rth; return rth;
...@@ -673,8 +553,8 @@ static void vrf_dev_uninit(struct net_device *dev) ...@@ -673,8 +553,8 @@ static void vrf_dev_uninit(struct net_device *dev)
struct net_device *port_dev; struct net_device *port_dev;
struct list_head *iter; struct list_head *iter;
vrf_rtable_destroy(vrf); vrf_rtable_release(vrf);
vrf_rt6_destroy(vrf); vrf_rt6_release(vrf);
netdev_for_each_lower_dev(dev, port_dev, iter) netdev_for_each_lower_dev(dev, port_dev, iter)
vrf_del_slave(dev, port_dev); vrf_del_slave(dev, port_dev);
...@@ -704,7 +584,7 @@ static int vrf_dev_init(struct net_device *dev) ...@@ -704,7 +584,7 @@ static int vrf_dev_init(struct net_device *dev)
return 0; return 0;
out_rth: out_rth:
vrf_rtable_destroy(vrf); vrf_rtable_release(vrf);
out_stats: out_stats:
free_percpu(dev->dstats); free_percpu(dev->dstats);
dev->dstats = NULL; dev->dstats = NULL;
...@@ -737,7 +617,7 @@ static struct rtable *vrf_get_rtable(const struct net_device *dev, ...@@ -737,7 +617,7 @@ static struct rtable *vrf_get_rtable(const struct net_device *dev,
struct net_vrf *vrf = netdev_priv(dev); struct net_vrf *vrf = netdev_priv(dev);
rth = vrf->rth; rth = vrf->rth;
atomic_inc(&rth->dst.__refcnt); dst_hold(&rth->dst);
} }
return rth; return rth;
...@@ -788,7 +668,7 @@ static struct dst_entry *vrf_get_rt6_dst(const struct net_device *dev, ...@@ -788,7 +668,7 @@ static struct dst_entry *vrf_get_rt6_dst(const struct net_device *dev,
struct net_vrf *vrf = netdev_priv(dev); struct net_vrf *vrf = netdev_priv(dev);
rt = vrf->rt6; rt = vrf->rt6;
atomic_inc(&rt->dst.__refcnt); dst_hold(&rt->dst);
} }
return (struct dst_entry *)rt; return (struct dst_entry *)rt;
...@@ -946,19 +826,6 @@ static int __init vrf_init_module(void) ...@@ -946,19 +826,6 @@ static int __init vrf_init_module(void)
{ {
int rc; int rc;
vrf_dst_ops.kmem_cachep =
kmem_cache_create("vrf_ip_dst_cache",
sizeof(struct rtable), 0,
SLAB_HWCACHE_ALIGN,
NULL);
if (!vrf_dst_ops.kmem_cachep)
return -ENOMEM;
rc = init_dst_ops6_kmem_cachep();
if (rc != 0)
goto error2;
register_netdevice_notifier(&vrf_notifier_block); register_netdevice_notifier(&vrf_notifier_block);
rc = rtnl_link_register(&vrf_link_ops); rc = rtnl_link_register(&vrf_link_ops);
...@@ -969,22 +836,10 @@ static int __init vrf_init_module(void) ...@@ -969,22 +836,10 @@ static int __init vrf_init_module(void)
error: error:
unregister_netdevice_notifier(&vrf_notifier_block); unregister_netdevice_notifier(&vrf_notifier_block);
free_dst_ops6_kmem_cachep();
error2:
kmem_cache_destroy(vrf_dst_ops.kmem_cachep);
return rc; return rc;
} }
static void __exit vrf_cleanup_module(void)
{
rtnl_link_unregister(&vrf_link_ops);
unregister_netdevice_notifier(&vrf_notifier_block);
kmem_cache_destroy(vrf_dst_ops.kmem_cachep);
free_dst_ops6_kmem_cachep();
}
module_init(vrf_init_module); module_init(vrf_init_module);
module_exit(vrf_cleanup_module);
MODULE_AUTHOR("Shrijeet Mukherjee, David Ahern"); MODULE_AUTHOR("Shrijeet Mukherjee, David Ahern");
MODULE_DESCRIPTION("Device driver to instantiate VRF domains"); MODULE_DESCRIPTION("Device driver to instantiate VRF domains");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
......
...@@ -101,6 +101,9 @@ void fib6_force_start_gc(struct net *net); ...@@ -101,6 +101,9 @@ void fib6_force_start_gc(struct net *net);
struct rt6_info *addrconf_dst_alloc(struct inet6_dev *idev, struct rt6_info *addrconf_dst_alloc(struct inet6_dev *idev,
const struct in6_addr *addr, bool anycast); const struct in6_addr *addr, bool anycast);
struct rt6_info *ip6_dst_alloc(struct net *net, struct net_device *dev,
int flags);
/* /*
* support functions for ND * support functions for ND
* *
......
...@@ -209,6 +209,9 @@ unsigned int inet_addr_type_dev_table(struct net *net, ...@@ -209,6 +209,9 @@ unsigned int inet_addr_type_dev_table(struct net *net,
void ip_rt_multicast_event(struct in_device *); void ip_rt_multicast_event(struct in_device *);
int ip_rt_ioctl(struct net *, unsigned int cmd, void __user *arg); int ip_rt_ioctl(struct net *, unsigned int cmd, void __user *arg);
void ip_rt_get_source(u8 *src, struct sk_buff *skb, struct rtable *rt); void ip_rt_get_source(u8 *src, struct sk_buff *skb, struct rtable *rt);
struct rtable *rt_dst_alloc(struct net_device *dev,
unsigned int flags, u16 type,
bool nopolicy, bool noxfrm, bool will_cache);
struct in_ifaddr; struct in_ifaddr;
void fib_add_ifaddr(struct in_ifaddr *); void fib_add_ifaddr(struct in_ifaddr *);
......
...@@ -1438,7 +1438,7 @@ static void rt_set_nexthop(struct rtable *rt, __be32 daddr, ...@@ -1438,7 +1438,7 @@ static void rt_set_nexthop(struct rtable *rt, __be32 daddr,
#endif #endif
} }
static struct rtable *rt_dst_alloc(struct net_device *dev, struct rtable *rt_dst_alloc(struct net_device *dev,
unsigned int flags, u16 type, unsigned int flags, u16 type,
bool nopolicy, bool noxfrm, bool will_cache) bool nopolicy, bool noxfrm, bool will_cache)
{ {
...@@ -1468,6 +1468,7 @@ static struct rtable *rt_dst_alloc(struct net_device *dev, ...@@ -1468,6 +1468,7 @@ static struct rtable *rt_dst_alloc(struct net_device *dev,
return rt; return rt;
} }
EXPORT_SYMBOL(rt_dst_alloc);
/* called in rcu_read_lock() section */ /* called in rcu_read_lock() section */
static int ip_route_input_mc(struct sk_buff *skb, __be32 daddr, __be32 saddr, static int ip_route_input_mc(struct sk_buff *skb, __be32 daddr, __be32 saddr,
......
...@@ -338,7 +338,7 @@ static struct rt6_info *__ip6_dst_alloc(struct net *net, ...@@ -338,7 +338,7 @@ static struct rt6_info *__ip6_dst_alloc(struct net *net,
return rt; return rt;
} }
static struct rt6_info *ip6_dst_alloc(struct net *net, struct rt6_info *ip6_dst_alloc(struct net *net,
struct net_device *dev, struct net_device *dev,
int flags) int flags)
{ {
...@@ -364,6 +364,7 @@ static struct rt6_info *ip6_dst_alloc(struct net *net, ...@@ -364,6 +364,7 @@ static struct rt6_info *ip6_dst_alloc(struct net *net,
return rt; return rt;
} }
EXPORT_SYMBOL(ip6_dst_alloc);
static void ip6_dst_destroy(struct dst_entry *dst) static void ip6_dst_destroy(struct dst_entry *dst)
{ {
......
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