diff options
author | Eric Dumazet <eric.dumazet@gmail.com> | 2009-10-06 20:37:59 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2009-10-08 01:00:22 -0400 |
commit | f86dcc5aa8c7908f2c287e7a211228df599e3e71 (patch) | |
tree | 1721c94d0254b337b0668b2e331771dfef4b142d /net/ipv6/udp.c | |
parent | 8a6dfd43d1891882f8ca05d73aa7735fb0edae3b (diff) |
udp: dynamically size hash tables at boot time
UDP_HTABLE_SIZE was initialy defined to 128, which is a bit small for
several setups.
4000 active UDP sockets -> 32 sockets per chain in average. An
incoming frame has to lookup all sockets to find best match, so long
chains hurt latency.
Instead of a fixed size hash table that cant be perfect for every
needs, let UDP stack choose its table size at boot time like tcp/ip
route, using alloc_large_system_hash() helper
Add an optional boot parameter, uhash_entries=x so that an admin can
force a size between 256 and 65536 if needed, like thash_entries and
rhash_entries.
dmesg logs two new lines :
[ 0.647039] UDP hash table entries: 512 (order: 0, 4096 bytes)
[ 0.647099] UDP Lite hash table entries: 512 (order: 0, 4096 bytes)
Maximal size on 64bit arches would be 65536 slots, ie 1 MBytes for non
debugging spinlocks.
Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv6/udp.c')
-rw-r--r-- | net/ipv6/udp.c | 6 |
1 files changed, 3 insertions, 3 deletions
diff --git a/net/ipv6/udp.c b/net/ipv6/udp.c index c6a303ec834c..ff778c172ef2 100644 --- a/net/ipv6/udp.c +++ b/net/ipv6/udp.c | |||
@@ -132,7 +132,7 @@ static struct sock *__udp6_lib_lookup(struct net *net, | |||
132 | struct sock *sk, *result; | 132 | struct sock *sk, *result; |
133 | struct hlist_nulls_node *node; | 133 | struct hlist_nulls_node *node; |
134 | unsigned short hnum = ntohs(dport); | 134 | unsigned short hnum = ntohs(dport); |
135 | unsigned int hash = udp_hashfn(net, hnum); | 135 | unsigned int hash = udp_hashfn(net, hnum, udptable->mask); |
136 | struct udp_hslot *hslot = &udptable->hash[hash]; | 136 | struct udp_hslot *hslot = &udptable->hash[hash]; |
137 | int score, badness; | 137 | int score, badness; |
138 | 138 | ||
@@ -452,7 +452,7 @@ static int __udp6_lib_mcast_deliver(struct net *net, struct sk_buff *skb, | |||
452 | { | 452 | { |
453 | struct sock *sk, *sk2; | 453 | struct sock *sk, *sk2; |
454 | const struct udphdr *uh = udp_hdr(skb); | 454 | const struct udphdr *uh = udp_hdr(skb); |
455 | struct udp_hslot *hslot = &udptable->hash[udp_hashfn(net, ntohs(uh->dest))]; | 455 | struct udp_hslot *hslot = udp_hashslot(udptable, net, ntohs(uh->dest)); |
456 | int dif; | 456 | int dif; |
457 | 457 | ||
458 | spin_lock(&hslot->lock); | 458 | spin_lock(&hslot->lock); |
@@ -1197,7 +1197,7 @@ static void udp6_sock_seq_show(struct seq_file *seq, struct sock *sp, int bucket | |||
1197 | destp = ntohs(inet->dport); | 1197 | destp = ntohs(inet->dport); |
1198 | srcp = ntohs(inet->sport); | 1198 | srcp = ntohs(inet->sport); |
1199 | seq_printf(seq, | 1199 | seq_printf(seq, |
1200 | "%4d: %08X%08X%08X%08X:%04X %08X%08X%08X%08X:%04X " | 1200 | "%5d: %08X%08X%08X%08X:%04X %08X%08X%08X%08X:%04X " |
1201 | "%02X %08X:%08X %02X:%08lX %08X %5d %8d %lu %d %p %d\n", | 1201 | "%02X %08X:%08X %02X:%08lX %08X %5d %8d %lu %d %p %d\n", |
1202 | bucket, | 1202 | bucket, |
1203 | src->s6_addr32[0], src->s6_addr32[1], | 1203 | src->s6_addr32[0], src->s6_addr32[1], |