diff options
author | Eric W. Biederman <ebiederm@xmission.com> | 2009-12-02 21:29:09 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2009-12-03 15:23:47 -0500 |
commit | b099ce2602d806deb41caaa578731848995cdb2a (patch) | |
tree | 7541d0d59c73e6b6274d420b8a3ec35d85ca5f2d /net/ipv4/inet_timewait_sock.c | |
parent | 575f4cd5a5b639457747434dbe18d175fa767db4 (diff) |
net: Batch inet_twsk_purge
This function walks the whole hashtable so there is no point in
passing it a network namespace. Instead I purge all timewait
sockets from dead network namespaces that I find. If the namespace
is one of the once I am trying to purge I am guaranteed no new timewait
sockets can be formed so this will get them all. If the namespace
is one I am not acting for it might form a few more but I will
call inet_twsk_purge again and shortly to get rid of them. In
any even if the network namespace is dead timewait sockets are
useless.
Move the calls of inet_twsk_purge into batch_exit routines so
that if I am killing a bunch of namespaces at once I will just
call inet_twsk_purge once and save a lot of redundant unnecessary
work.
My simple 4k network namespace exit test the cleanup time dropped from
roughly 8.2s to 1.6s. While the time spent running inet_twsk_purge fell
to about 2ms. 1ms for ipv4 and 1ms for ipv6.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv4/inet_timewait_sock.c')
-rw-r--r-- | net/ipv4/inet_timewait_sock.c | 10 |
1 files changed, 5 insertions, 5 deletions
diff --git a/net/ipv4/inet_timewait_sock.c b/net/ipv4/inet_timewait_sock.c index d38ca7c77b93..31f931ef3daf 100644 --- a/net/ipv4/inet_timewait_sock.c +++ b/net/ipv4/inet_timewait_sock.c | |||
@@ -421,7 +421,7 @@ out: | |||
421 | 421 | ||
422 | EXPORT_SYMBOL_GPL(inet_twdr_twcal_tick); | 422 | EXPORT_SYMBOL_GPL(inet_twdr_twcal_tick); |
423 | 423 | ||
424 | void inet_twsk_purge(struct net *net, struct inet_hashinfo *hashinfo, | 424 | void inet_twsk_purge(struct inet_hashinfo *hashinfo, |
425 | struct inet_timewait_death_row *twdr, int family) | 425 | struct inet_timewait_death_row *twdr, int family) |
426 | { | 426 | { |
427 | struct inet_timewait_sock *tw; | 427 | struct inet_timewait_sock *tw; |
@@ -436,15 +436,15 @@ restart_rcu: | |||
436 | restart: | 436 | restart: |
437 | sk_nulls_for_each_rcu(sk, node, &head->twchain) { | 437 | sk_nulls_for_each_rcu(sk, node, &head->twchain) { |
438 | tw = inet_twsk(sk); | 438 | tw = inet_twsk(sk); |
439 | if (!net_eq(twsk_net(tw), net) || | 439 | if ((tw->tw_family != family) || |
440 | tw->tw_family != family) | 440 | atomic_read(&twsk_net(tw)->count)) |
441 | continue; | 441 | continue; |
442 | 442 | ||
443 | if (unlikely(!atomic_inc_not_zero(&tw->tw_refcnt))) | 443 | if (unlikely(!atomic_inc_not_zero(&tw->tw_refcnt))) |
444 | continue; | 444 | continue; |
445 | 445 | ||
446 | if (unlikely(!net_eq(twsk_net(tw), net) || | 446 | if (unlikely((tw->tw_family != family) || |
447 | tw->tw_family != family)) { | 447 | atomic_read(&twsk_net(tw)->count))) { |
448 | inet_twsk_put(tw); | 448 | inet_twsk_put(tw); |
449 | goto restart; | 449 | goto restart; |
450 | } | 450 | } |