Skip to content

Commit dbe6974

Browse files
Trond Myklebustgregkh
authored andcommitted
SUNRPC: Don't call connect() more than once on a TCP socket
commit 89f4249 upstream. Avoid socket state races due to repeated calls to ->connect() using the same socket. If connect() returns 0 due to the connection having completed, but we are in fact in a closing state, then we may leave the XPRT_CONNECTING flag set on the transport. Reported-by: Enrico Scholz <[email protected]> Fixes: 3be232f ("SUNRPC: Prevent immediate close+reconnect") Signed-off-by: Trond Myklebust <[email protected]> [meenashanmugam: Backported to 5.10: Fixed merge conflict in xs_tcp_setup_socket] Signed-off-by: Meena Shanmugam <[email protected]> Signed-off-by: Greg Kroah-Hartman <[email protected]>
1 parent 47541ed commit dbe6974

File tree

2 files changed

+13
-10
lines changed

2 files changed

+13
-10
lines changed

include/linux/sunrpc/xprtsock.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -89,5 +89,6 @@ struct sock_xprt {
8989
#define XPRT_SOCK_WAKE_WRITE (5)
9090
#define XPRT_SOCK_WAKE_PENDING (6)
9191
#define XPRT_SOCK_WAKE_DISCONNECT (7)
92+
#define XPRT_SOCK_CONNECT_SENT (8)
9293

9394
#endif /* _LINUX_SUNRPC_XPRTSOCK_H */

net/sunrpc/xprtsock.c

Lines changed: 12 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -2260,10 +2260,14 @@ static void xs_tcp_setup_socket(struct work_struct *work)
22602260
struct rpc_xprt *xprt = &transport->xprt;
22612261
int status = -EIO;
22622262

2263-
if (!sock) {
2264-
sock = xs_create_sock(xprt, transport,
2265-
xs_addr(xprt)->sa_family, SOCK_STREAM,
2266-
IPPROTO_TCP, true);
2263+
if (xprt_connected(xprt))
2264+
goto out;
2265+
if (test_and_clear_bit(XPRT_SOCK_CONNECT_SENT,
2266+
&transport->sock_state) ||
2267+
!sock) {
2268+
xs_reset_transport(transport);
2269+
sock = xs_create_sock(xprt, transport, xs_addr(xprt)->sa_family,
2270+
SOCK_STREAM, IPPROTO_TCP, true);
22672271
if (IS_ERR(sock)) {
22682272
status = PTR_ERR(sock);
22692273
goto out;
@@ -2294,6 +2298,8 @@ static void xs_tcp_setup_socket(struct work_struct *work)
22942298
break;
22952299
case 0:
22962300
case -EINPROGRESS:
2301+
set_bit(XPRT_SOCK_CONNECT_SENT, &transport->sock_state);
2302+
fallthrough;
22972303
case -EALREADY:
22982304
xprt_unlock_connect(xprt, transport);
22992305
return;
@@ -2345,13 +2351,9 @@ static void xs_connect(struct rpc_xprt *xprt, struct rpc_task *task)
23452351

23462352
WARN_ON_ONCE(!xprt_lock_connect(xprt, task, transport));
23472353

2348-
if (transport->sock != NULL && !xprt_connecting(xprt)) {
2354+
if (transport->sock != NULL) {
23492355
dprintk("RPC: xs_connect delayed xprt %p for %lu "
2350-
"seconds\n",
2351-
xprt, xprt->reestablish_timeout / HZ);
2352-
2353-
/* Start by resetting any existing state */
2354-
xs_reset_transport(transport);
2356+
"seconds\n", xprt, xprt->reestablish_timeout / HZ);
23552357

23562358
delay = xprt_reconnect_delay(xprt);
23572359
xprt_reconnect_backoff(xprt, XS_TCP_INIT_REEST_TO);

0 commit comments

Comments
 (0)