Skip to content

Commit 3f71753

Browse files
mmhalgregkh
authored andcommitted
vsock: Ignore signal/timeout on connect() if already established
[ Upstream commit 002541e ] During connect(), acting on a signal/timeout by disconnecting an already established socket leads to several issues: 1. connect() invoking vsock_transport_cancel_pkt() -> virtio_transport_purge_skbs() may race with sendmsg() invoking virtio_transport_get_credit(). This results in a permanently elevated `vvs->bytes_unsent`. Which, in turn, confuses the SOCK_LINGER handling. 2. connect() resetting a connected socket's state may race with socket being placed in a sockmap. A disconnected socket remaining in a sockmap breaks sockmap's assumptions. And gives rise to WARNs. 3. connect() transitioning SS_CONNECTED -> SS_UNCONNECTED allows for a transport change/drop after TCP_ESTABLISHED. Which poses a problem for any simultaneous sendmsg() or connect() and may result in a use-after-free/null-ptr-deref. Do not disconnect socket on signal/timeout. Keep the logic for unconnected sockets: they don't linger, can't be placed in a sockmap, are rejected by sendmsg(). [1]: https://lore.kernel.org/netdev/e07fd95c-9a38-4eea-9638-133e38c2ec9b@rbox.co/ [2]: https://lore.kernel.org/netdev/20250317-vsock-trans-signal-race-v4-0-fc8837f3f1d4@rbox.co/ [3]: https://lore.kernel.org/netdev/60f1b7db-3099-4f6a-875e-af9f6ef194f6@rbox.co/ Fixes: d021c34 ("VSOCK: Introduce VM Sockets") Signed-off-by: Michal Luczaj <mhal@rbox.co> Reviewed-by: Stefano Garzarella <sgarzare@redhat.com> Link: https://patch.msgid.link/20251119-vsock-interrupted-connect-v2-1-70734cf1233f@rbox.co Signed-off-by: Jakub Kicinski <kuba@kernel.org> Signed-off-by: Sasha Levin <sashal@kernel.org>
1 parent 06f1dd1 commit 3f71753

1 file changed

Lines changed: 31 additions & 9 deletions

File tree

net/vmw_vsock/af_vsock.c

Lines changed: 31 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -1235,18 +1235,40 @@ static int vsock_stream_connect(struct socket *sock, struct sockaddr *addr,
12351235
timeout = schedule_timeout(timeout);
12361236
lock_sock(sk);
12371237

1238-
if (signal_pending(current)) {
1239-
err = sock_intr_errno(timeout);
1240-
sk->sk_state = sk->sk_state == TCP_ESTABLISHED ? TCP_CLOSING : TCP_CLOSE;
1241-
sock->state = SS_UNCONNECTED;
1242-
vsock_transport_cancel_pkt(vsk);
1243-
vsock_remove_connected(vsk);
1244-
goto out_wait;
1245-
} else if ((sk->sk_state != TCP_ESTABLISHED) && (timeout == 0)) {
1246-
err = -ETIMEDOUT;
1238+
/* Connection established. Whatever happens to socket once we
1239+
* release it, that's not connect()'s concern. No need to go
1240+
* into signal and timeout handling. Call it a day.
1241+
*
1242+
* Note that allowing to "reset" an already established socket
1243+
* here is racy and insecure.
1244+
*/
1245+
if (sk->sk_state == TCP_ESTABLISHED)
1246+
break;
1247+
1248+
/* If connection was _not_ established and a signal/timeout came
1249+
* to be, we want the socket's state reset. User space may want
1250+
* to retry.
1251+
*
1252+
* sk_state != TCP_ESTABLISHED implies that socket is not on
1253+
* vsock_connected_table. We keep the binding and the transport
1254+
* assigned.
1255+
*/
1256+
if (signal_pending(current) || timeout == 0) {
1257+
err = timeout == 0 ? -ETIMEDOUT : sock_intr_errno(timeout);
1258+
1259+
/* Listener might have already responded with
1260+
* VIRTIO_VSOCK_OP_RESPONSE. Its handling expects our
1261+
* sk_state == TCP_SYN_SENT, which hereby we break.
1262+
* In such case VIRTIO_VSOCK_OP_RST will follow.
1263+
*/
12471264
sk->sk_state = TCP_CLOSE;
12481265
sock->state = SS_UNCONNECTED;
1266+
1267+
/* Try to cancel VIRTIO_VSOCK_OP_REQUEST skb sent out by
1268+
* transport->connect().
1269+
*/
12491270
vsock_transport_cancel_pkt(vsk);
1271+
12501272
goto out_wait;
12511273
}
12521274

0 commit comments

Comments
 (0)