diff options
-rw-r--r-- | include/net/inet_hashtables.h | 34 | ||||
-rw-r--r-- | net/ipv4/tcp_ipv4.c | 29 |
2 files changed, 36 insertions, 27 deletions
diff --git a/include/net/inet_hashtables.h b/include/net/inet_hashtables.h index c816708fa556..6731df2cea67 100644 --- a/include/net/inet_hashtables.h +++ b/include/net/inet_hashtables.h | |||
@@ -240,4 +240,38 @@ static inline void __inet_hash(struct inet_hashinfo *hashinfo, | |||
240 | if (listen_possible && sk->sk_state == TCP_LISTEN) | 240 | if (listen_possible && sk->sk_state == TCP_LISTEN) |
241 | wake_up(&hashinfo->lhash_wait); | 241 | wake_up(&hashinfo->lhash_wait); |
242 | } | 242 | } |
243 | |||
244 | static inline void inet_hash(struct inet_hashinfo *hashinfo, struct sock *sk) | ||
245 | { | ||
246 | if (sk->sk_state != TCP_CLOSE) { | ||
247 | local_bh_disable(); | ||
248 | __inet_hash(hashinfo, sk, 1); | ||
249 | local_bh_enable(); | ||
250 | } | ||
251 | } | ||
252 | |||
253 | static inline void inet_unhash(struct inet_hashinfo *hashinfo, struct sock *sk) | ||
254 | { | ||
255 | rwlock_t *lock; | ||
256 | |||
257 | if (sk_unhashed(sk)) | ||
258 | goto out; | ||
259 | |||
260 | if (sk->sk_state == TCP_LISTEN) { | ||
261 | local_bh_disable(); | ||
262 | inet_listen_wlock(hashinfo); | ||
263 | lock = &hashinfo->lhash_lock; | ||
264 | } else { | ||
265 | struct inet_ehash_bucket *head = &hashinfo->ehash[sk->sk_hashent]; | ||
266 | lock = &head->lock; | ||
267 | write_lock_bh(&head->lock); | ||
268 | } | ||
269 | |||
270 | if (__sk_del_node_init(sk)) | ||
271 | sock_prot_dec_use(sk->sk_prot); | ||
272 | write_unlock_bh(lock); | ||
273 | out: | ||
274 | if (sk->sk_state == TCP_LISTEN) | ||
275 | wake_up(&hashinfo->lhash_wait); | ||
276 | } | ||
243 | #endif /* _INET_HASHTABLES_H */ | 277 | #endif /* _INET_HASHTABLES_H */ |
diff --git a/net/ipv4/tcp_ipv4.c b/net/ipv4/tcp_ipv4.c index 5f9ad95304ca..dca1be67164b 100644 --- a/net/ipv4/tcp_ipv4.c +++ b/net/ipv4/tcp_ipv4.c | |||
@@ -230,37 +230,12 @@ fail: | |||
230 | 230 | ||
231 | static void tcp_v4_hash(struct sock *sk) | 231 | static void tcp_v4_hash(struct sock *sk) |
232 | { | 232 | { |
233 | if (sk->sk_state != TCP_CLOSE) { | 233 | inet_hash(&tcp_hashinfo, sk); |
234 | local_bh_disable(); | ||
235 | __inet_hash(&tcp_hashinfo, sk, 1); | ||
236 | local_bh_enable(); | ||
237 | } | ||
238 | } | 234 | } |
239 | 235 | ||
240 | void tcp_unhash(struct sock *sk) | 236 | void tcp_unhash(struct sock *sk) |
241 | { | 237 | { |
242 | rwlock_t *lock; | 238 | inet_unhash(&tcp_hashinfo, sk); |
243 | |||
244 | if (sk_unhashed(sk)) | ||
245 | goto ende; | ||
246 | |||
247 | if (sk->sk_state == TCP_LISTEN) { | ||
248 | local_bh_disable(); | ||
249 | inet_listen_wlock(&tcp_hashinfo); | ||
250 | lock = &tcp_hashinfo.lhash_lock; | ||
251 | } else { | ||
252 | struct inet_ehash_bucket *head = &tcp_hashinfo.ehash[sk->sk_hashent]; | ||
253 | lock = &head->lock; | ||
254 | write_lock_bh(&head->lock); | ||
255 | } | ||
256 | |||
257 | if (__sk_del_node_init(sk)) | ||
258 | sock_prot_dec_use(sk->sk_prot); | ||
259 | write_unlock_bh(lock); | ||
260 | |||
261 | ende: | ||
262 | if (sk->sk_state == TCP_LISTEN) | ||
263 | wake_up(&tcp_hashinfo.lhash_wait); | ||
264 | } | 239 | } |
265 | 240 | ||
266 | /* Don't inline this cruft. Here are some nice properties to | 241 | /* Don't inline this cruft. Here are some nice properties to |