Commit 0f24558e authored by Horia Geanta's avatar Horia Geanta Committed by Steffen Klassert

xfrm: avoid creating temporary SA when there are no listeners

In the case when KMs have no listeners, km_query() will fail and
temporary SAs are garbage collected immediately after their allocation.
This causes strain on memory allocation, leading even to OOM since
temporary SA alloc/free cycle is performed for every packet
and garbage collection does not keep up the pace.

The sane thing to do is to make sure we have audience before
temporary SA allocation.
Signed-off-by: default avatarHoria Geanta <horia.geanta@freescale.com>
Signed-off-by: default avatarSteffen Klassert <steffen.klassert@secunet.com>
parent 5826bdd1
...@@ -594,6 +594,7 @@ struct xfrm_mgr { ...@@ -594,6 +594,7 @@ struct xfrm_mgr {
const struct xfrm_migrate *m, const struct xfrm_migrate *m,
int num_bundles, int num_bundles,
const struct xfrm_kmaddress *k); const struct xfrm_kmaddress *k);
bool (*is_alive)(const struct km_event *c);
}; };
int xfrm_register_km(struct xfrm_mgr *km); int xfrm_register_km(struct xfrm_mgr *km);
...@@ -1646,6 +1647,20 @@ static inline int xfrm_aevent_is_on(struct net *net) ...@@ -1646,6 +1647,20 @@ static inline int xfrm_aevent_is_on(struct net *net)
rcu_read_unlock(); rcu_read_unlock();
return ret; return ret;
} }
static inline int xfrm_acquire_is_on(struct net *net)
{
struct sock *nlsk;
int ret = 0;
rcu_read_lock();
nlsk = rcu_dereference(net->xfrm.nlsk);
if (nlsk)
ret = netlink_has_listeners(nlsk, XFRMNLGRP_ACQUIRE);
rcu_read_unlock();
return ret;
}
#endif #endif
static inline int xfrm_alg_len(const struct xfrm_algo *alg) static inline int xfrm_alg_len(const struct xfrm_algo *alg)
......
...@@ -3059,6 +3059,24 @@ static u32 get_acqseq(void) ...@@ -3059,6 +3059,24 @@ static u32 get_acqseq(void)
return res; return res;
} }
static bool pfkey_is_alive(const struct km_event *c)
{
struct netns_pfkey *net_pfkey = net_generic(c->net, pfkey_net_id);
struct sock *sk;
bool is_alive = false;
rcu_read_lock();
sk_for_each_rcu(sk, &net_pfkey->table) {
if (pfkey_sk(sk)->registered) {
is_alive = true;
break;
}
}
rcu_read_unlock();
return is_alive;
}
static int pfkey_send_acquire(struct xfrm_state *x, struct xfrm_tmpl *t, struct xfrm_policy *xp) static int pfkey_send_acquire(struct xfrm_state *x, struct xfrm_tmpl *t, struct xfrm_policy *xp)
{ {
struct sk_buff *skb; struct sk_buff *skb;
...@@ -3784,6 +3802,7 @@ static struct xfrm_mgr pfkeyv2_mgr = ...@@ -3784,6 +3802,7 @@ static struct xfrm_mgr pfkeyv2_mgr =
.new_mapping = pfkey_send_new_mapping, .new_mapping = pfkey_send_new_mapping,
.notify_policy = pfkey_send_policy_notify, .notify_policy = pfkey_send_policy_notify,
.migrate = pfkey_send_migrate, .migrate = pfkey_send_migrate,
.is_alive = pfkey_is_alive,
}; };
static int __net_init pfkey_net_init(struct net *net) static int __net_init pfkey_net_init(struct net *net)
......
...@@ -161,6 +161,7 @@ static DEFINE_SPINLOCK(xfrm_state_gc_lock); ...@@ -161,6 +161,7 @@ static DEFINE_SPINLOCK(xfrm_state_gc_lock);
int __xfrm_state_delete(struct xfrm_state *x); int __xfrm_state_delete(struct xfrm_state *x);
int km_query(struct xfrm_state *x, struct xfrm_tmpl *t, struct xfrm_policy *pol); int km_query(struct xfrm_state *x, struct xfrm_tmpl *t, struct xfrm_policy *pol);
bool km_is_alive(const struct km_event *c);
void km_state_expired(struct xfrm_state *x, int hard, u32 portid); void km_state_expired(struct xfrm_state *x, int hard, u32 portid);
static DEFINE_SPINLOCK(xfrm_type_lock); static DEFINE_SPINLOCK(xfrm_type_lock);
...@@ -788,6 +789,7 @@ xfrm_state_find(const xfrm_address_t *daddr, const xfrm_address_t *saddr, ...@@ -788,6 +789,7 @@ xfrm_state_find(const xfrm_address_t *daddr, const xfrm_address_t *saddr,
struct xfrm_state *best = NULL; struct xfrm_state *best = NULL;
u32 mark = pol->mark.v & pol->mark.m; u32 mark = pol->mark.v & pol->mark.m;
unsigned short encap_family = tmpl->encap_family; unsigned short encap_family = tmpl->encap_family;
struct km_event c;
to_put = NULL; to_put = NULL;
...@@ -832,6 +834,17 @@ xfrm_state_find(const xfrm_address_t *daddr, const xfrm_address_t *saddr, ...@@ -832,6 +834,17 @@ xfrm_state_find(const xfrm_address_t *daddr, const xfrm_address_t *saddr,
error = -EEXIST; error = -EEXIST;
goto out; goto out;
} }
c.net = net;
/* If the KMs have no listeners (yet...), avoid allocating an SA
* for each and every packet - garbage collection might not
* handle the flood.
*/
if (!km_is_alive(&c)) {
error = -ESRCH;
goto out;
}
x = xfrm_state_alloc(net); x = xfrm_state_alloc(net);
if (x == NULL) { if (x == NULL) {
error = -ENOMEM; error = -ENOMEM;
...@@ -1793,6 +1806,24 @@ int km_report(struct net *net, u8 proto, struct xfrm_selector *sel, xfrm_address ...@@ -1793,6 +1806,24 @@ int km_report(struct net *net, u8 proto, struct xfrm_selector *sel, xfrm_address
} }
EXPORT_SYMBOL(km_report); EXPORT_SYMBOL(km_report);
bool km_is_alive(const struct km_event *c)
{
struct xfrm_mgr *km;
bool is_alive = false;
rcu_read_lock();
list_for_each_entry_rcu(km, &xfrm_km_list, list) {
if (km->is_alive && km->is_alive(c)) {
is_alive = true;
break;
}
}
rcu_read_unlock();
return is_alive;
}
EXPORT_SYMBOL(km_is_alive);
int xfrm_user_policy(struct sock *sk, int optname, u8 __user *optval, int optlen) int xfrm_user_policy(struct sock *sk, int optname, u8 __user *optval, int optlen)
{ {
int err; int err;
......
...@@ -2982,6 +2982,11 @@ static int xfrm_send_mapping(struct xfrm_state *x, xfrm_address_t *ipaddr, ...@@ -2982,6 +2982,11 @@ static int xfrm_send_mapping(struct xfrm_state *x, xfrm_address_t *ipaddr,
return nlmsg_multicast(net->xfrm.nlsk, skb, 0, XFRMNLGRP_MAPPING, GFP_ATOMIC); return nlmsg_multicast(net->xfrm.nlsk, skb, 0, XFRMNLGRP_MAPPING, GFP_ATOMIC);
} }
static bool xfrm_is_alive(const struct km_event *c)
{
return (bool)xfrm_acquire_is_on(c->net);
}
static struct xfrm_mgr netlink_mgr = { static struct xfrm_mgr netlink_mgr = {
.id = "netlink", .id = "netlink",
.notify = xfrm_send_state_notify, .notify = xfrm_send_state_notify,
...@@ -2991,6 +2996,7 @@ static struct xfrm_mgr netlink_mgr = { ...@@ -2991,6 +2996,7 @@ static struct xfrm_mgr netlink_mgr = {
.report = xfrm_send_report, .report = xfrm_send_report,
.migrate = xfrm_send_migrate, .migrate = xfrm_send_migrate,
.new_mapping = xfrm_send_mapping, .new_mapping = xfrm_send_mapping,
.is_alive = xfrm_is_alive,
}; };
static int __net_init xfrm_user_net_init(struct net *net) static int __net_init xfrm_user_net_init(struct net *net)
......
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