Commit 48f238a7 authored by Long Li's avatar Long Li Committed by Steve French

cifs: smbd: avoid reconnect lockup

During transport reconnect, other processes may have registered memory
and blocked on transport. This creates a deadlock situation because the
transport resources can't be freed, and reconnect is blocked.

Fix this by returning to upper layer on timeout. Before returning,
transport status is set to reconnecting so other processes will release
memory registration resources.

Upper layer will retry the reconnect. This is not in fast I/O path so
setting the timeout to 5 seconds.
Signed-off-by: default avatarLong Li <longli@microsoft.com>
Signed-off-by: default avatarSteve French <smfrench@gmail.com>
Reviewed-by: default avatarRonnie Sahlberg <lsahlber@redhat.com>
CC: Stable <stable@vger.kernel.org>
parent 2a18287b
...@@ -1498,8 +1498,8 @@ int smbd_reconnect(struct TCP_Server_Info *server) ...@@ -1498,8 +1498,8 @@ int smbd_reconnect(struct TCP_Server_Info *server)
log_rdma_event(INFO, "reconnecting rdma session\n"); log_rdma_event(INFO, "reconnecting rdma session\n");
if (!server->smbd_conn) { if (!server->smbd_conn) {
log_rdma_event(ERR, "rdma session already destroyed\n"); log_rdma_event(INFO, "rdma session already destroyed\n");
return -EINVAL; goto create_conn;
} }
/* /*
...@@ -1512,15 +1512,19 @@ int smbd_reconnect(struct TCP_Server_Info *server) ...@@ -1512,15 +1512,19 @@ int smbd_reconnect(struct TCP_Server_Info *server)
} }
/* wait until the transport is destroyed */ /* wait until the transport is destroyed */
wait_event(server->smbd_conn->wait_destroy, if (!wait_event_timeout(server->smbd_conn->wait_destroy,
server->smbd_conn->transport_status == SMBD_DESTROYED); server->smbd_conn->transport_status == SMBD_DESTROYED, 5*HZ))
return -EAGAIN;
destroy_workqueue(server->smbd_conn->workqueue); destroy_workqueue(server->smbd_conn->workqueue);
kfree(server->smbd_conn); kfree(server->smbd_conn);
create_conn:
log_rdma_event(INFO, "creating rdma session\n"); log_rdma_event(INFO, "creating rdma session\n");
server->smbd_conn = smbd_get_connection( server->smbd_conn = smbd_get_connection(
server, (struct sockaddr *) &server->dstaddr); server, (struct sockaddr *) &server->dstaddr);
log_rdma_event(INFO, "created rdma session info=%p\n",
server->smbd_conn);
return server->smbd_conn ? 0 : -ENOENT; return server->smbd_conn ? 0 : -ENOENT;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment