diff options
author | Herton R. Krzesinski <herton@redhat.com> | 2014-10-01 17:49:53 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2014-10-03 15:51:59 -0400 |
commit | eb74cc97b830c1e438dc1d6b049f17bdb2b9aae5 (patch) | |
tree | f3b45a40a460a4a70ad9436ded5b1daa1da802fc | |
parent | 310886dd5fa3606d9325b10caf7c8ba5e9f9ab03 (diff) |
net/rds: do proper house keeping if connection fails in rds_tcp_conn_connect
I see two problems if we consider the sock->ops->connect attempt to fail in
rds_tcp_conn_connect. The first issue is that for example we don't remove the
previously added rds_tcp_connection item to rds_tcp_tc_list at
rds_tcp_set_callbacks, which means that on a next reconnect attempt for the
same rds_connection, when rds_tcp_conn_connect is called we can again call
rds_tcp_set_callbacks, resulting in duplicated items on rds_tcp_tc_list,
leading to list corruption: to avoid this just make sure we call
properly rds_tcp_restore_callbacks before we exit. The second issue
is that we should also release the sock properly, by setting sock = NULL
only if we are returning without error.
Signed-off-by: Herton R. Krzesinski <herton@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | net/rds/tcp_connect.c | 5 |
1 files changed, 4 insertions, 1 deletions
diff --git a/net/rds/tcp_connect.c b/net/rds/tcp_connect.c index a65ee78db0c5..f9f564a6c960 100644 --- a/net/rds/tcp_connect.c +++ b/net/rds/tcp_connect.c | |||
@@ -106,11 +106,14 @@ int rds_tcp_conn_connect(struct rds_connection *conn) | |||
106 | rds_tcp_set_callbacks(sock, conn); | 106 | rds_tcp_set_callbacks(sock, conn); |
107 | ret = sock->ops->connect(sock, (struct sockaddr *)&dest, sizeof(dest), | 107 | ret = sock->ops->connect(sock, (struct sockaddr *)&dest, sizeof(dest), |
108 | O_NONBLOCK); | 108 | O_NONBLOCK); |
109 | sock = NULL; | ||
110 | 109 | ||
111 | rdsdebug("connect to address %pI4 returned %d\n", &conn->c_faddr, ret); | 110 | rdsdebug("connect to address %pI4 returned %d\n", &conn->c_faddr, ret); |
112 | if (ret == -EINPROGRESS) | 111 | if (ret == -EINPROGRESS) |
113 | ret = 0; | 112 | ret = 0; |
113 | if (ret == 0) | ||
114 | sock = NULL; | ||
115 | else | ||
116 | rds_tcp_restore_callbacks(sock, conn->c_transport_data); | ||
114 | 117 | ||
115 | out: | 118 | out: |
116 | if (sock) | 119 | if (sock) |