diff options
| author | Ursula Braun <ubraun@linux.vnet.ibm.com> | 2018-01-26 09:28:49 +0100 | 
|---|---|---|
| committer | David S. Miller <davem@davemloft.net> | 2018-01-26 10:41:56 -0500 | 
| commit | 127f497058236e5f07672e11382232f80cb7e8c4 (patch) | |
| tree | 6725c4e961d04c93677b71ea4615985b6effdef6 /net/smc/smc_close.c | |
| 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>
Diffstat (limited to 'net/smc/smc_close.c')
| -rw-r--r-- | net/smc/smc_close.c | 33 | 
1 files changed, 24 insertions, 9 deletions
| 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 @@  #include "smc_cdc.h"  #include "smc_close.h" +#define SMC_CLOSE_WAIT_LISTEN_CLCSOCK_TIME	(5 * HZ) +  static void smc_close_cleanup_listen(struct sock *parent)  {  	struct sock *sk; @@ -28,6 +30,27 @@ static void smc_close_cleanup_listen(struct sock *parent)  		smc_close_non_accepted(sk);  } +static void smc_close_wait_listen_clcsock(struct smc_sock *smc) +{ +	DEFINE_WAIT_FUNC(wait, woken_wake_function); +	struct sock *sk = &smc->sk; +	signed long timeout; + +	timeout = SMC_CLOSE_WAIT_LISTEN_CLCSOCK_TIME; +	add_wait_queue(sk_sleep(sk), &wait); +	do { +		release_sock(sk); +		if (smc->clcsock) +			timeout = wait_woken(&wait, TASK_UNINTERRUPTIBLE, +					     timeout); +		sched_annotate_sleep(); +		lock_sock(sk); +		if (!smc->clcsock) +			break; +	} while (timeout); +	remove_wait_queue(sk_sleep(sk), &wait); +} +  /* wait for sndbuf data being transmitted */  static void smc_close_stream_wait(struct smc_sock *smc, long timeout)  { @@ -114,7 +137,6 @@ static void smc_close_active_abort(struct smc_sock *smc)  		break;  	case SMC_APPCLOSEWAIT1:  	case SMC_APPCLOSEWAIT2: -		sock_release(smc->clcsock);  		if (!smc_cdc_rxed_any_close(&smc->conn))  			sk->sk_state = SMC_PEERABORTWAIT;  		else @@ -128,7 +150,6 @@ static void smc_close_active_abort(struct smc_sock *smc)  		if (!txflags->peer_conn_closed) {  			/* just SHUTDOWN_SEND done */  			sk->sk_state = SMC_PEERABORTWAIT; -			sock_release(smc->clcsock);  		} else {  			sk->sk_state = SMC_CLOSED;  		} @@ -136,8 +157,6 @@ static void smc_close_active_abort(struct smc_sock *smc)  		break;  	case SMC_PROCESSABORT:  	case SMC_APPFINCLOSEWAIT: -		if (!txflags->peer_conn_closed) -			sock_release(smc->clcsock);  		sk->sk_state = SMC_CLOSED;  		break;  	case SMC_PEERFINCLOSEWAIT: @@ -177,8 +196,6 @@ again:  	switch (sk->sk_state) {  	case SMC_INIT:  		sk->sk_state = SMC_CLOSED; -		if (smc->smc_listen_work.func) -			cancel_work_sync(&smc->smc_listen_work);  		break;  	case SMC_LISTEN:  		sk->sk_state = SMC_CLOSED; @@ -187,11 +204,9 @@ again:  			rc = kernel_sock_shutdown(smc->clcsock, SHUT_RDWR);  			/* wake up kernel_accept of smc_tcp_listen_worker */  			smc->clcsock->sk->sk_data_ready(smc->clcsock->sk); +			smc_close_wait_listen_clcsock(smc);  		} -		release_sock(sk);  		smc_close_cleanup_listen(sk); -		cancel_work_sync(&smc->smc_listen_work); -		lock_sock(sk);  		break;  	case SMC_ACTIVE:  		smc_close_stream_wait(smc, timeout); | 
