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/ipv6/tcp_ipv6.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/ipv6/tcp_ipv6.c')
| -rw-r--r-- | net/ipv6/tcp_ipv6.c | 11 |
1 files changed, 8 insertions, 3 deletions
diff --git a/net/ipv6/tcp_ipv6.c b/net/ipv6/tcp_ipv6.c index fc0a4e5895ee..aadd7cef73b3 100644 --- a/net/ipv6/tcp_ipv6.c +++ b/net/ipv6/tcp_ipv6.c | |||
| @@ -2184,12 +2184,17 @@ static int tcpv6_net_init(struct net *net) | |||
| 2184 | static void tcpv6_net_exit(struct net *net) | 2184 | static void tcpv6_net_exit(struct net *net) |
| 2185 | { | 2185 | { |
| 2186 | inet_ctl_sock_destroy(net->ipv6.tcp_sk); | 2186 | inet_ctl_sock_destroy(net->ipv6.tcp_sk); |
| 2187 | inet_twsk_purge(net, &tcp_hashinfo, &tcp_death_row, AF_INET6); | 2187 | } |
| 2188 | |||
| 2189 | static void tcpv6_net_exit_batch(struct list_head *net_exit_list) | ||
| 2190 | { | ||
| 2191 | inet_twsk_purge(&tcp_hashinfo, &tcp_death_row, AF_INET6); | ||
| 2188 | } | 2192 | } |
| 2189 | 2193 | ||
| 2190 | static struct pernet_operations tcpv6_net_ops = { | 2194 | static struct pernet_operations tcpv6_net_ops = { |
| 2191 | .init = tcpv6_net_init, | 2195 | .init = tcpv6_net_init, |
| 2192 | .exit = tcpv6_net_exit, | 2196 | .exit = tcpv6_net_exit, |
| 2197 | .exit_batch = tcpv6_net_exit_batch, | ||
| 2193 | }; | 2198 | }; |
| 2194 | 2199 | ||
| 2195 | int __init tcpv6_init(void) | 2200 | int __init tcpv6_init(void) |
