net/smc: wait for pending work before clcsock release_sock
When the clcsock is already released using sock_release() and a pending smc_listen_work accesses the clcsock than that will fail. Solve this by canceling and waiting for the work to complete first. Because the work holds the sock_lock it must make sure that the lock is not hold before the new helper smc_clcsock_release() is invoked. And before the smc_listen_work starts working check if the parent listen socket is still valid, otherwise stop the work early. Signed-off-by: Karsten Graul <kgraul@linux.ibm.com> Signed-off-by: Ursula Braun <ubraun@linux.ibm.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:

committed by
David S. Miller

parent
988dc4a9a3
commit
fd57770dd1
@@ -21,6 +21,22 @@
|
||||
|
||||
#define SMC_CLOSE_WAIT_LISTEN_CLCSOCK_TIME (5 * HZ)
|
||||
|
||||
/* release the clcsock that is assigned to the smc_sock */
|
||||
void smc_clcsock_release(struct smc_sock *smc)
|
||||
{
|
||||
struct socket *tcp;
|
||||
|
||||
if (smc->listen_smc && current_work() != &smc->smc_listen_work)
|
||||
cancel_work_sync(&smc->smc_listen_work);
|
||||
mutex_lock(&smc->clcsock_release_lock);
|
||||
if (smc->clcsock) {
|
||||
tcp = smc->clcsock;
|
||||
smc->clcsock = NULL;
|
||||
sock_release(tcp);
|
||||
}
|
||||
mutex_unlock(&smc->clcsock_release_lock);
|
||||
}
|
||||
|
||||
static void smc_close_cleanup_listen(struct sock *parent)
|
||||
{
|
||||
struct sock *sk;
|
||||
@@ -321,6 +337,7 @@ static void smc_close_passive_work(struct work_struct *work)
|
||||
close_work);
|
||||
struct smc_sock *smc = container_of(conn, struct smc_sock, conn);
|
||||
struct smc_cdc_conn_state_flags *rxflags;
|
||||
bool release_clcsock = false;
|
||||
struct sock *sk = &smc->sk;
|
||||
int old_state;
|
||||
|
||||
@@ -400,13 +417,13 @@ wakeup:
|
||||
if ((sk->sk_state == SMC_CLOSED) &&
|
||||
(sock_flag(sk, SOCK_DEAD) || !sk->sk_socket)) {
|
||||
smc_conn_free(conn);
|
||||
if (smc->clcsock) {
|
||||
sock_release(smc->clcsock);
|
||||
smc->clcsock = NULL;
|
||||
}
|
||||
if (smc->clcsock)
|
||||
release_clcsock = true;
|
||||
}
|
||||
}
|
||||
release_sock(sk);
|
||||
if (release_clcsock)
|
||||
smc_clcsock_release(smc);
|
||||
sock_put(sk); /* sock_hold done by schedulers of close_work */
|
||||
}
|
||||
|
||||
|
Reference in New Issue
Block a user