Commit 8eb4adf6 authored by David S. Miller's avatar David S. Miller

Merge branch 'master' of git://git.kernel.org/pub/scm/linux/kernel/git/klassert/ipsec

Steffen Klassert says:

====================
pull request (net): ipsec 2016-11-25

1) Fix a refcount leak in vti6.
   From Nicolas Dichtel.

2) Fix a wrong if statement in xfrm_sk_policy_lookup.
   From Florian Westphal.

3) The flowcache watermarks are per cpu. Take this into
   account when comparing to the threshold where we
   refusing new allocations. From Miroslav Urbanek.

Please pull or let me know if there are problems.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents e824265d 6b226487
...@@ -95,7 +95,6 @@ static void flow_cache_gc_task(struct work_struct *work) ...@@ -95,7 +95,6 @@ static void flow_cache_gc_task(struct work_struct *work)
list_for_each_entry_safe(fce, n, &gc_list, u.gc_list) { list_for_each_entry_safe(fce, n, &gc_list, u.gc_list) {
flow_entry_kill(fce, xfrm); flow_entry_kill(fce, xfrm);
atomic_dec(&xfrm->flow_cache_gc_count); atomic_dec(&xfrm->flow_cache_gc_count);
WARN_ON(atomic_read(&xfrm->flow_cache_gc_count) < 0);
} }
} }
...@@ -236,9 +235,8 @@ flow_cache_lookup(struct net *net, const struct flowi *key, u16 family, u8 dir, ...@@ -236,9 +235,8 @@ flow_cache_lookup(struct net *net, const struct flowi *key, u16 family, u8 dir,
if (fcp->hash_count > fc->high_watermark) if (fcp->hash_count > fc->high_watermark)
flow_cache_shrink(fc, fcp); flow_cache_shrink(fc, fcp);
if (fcp->hash_count > 2 * fc->high_watermark || if (atomic_read(&net->xfrm.flow_cache_gc_count) >
atomic_read(&net->xfrm.flow_cache_gc_count) > fc->high_watermark) { 2 * num_online_cpus() * fc->high_watermark) {
atomic_inc(&net->xfrm.flow_cache_genid);
flo = ERR_PTR(-ENOBUFS); flo = ERR_PTR(-ENOBUFS);
goto ret_object; goto ret_object;
} }
......
...@@ -1138,6 +1138,33 @@ static struct xfrm6_protocol vti_ipcomp6_protocol __read_mostly = { ...@@ -1138,6 +1138,33 @@ static struct xfrm6_protocol vti_ipcomp6_protocol __read_mostly = {
.priority = 100, .priority = 100,
}; };
static bool is_vti6_tunnel(const struct net_device *dev)
{
return dev->netdev_ops == &vti6_netdev_ops;
}
static int vti6_device_event(struct notifier_block *unused,
unsigned long event, void *ptr)
{
struct net_device *dev = netdev_notifier_info_to_dev(ptr);
struct ip6_tnl *t = netdev_priv(dev);
if (!is_vti6_tunnel(dev))
return NOTIFY_DONE;
switch (event) {
case NETDEV_DOWN:
if (!net_eq(t->net, dev_net(dev)))
xfrm_garbage_collect(t->net);
break;
}
return NOTIFY_DONE;
}
static struct notifier_block vti6_notifier_block __read_mostly = {
.notifier_call = vti6_device_event,
};
/** /**
* vti6_tunnel_init - register protocol and reserve needed resources * vti6_tunnel_init - register protocol and reserve needed resources
* *
...@@ -1148,6 +1175,8 @@ static int __init vti6_tunnel_init(void) ...@@ -1148,6 +1175,8 @@ static int __init vti6_tunnel_init(void)
const char *msg; const char *msg;
int err; int err;
register_netdevice_notifier(&vti6_notifier_block);
msg = "tunnel device"; msg = "tunnel device";
err = register_pernet_device(&vti6_net_ops); err = register_pernet_device(&vti6_net_ops);
if (err < 0) if (err < 0)
...@@ -1180,6 +1209,7 @@ static int __init vti6_tunnel_init(void) ...@@ -1180,6 +1209,7 @@ static int __init vti6_tunnel_init(void)
xfrm_proto_esp_failed: xfrm_proto_esp_failed:
unregister_pernet_device(&vti6_net_ops); unregister_pernet_device(&vti6_net_ops);
pernet_dev_failed: pernet_dev_failed:
unregister_netdevice_notifier(&vti6_notifier_block);
pr_err("vti6 init: failed to register %s\n", msg); pr_err("vti6 init: failed to register %s\n", msg);
return err; return err;
} }
...@@ -1194,6 +1224,7 @@ static void __exit vti6_tunnel_cleanup(void) ...@@ -1194,6 +1224,7 @@ static void __exit vti6_tunnel_cleanup(void)
xfrm6_protocol_deregister(&vti_ah6_protocol, IPPROTO_AH); xfrm6_protocol_deregister(&vti_ah6_protocol, IPPROTO_AH);
xfrm6_protocol_deregister(&vti_esp6_protocol, IPPROTO_ESP); xfrm6_protocol_deregister(&vti_esp6_protocol, IPPROTO_ESP);
unregister_pernet_device(&vti6_net_ops); unregister_pernet_device(&vti6_net_ops);
unregister_netdevice_notifier(&vti6_notifier_block);
} }
module_init(vti6_tunnel_init); module_init(vti6_tunnel_init);
......
...@@ -1268,12 +1268,14 @@ static struct xfrm_policy *xfrm_sk_policy_lookup(const struct sock *sk, int dir, ...@@ -1268,12 +1268,14 @@ static struct xfrm_policy *xfrm_sk_policy_lookup(const struct sock *sk, int dir,
err = security_xfrm_policy_lookup(pol->security, err = security_xfrm_policy_lookup(pol->security,
fl->flowi_secid, fl->flowi_secid,
policy_to_flow_dir(dir)); policy_to_flow_dir(dir));
if (!err && !xfrm_pol_hold_rcu(pol)) if (!err) {
if (!xfrm_pol_hold_rcu(pol))
goto again; goto again;
else if (err == -ESRCH) } else if (err == -ESRCH) {
pol = NULL; pol = NULL;
else } else {
pol = ERR_PTR(err); pol = ERR_PTR(err);
}
} else } else
pol = NULL; pol = NULL;
} }
......
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