diff options
author | Ursula Braun <ubraun@linux.vnet.ibm.com> | 2018-01-26 03:28:49 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2018-01-26 10:41:56 -0500 |
commit | 127f497058236e5f07672e11382232f80cb7e8c4 (patch) | |
tree | 6725c4e961d04c93677b71ea4615985b6effdef6 | |
parent | 51f1de79ad8ed3555fd01ae8fd432691d397684b (diff) |
net/smc: release clcsock from tcp_listen_worker
Closing a listen socket may hit the warning
WARN_ON(sock_owned_by_user(sk)) of tcp_close(), if the wake up of
the smc_tcp_listen_worker has not yet finished.
This patch introduces smc_close_wait_listen_clcsock() making sure
the listening internal clcsock has been closed in smc_tcp_listen_work(),
before the listening external SMC socket finishes closing.
Signed-off-by: Ursula Braun <ubraun@linux.vnet.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | net/smc/af_smc.c | 13 | ||||
-rw-r--r-- | net/smc/smc_close.c | 33 |
2 files changed, 36 insertions, 10 deletions
diff --git a/net/smc/af_smc.c b/net/smc/af_smc.c index 732a37ddbc21..267e68379110 100644 --- a/net/smc/af_smc.c +++ b/net/smc/af_smc.c | |||
@@ -670,6 +670,10 @@ struct sock *smc_accept_dequeue(struct sock *parent, | |||
670 | 670 | ||
671 | smc_accept_unlink(new_sk); | 671 | smc_accept_unlink(new_sk); |
672 | if (new_sk->sk_state == SMC_CLOSED) { | 672 | if (new_sk->sk_state == SMC_CLOSED) { |
673 | if (isk->clcsock) { | ||
674 | sock_release(isk->clcsock); | ||
675 | isk->clcsock = NULL; | ||
676 | } | ||
673 | new_sk->sk_prot->unhash(new_sk); | 677 | new_sk->sk_prot->unhash(new_sk); |
674 | sock_put(new_sk); /* final */ | 678 | sock_put(new_sk); /* final */ |
675 | continue; | 679 | continue; |
@@ -969,8 +973,15 @@ static void smc_tcp_listen_work(struct work_struct *work) | |||
969 | } | 973 | } |
970 | 974 | ||
971 | out: | 975 | out: |
976 | if (lsmc->clcsock) { | ||
977 | sock_release(lsmc->clcsock); | ||
978 | lsmc->clcsock = NULL; | ||
979 | } | ||
972 | release_sock(lsk); | 980 | release_sock(lsk); |
973 | lsk->sk_data_ready(lsk); /* no more listening, wake accept */ | 981 | /* no more listening, wake up smc_close_wait_listen_clcsock and |
982 | * accept | ||
983 | */ | ||
984 | lsk->sk_state_change(lsk); | ||
974 | sock_put(&lsmc->sk); /* sock_hold in smc_listen */ | 985 | sock_put(&lsmc->sk); /* sock_hold in smc_listen */ |
975 | } | 986 | } |
976 | 987 | ||
diff --git a/net/smc/smc_close.c b/net/smc/smc_close.c index 4339852a8910..e339c0186dcf 100644 --- a/net/smc/smc_close.c +++ b/net/smc/smc_close.c | |||
@@ -19,6 +19,8 @@ | |||
19 | #include "smc_cdc.h" | 19 | #include "smc_cdc.h" |
20 | #include "smc_close.h" | 20 | #include "smc_close.h" |
21 | 21 | ||
22 | #define SMC_CLOSE_WAIT_LISTEN_CLCSOCK_TIME (5 * HZ) | ||
23 | |||
22 | static void smc_close_cleanup_listen(struct sock *parent) | 24 | static void smc_close_cleanup_listen(struct sock *parent) |
23 | { | 25 | { |
24 | struct sock *sk; | 26 | struct sock *sk; |
@@ -28,6 +30,27 @@ static void smc_close_cleanup_listen(struct sock *parent) | |||
28 | smc_close_non_accepted(sk); | 30 | smc_close_non_accepted(sk); |
29 | } | 31 | } |
30 | 32 | ||
33 | static void smc_close_wait_listen_clcsock(struct smc_sock *smc) | ||
34 | { | ||
35 | DEFINE_WAIT_FUNC(wait, woken_wake_function); | ||
36 | struct sock *sk = &smc->sk; | ||
37 | signed long timeout; | ||
38 | |||
39 | timeout = SMC_CLOSE_WAIT_LISTEN_CLCSOCK_TIME; | ||
40 | add_wait_queue(sk_sleep(sk), &wait); | ||
41 | do { | ||
42 | release_sock(sk); | ||
43 | if (smc->clcsock) | ||
44 | timeout = wait_woken(&wait, TASK_UNINTERRUPTIBLE, | ||
45 | timeout); | ||
46 | sched_annotate_sleep(); | ||
47 | lock_sock(sk); | ||
48 | if (!smc->clcsock) | ||
49 | break; | ||
50 | } while (timeout); | ||
51 | remove_wait_queue(sk_sleep(sk), &wait); | ||
52 | } | ||
53 | |||
31 | /* wait for sndbuf data being transmitted */ | 54 | /* wait for sndbuf data being transmitted */ |
32 | static void smc_close_stream_wait(struct smc_sock *smc, long timeout) | 55 | static void smc_close_stream_wait(struct smc_sock *smc, long timeout) |
33 | { | 56 | { |
@@ -114,7 +137,6 @@ static void smc_close_active_abort(struct smc_sock *smc) | |||
114 | break; | 137 | break; |
115 | case SMC_APPCLOSEWAIT1: | 138 | case SMC_APPCLOSEWAIT1: |
116 | case SMC_APPCLOSEWAIT2: | 139 | case SMC_APPCLOSEWAIT2: |
117 | sock_release(smc->clcsock); | ||
118 | if (!smc_cdc_rxed_any_close(&smc->conn)) | 140 | if (!smc_cdc_rxed_any_close(&smc->conn)) |
119 | sk->sk_state = SMC_PEERABORTWAIT; | 141 | sk->sk_state = SMC_PEERABORTWAIT; |
120 | else | 142 | else |
@@ -128,7 +150,6 @@ static void smc_close_active_abort(struct smc_sock *smc) | |||
128 | if (!txflags->peer_conn_closed) { | 150 | if (!txflags->peer_conn_closed) { |
129 | /* just SHUTDOWN_SEND done */ | 151 | /* just SHUTDOWN_SEND done */ |
130 | sk->sk_state = SMC_PEERABORTWAIT; | 152 | sk->sk_state = SMC_PEERABORTWAIT; |
131 | sock_release(smc->clcsock); | ||
132 | } else { | 153 | } else { |
133 | sk->sk_state = SMC_CLOSED; | 154 | sk->sk_state = SMC_CLOSED; |
134 | } | 155 | } |
@@ -136,8 +157,6 @@ static void smc_close_active_abort(struct smc_sock *smc) | |||
136 | break; | 157 | break; |
137 | case SMC_PROCESSABORT: | 158 | case SMC_PROCESSABORT: |
138 | case SMC_APPFINCLOSEWAIT: | 159 | case SMC_APPFINCLOSEWAIT: |
139 | if (!txflags->peer_conn_closed) | ||
140 | sock_release(smc->clcsock); | ||
141 | sk->sk_state = SMC_CLOSED; | 160 | sk->sk_state = SMC_CLOSED; |
142 | break; | 161 | break; |
143 | case SMC_PEERFINCLOSEWAIT: | 162 | case SMC_PEERFINCLOSEWAIT: |
@@ -177,8 +196,6 @@ again: | |||
177 | switch (sk->sk_state) { | 196 | switch (sk->sk_state) { |
178 | case SMC_INIT: | 197 | case SMC_INIT: |
179 | sk->sk_state = SMC_CLOSED; | 198 | sk->sk_state = SMC_CLOSED; |
180 | if (smc->smc_listen_work.func) | ||
181 | cancel_work_sync(&smc->smc_listen_work); | ||
182 | break; | 199 | break; |
183 | case SMC_LISTEN: | 200 | case SMC_LISTEN: |
184 | sk->sk_state = SMC_CLOSED; | 201 | sk->sk_state = SMC_CLOSED; |
@@ -187,11 +204,9 @@ again: | |||
187 | rc = kernel_sock_shutdown(smc->clcsock, SHUT_RDWR); | 204 | rc = kernel_sock_shutdown(smc->clcsock, SHUT_RDWR); |
188 | /* wake up kernel_accept of smc_tcp_listen_worker */ | 205 | /* wake up kernel_accept of smc_tcp_listen_worker */ |
189 | smc->clcsock->sk->sk_data_ready(smc->clcsock->sk); | 206 | smc->clcsock->sk->sk_data_ready(smc->clcsock->sk); |
207 | smc_close_wait_listen_clcsock(smc); | ||
190 | } | 208 | } |
191 | release_sock(sk); | ||
192 | smc_close_cleanup_listen(sk); | 209 | smc_close_cleanup_listen(sk); |
193 | cancel_work_sync(&smc->smc_listen_work); | ||
194 | lock_sock(sk); | ||
195 | break; | 210 | break; |
196 | case SMC_ACTIVE: | 211 | case SMC_ACTIVE: |
197 | smc_close_stream_wait(smc, timeout); | 212 | smc_close_stream_wait(smc, timeout); |