aboutsummaryrefslogtreecommitdiffstats
path: root/include/net
diff options
context:
space:
mode:
Diffstat (limited to 'include/net')
-rw-r--r--include/net/inet6_hashtables.h12
-rw-r--r--include/net/inet_hashtables.h10
-rw-r--r--include/net/sock.h12
3 files changed, 27 insertions, 7 deletions
diff --git a/include/net/inet6_hashtables.h b/include/net/inet6_hashtables.h
index 995efbb031a..f74665d7bea 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 3522bbcd546..5cc182f9eca 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 75a312d3888..18f96708f3a 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 *);