diff options
author | Shawn Bohrer <sbohrer@rgmadvisors.com> | 2013-10-07 12:01:39 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-10-08 16:27:33 -0400 |
commit | 421b3885bf6d56391297844f43fb7154a6396e12 (patch) | |
tree | 01b118c57d33da0e19be3ebaa5d4263025b86994 /include | |
parent | 005ec9743394010cd37d86c3fd2e81978231cdbf (diff) |
udp: ipv4: Add udp early demux
The removal of the routing cache introduced a performance regression for
some UDP workloads since a dst lookup must be done for each packet.
This change caches the dst per socket in a similar manner to what we do
for TCP by implementing early_demux.
For UDP multicast we can only cache the dst if there is only one
receiving socket on the host. Since caching only works when there is
one receiving socket we do the multicast socket lookup using RCU.
For UDP unicast we only demux sockets with an exact match in order to
not break forwarding setups. Additionally since the hash chains may be
long we only check the first socket to see if it is a match and not
waste extra time searching the whole chain when we might not find an
exact match.
Benchmark results from a netperf UDP_RR test:
Before 87961.22 transactions/s
After 89789.68 transactions/s
Benchmark results from a fio 1 byte UDP multicast pingpong test
(Multicast one way unicast response):
Before 12.97us RTT
After 12.63us RTT
Signed-off-by: Shawn Bohrer <sbohrer@rgmadvisors.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'include')
-rw-r--r-- | include/net/sock.h | 2 | ||||
-rw-r--r-- | include/net/udp.h | 1 |
2 files changed, 2 insertions, 1 deletions
diff --git a/include/net/sock.h b/include/net/sock.h index e3bf213be625..79532540201b 100644 --- a/include/net/sock.h +++ b/include/net/sock.h | |||
@@ -218,7 +218,7 @@ struct cg_proto; | |||
218 | * @sk_lock: synchronizer | 218 | * @sk_lock: synchronizer |
219 | * @sk_rcvbuf: size of receive buffer in bytes | 219 | * @sk_rcvbuf: size of receive buffer in bytes |
220 | * @sk_wq: sock wait queue and async head | 220 | * @sk_wq: sock wait queue and async head |
221 | * @sk_rx_dst: receive input route used by early tcp demux | 221 | * @sk_rx_dst: receive input route used by early demux |
222 | * @sk_dst_cache: destination cache | 222 | * @sk_dst_cache: destination cache |
223 | * @sk_dst_lock: destination cache lock | 223 | * @sk_dst_lock: destination cache lock |
224 | * @sk_policy: flow policy | 224 | * @sk_policy: flow policy |
diff --git a/include/net/udp.h b/include/net/udp.h index 510b8cb4d1a7..fe4ba9f32429 100644 --- a/include/net/udp.h +++ b/include/net/udp.h | |||
@@ -175,6 +175,7 @@ int udp_lib_get_port(struct sock *sk, unsigned short snum, | |||
175 | unsigned int hash2_nulladdr); | 175 | unsigned int hash2_nulladdr); |
176 | 176 | ||
177 | /* net/ipv4/udp.c */ | 177 | /* net/ipv4/udp.c */ |
178 | void udp_v4_early_demux(struct sk_buff *skb); | ||
178 | int udp_get_port(struct sock *sk, unsigned short snum, | 179 | int udp_get_port(struct sock *sk, unsigned short snum, |
179 | int (*saddr_cmp)(const struct sock *, | 180 | int (*saddr_cmp)(const struct sock *, |
180 | const struct sock *)); | 181 | const struct sock *)); |