aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/net/inet6_hashtables.h12
-rw-r--r--include/net/inet_hashtables.h10
-rw-r--r--include/net/sock.h12
-rw-r--r--net/ipv4/udp.c10
-rw-r--r--net/ipv6/udp.c10
5 files changed, 41 insertions, 13 deletions
diff --git a/include/net/inet6_hashtables.h b/include/net/inet6_hashtables.h
index 995efbb031ab..f74665d7bea8 100644
--- a/include/net/inet6_hashtables.h
+++ b/include/net/inet6_hashtables.h
@@ -96,10 +96,14 @@ static inline struct sock *__inet6_lookup_skb(struct inet_hashinfo *hashinfo,
96 const __be16 sport, 96 const __be16 sport,
97 const __be16 dport) 97 const __be16 dport)
98{ 98{
99 return __inet6_lookup(dev_net(skb->dst->dev), hashinfo, 99 struct sock *sk;
100 &ipv6_hdr(skb)->saddr, sport, 100
101 &ipv6_hdr(skb)->daddr, ntohs(dport), 101 if (unlikely(sk = skb_steal_sock(skb)))
102 inet6_iif(skb)); 102 return sk;
103 else return __inet6_lookup(dev_net(skb->dst->dev), hashinfo,
104 &ipv6_hdr(skb)->saddr, sport,
105 &ipv6_hdr(skb)->daddr, ntohs(dport),
106 inet6_iif(skb));
103} 107}
104 108
105extern struct sock *inet6_lookup(struct net *net, struct inet_hashinfo *hashinfo, 109extern struct sock *inet6_lookup(struct net *net, struct inet_hashinfo *hashinfo,
diff --git a/include/net/inet_hashtables.h b/include/net/inet_hashtables.h
index 3522bbcd546d..5cc182f9ecae 100644
--- a/include/net/inet_hashtables.h
+++ b/include/net/inet_hashtables.h
@@ -378,11 +378,15 @@ static inline struct sock *__inet_lookup_skb(struct inet_hashinfo *hashinfo,
378 const __be16 sport, 378 const __be16 sport,
379 const __be16 dport) 379 const __be16 dport)
380{ 380{
381 struct sock *sk;
381 const struct iphdr *iph = ip_hdr(skb); 382 const struct iphdr *iph = ip_hdr(skb);
382 383
383 return __inet_lookup(dev_net(skb->dst->dev), hashinfo, 384 if (unlikely(sk = skb_steal_sock(skb)))
384 iph->saddr, sport, 385 return sk;
385 iph->daddr, dport, inet_iif(skb)); 386 else
387 return __inet_lookup(dev_net(skb->dst->dev), hashinfo,
388 iph->saddr, sport,
389 iph->daddr, dport, inet_iif(skb));
386} 390}
387 391
388extern int __inet_hash_connect(struct inet_timewait_death_row *death_row, 392extern int __inet_hash_connect(struct inet_timewait_death_row *death_row,
diff --git a/include/net/sock.h b/include/net/sock.h
index 75a312d3888a..18f96708f3a6 100644
--- a/include/net/sock.h
+++ b/include/net/sock.h
@@ -1324,6 +1324,18 @@ static inline void sk_change_net(struct sock *sk, struct net *net)
1324 sock_net_set(sk, hold_net(net)); 1324 sock_net_set(sk, hold_net(net));
1325} 1325}
1326 1326
1327static inline struct sock *skb_steal_sock(struct sk_buff *skb)
1328{
1329 if (unlikely(skb->sk)) {
1330 struct sock *sk = skb->sk;
1331
1332 skb->destructor = NULL;
1333 skb->sk = NULL;
1334 return sk;
1335 }
1336 return NULL;
1337}
1338
1327extern void sock_enable_timestamp(struct sock *sk); 1339extern void sock_enable_timestamp(struct sock *sk);
1328extern int sock_get_timestamp(struct sock *, struct timeval __user *); 1340extern int sock_get_timestamp(struct sock *, struct timeval __user *);
1329extern int sock_get_timestampns(struct sock *, struct timespec __user *); 1341extern int sock_get_timestampns(struct sock *, struct timespec __user *);
diff --git a/net/ipv4/udp.c b/net/ipv4/udp.c
index c7a90b546b21..822c9deac83b 100644
--- a/net/ipv4/udp.c
+++ b/net/ipv4/udp.c
@@ -306,11 +306,15 @@ static inline struct sock *__udp4_lib_lookup_skb(struct sk_buff *skb,
306 __be16 sport, __be16 dport, 306 __be16 sport, __be16 dport,
307 struct hlist_head udptable[]) 307 struct hlist_head udptable[])
308{ 308{
309 struct sock *sk;
309 const struct iphdr *iph = ip_hdr(skb); 310 const struct iphdr *iph = ip_hdr(skb);
310 311
311 return __udp4_lib_lookup(dev_net(skb->dst->dev), iph->saddr, sport, 312 if (unlikely(sk = skb_steal_sock(skb)))
312 iph->daddr, dport, inet_iif(skb), 313 return sk;
313 udptable); 314 else
315 return __udp4_lib_lookup(dev_net(skb->dst->dev), iph->saddr, sport,
316 iph->daddr, dport, inet_iif(skb),
317 udptable);
314} 318}
315 319
316struct sock *udp4_lib_lookup(struct net *net, __be32 saddr, __be16 sport, 320struct sock *udp4_lib_lookup(struct net *net, __be32 saddr, __be16 sport,
diff --git a/net/ipv6/udp.c b/net/ipv6/udp.c
index ce26c41d157d..e51da8c092fa 100644
--- a/net/ipv6/udp.c
+++ b/net/ipv6/udp.c
@@ -111,11 +111,15 @@ static struct sock *__udp6_lib_lookup_skb(struct sk_buff *skb,
111 __be16 sport, __be16 dport, 111 __be16 sport, __be16 dport,
112 struct hlist_head udptable[]) 112 struct hlist_head udptable[])
113{ 113{
114 struct sock *sk;
114 struct ipv6hdr *iph = ipv6_hdr(skb); 115 struct ipv6hdr *iph = ipv6_hdr(skb);
115 116
116 return __udp6_lib_lookup(dev_net(skb->dst->dev), &iph->saddr, sport, 117 if (unlikely(sk = skb_steal_sock(skb)))
117 &iph->daddr, dport, inet6_iif(skb), 118 return sk;
118 udptable); 119 else
120 return __udp6_lib_lookup(dev_net(skb->dst->dev), &iph->saddr, sport,
121 &iph->daddr, dport, inet6_iif(skb),
122 udptable);
119} 123}
120 124
121/* 125/*