aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorUrsula Braun <ubraun@linux.vnet.ibm.com>2018-01-26 03:28:49 -0500
committerDavid S. Miller <davem@davemloft.net>2018-01-26 10:41:56 -0500
commit127f497058236e5f07672e11382232f80cb7e8c4 (patch)
tree6725c4e961d04c93677b71ea4615985b6effdef6
parent51f1de79ad8ed3555fd01ae8fd432691d397684b (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.c13
-rw-r--r--net/smc/smc_close.c33
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
971out: 975out:
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
22static void smc_close_cleanup_listen(struct sock *parent) 24static 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
33static 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 */
32static void smc_close_stream_wait(struct smc_sock *smc, long timeout) 55static 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);