Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 127f4970 authored by Ursula Braun's avatar Ursula Braun Committed by David S. Miller
Browse files

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: default avatarUrsula Braun <ubraun@linux.vnet.ibm.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 51f1de79
Loading
Loading
Loading
Loading
+12 −1
Original line number Original line Diff line number Diff line
@@ -670,6 +670,10 @@ struct sock *smc_accept_dequeue(struct sock *parent,


		smc_accept_unlink(new_sk);
		smc_accept_unlink(new_sk);
		if (new_sk->sk_state == SMC_CLOSED) {
		if (new_sk->sk_state == SMC_CLOSED) {
			if (isk->clcsock) {
				sock_release(isk->clcsock);
				isk->clcsock = NULL;
			}
			new_sk->sk_prot->unhash(new_sk);
			new_sk->sk_prot->unhash(new_sk);
			sock_put(new_sk); /* final */
			sock_put(new_sk); /* final */
			continue;
			continue;
@@ -969,8 +973,15 @@ static void smc_tcp_listen_work(struct work_struct *work)
	}
	}


out:
out:
	if (lsmc->clcsock) {
		sock_release(lsmc->clcsock);
		lsmc->clcsock = NULL;
	}
	release_sock(lsk);
	release_sock(lsk);
	lsk->sk_data_ready(lsk); /* no more listening, wake accept */
	/* no more listening, wake up smc_close_wait_listen_clcsock and
	 * accept
	 */
	lsk->sk_state_change(lsk);
	sock_put(&lsmc->sk); /* sock_hold in smc_listen */
	sock_put(&lsmc->sk); /* sock_hold in smc_listen */
}
}


+24 −9
Original line number Original line Diff line number Diff line
@@ -19,6 +19,8 @@
#include "smc_cdc.h"
#include "smc_cdc.h"
#include "smc_close.h"
#include "smc_close.h"


#define SMC_CLOSE_WAIT_LISTEN_CLCSOCK_TIME	(5 * HZ)

static void smc_close_cleanup_listen(struct sock *parent)
static void smc_close_cleanup_listen(struct sock *parent)
{
{
	struct sock *sk;
	struct sock *sk;
@@ -28,6 +30,27 @@ static void smc_close_cleanup_listen(struct sock *parent)
		smc_close_non_accepted(sk);
		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 */
/* wait for sndbuf data being transmitted */
static void smc_close_stream_wait(struct smc_sock *smc, long timeout)
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;
		break;
	case SMC_APPCLOSEWAIT1:
	case SMC_APPCLOSEWAIT1:
	case SMC_APPCLOSEWAIT2:
	case SMC_APPCLOSEWAIT2:
		sock_release(smc->clcsock);
		if (!smc_cdc_rxed_any_close(&smc->conn))
		if (!smc_cdc_rxed_any_close(&smc->conn))
			sk->sk_state = SMC_PEERABORTWAIT;
			sk->sk_state = SMC_PEERABORTWAIT;
		else
		else
@@ -128,7 +150,6 @@ static void smc_close_active_abort(struct smc_sock *smc)
		if (!txflags->peer_conn_closed) {
		if (!txflags->peer_conn_closed) {
			/* just SHUTDOWN_SEND done */
			/* just SHUTDOWN_SEND done */
			sk->sk_state = SMC_PEERABORTWAIT;
			sk->sk_state = SMC_PEERABORTWAIT;
			sock_release(smc->clcsock);
		} else {
		} else {
			sk->sk_state = SMC_CLOSED;
			sk->sk_state = SMC_CLOSED;
		}
		}
@@ -136,8 +157,6 @@ static void smc_close_active_abort(struct smc_sock *smc)
		break;
		break;
	case SMC_PROCESSABORT:
	case SMC_PROCESSABORT:
	case SMC_APPFINCLOSEWAIT:
	case SMC_APPFINCLOSEWAIT:
		if (!txflags->peer_conn_closed)
			sock_release(smc->clcsock);
		sk->sk_state = SMC_CLOSED;
		sk->sk_state = SMC_CLOSED;
		break;
		break;
	case SMC_PEERFINCLOSEWAIT:
	case SMC_PEERFINCLOSEWAIT:
@@ -177,8 +196,6 @@ int smc_close_active(struct smc_sock *smc)
	switch (sk->sk_state) {
	switch (sk->sk_state) {
	case SMC_INIT:
	case SMC_INIT:
		sk->sk_state = SMC_CLOSED;
		sk->sk_state = SMC_CLOSED;
		if (smc->smc_listen_work.func)
			cancel_work_sync(&smc->smc_listen_work);
		break;
		break;
	case SMC_LISTEN:
	case SMC_LISTEN:
		sk->sk_state = SMC_CLOSED;
		sk->sk_state = SMC_CLOSED;
@@ -187,11 +204,9 @@ int smc_close_active(struct smc_sock *smc)
			rc = kernel_sock_shutdown(smc->clcsock, SHUT_RDWR);
			rc = kernel_sock_shutdown(smc->clcsock, SHUT_RDWR);
			/* wake up kernel_accept of smc_tcp_listen_worker */
			/* wake up kernel_accept of smc_tcp_listen_worker */
			smc->clcsock->sk->sk_data_ready(smc->clcsock->sk);
			smc->clcsock->sk->sk_data_ready(smc->clcsock->sk);
			smc_close_wait_listen_clcsock(smc);
		}
		}
		release_sock(sk);
		smc_close_cleanup_listen(sk);
		smc_close_cleanup_listen(sk);
		cancel_work_sync(&smc->smc_listen_work);
		lock_sock(sk);
		break;
		break;
	case SMC_ACTIVE:
	case SMC_ACTIVE:
		smc_close_stream_wait(smc, timeout);
		smc_close_stream_wait(smc, timeout);