Merge branch 'rds'
Sowmini Varadhan says: ==================== net/rds: RDS-TCP robustness fixes This patch-set contains bug fixes for state-recovery at the RDS layer when the underlying transport is TCP and the TCP state at one of the endpoints is reset V2 changes: DaveM comments to reduce memory footprint, follow NFS/RPC model where possible. Added test-case #3 Without the changes in this set, when one of the endpoints is reset, the existing code does not correctly clean up RDS socket state for stale connections, resulting in some unstable, timing-dependant behavior on the wire, including an infinite exchange of 3WHs back-and-forth, and a resulting potential to never converge RDS state. Test cases used to verify the changes in this set are: 1. Start rds client/server applications on two participating nodes, node1 and node2. After at least one packet has been sent (to establish the TCP connection), restart the rds_tcp module on the client, and now resend packets. Tcpdump should show server sending a FIN for the "old" client port, and clean connection establishment/exchange for the new client port. 2. At the end of step 1, restart rds srever on node2, and start client on node1, make sure using tcpdump, 'netstat -an|grep 16385' that packets flow correctly. 3. start RDS client/server application on two participating nodes, and repeat steps 1 and 2, but this time, simulate node failure by doing "ifconfig <intf> down", so no FIN is sent. ==================== Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
d3196a2468
|
@ -126,7 +126,10 @@ static struct rds_connection *__rds_conn_create(__be32 laddr, __be32 faddr,
|
||||||
struct rds_transport *loop_trans;
|
struct rds_transport *loop_trans;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int ret;
|
int ret;
|
||||||
|
struct rds_transport *otrans = trans;
|
||||||
|
|
||||||
|
if (!is_outgoing && otrans->t_type == RDS_TRANS_TCP)
|
||||||
|
goto new_conn;
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
conn = rds_conn_lookup(head, laddr, faddr, trans);
|
conn = rds_conn_lookup(head, laddr, faddr, trans);
|
||||||
if (conn && conn->c_loopback && conn->c_trans != &rds_loop_transport &&
|
if (conn && conn->c_loopback && conn->c_trans != &rds_loop_transport &&
|
||||||
|
@ -142,6 +145,7 @@ static struct rds_connection *__rds_conn_create(__be32 laddr, __be32 faddr,
|
||||||
if (conn)
|
if (conn)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
new_conn:
|
||||||
conn = kmem_cache_zalloc(rds_conn_slab, gfp);
|
conn = kmem_cache_zalloc(rds_conn_slab, gfp);
|
||||||
if (!conn) {
|
if (!conn) {
|
||||||
conn = ERR_PTR(-ENOMEM);
|
conn = ERR_PTR(-ENOMEM);
|
||||||
|
@ -230,13 +234,22 @@ static struct rds_connection *__rds_conn_create(__be32 laddr, __be32 faddr,
|
||||||
/* Creating normal conn */
|
/* Creating normal conn */
|
||||||
struct rds_connection *found;
|
struct rds_connection *found;
|
||||||
|
|
||||||
|
if (!is_outgoing && otrans->t_type == RDS_TRANS_TCP)
|
||||||
|
found = NULL;
|
||||||
|
else
|
||||||
found = rds_conn_lookup(head, laddr, faddr, trans);
|
found = rds_conn_lookup(head, laddr, faddr, trans);
|
||||||
if (found) {
|
if (found) {
|
||||||
trans->conn_free(conn->c_transport_data);
|
trans->conn_free(conn->c_transport_data);
|
||||||
kmem_cache_free(rds_conn_slab, conn);
|
kmem_cache_free(rds_conn_slab, conn);
|
||||||
conn = found;
|
conn = found;
|
||||||
} else {
|
} else {
|
||||||
|
if ((is_outgoing && otrans->t_type == RDS_TRANS_TCP) ||
|
||||||
|
(otrans->t_type != RDS_TRANS_TCP)) {
|
||||||
|
/* Only the active side should be added to
|
||||||
|
* reconnect list for TCP.
|
||||||
|
*/
|
||||||
hlist_add_head_rcu(&conn->c_hash_node, head);
|
hlist_add_head_rcu(&conn->c_hash_node, head);
|
||||||
|
}
|
||||||
rds_cong_add_conn(conn);
|
rds_cong_add_conn(conn);
|
||||||
rds_conn_count++;
|
rds_conn_count++;
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,6 +62,7 @@ void rds_tcp_state_change(struct sock *sk)
|
||||||
case TCP_ESTABLISHED:
|
case TCP_ESTABLISHED:
|
||||||
rds_connect_complete(conn);
|
rds_connect_complete(conn);
|
||||||
break;
|
break;
|
||||||
|
case TCP_CLOSE_WAIT:
|
||||||
case TCP_CLOSE:
|
case TCP_CLOSE:
|
||||||
rds_conn_drop(conn);
|
rds_conn_drop(conn);
|
||||||
default:
|
default:
|
||||||
|
|
|
@ -45,12 +45,45 @@ static void rds_tcp_accept_worker(struct work_struct *work);
|
||||||
static DECLARE_WORK(rds_tcp_listen_work, rds_tcp_accept_worker);
|
static DECLARE_WORK(rds_tcp_listen_work, rds_tcp_accept_worker);
|
||||||
static struct socket *rds_tcp_listen_sock;
|
static struct socket *rds_tcp_listen_sock;
|
||||||
|
|
||||||
|
static int rds_tcp_keepalive(struct socket *sock)
|
||||||
|
{
|
||||||
|
/* values below based on xs_udp_default_timeout */
|
||||||
|
int keepidle = 5; /* send a probe 'keepidle' secs after last data */
|
||||||
|
int keepcnt = 5; /* number of unack'ed probes before declaring dead */
|
||||||
|
int keepalive = 1;
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
ret = kernel_setsockopt(sock, SOL_SOCKET, SO_KEEPALIVE,
|
||||||
|
(char *)&keepalive, sizeof(keepalive));
|
||||||
|
if (ret < 0)
|
||||||
|
goto bail;
|
||||||
|
|
||||||
|
ret = kernel_setsockopt(sock, IPPROTO_TCP, TCP_KEEPCNT,
|
||||||
|
(char *)&keepcnt, sizeof(keepcnt));
|
||||||
|
if (ret < 0)
|
||||||
|
goto bail;
|
||||||
|
|
||||||
|
ret = kernel_setsockopt(sock, IPPROTO_TCP, TCP_KEEPIDLE,
|
||||||
|
(char *)&keepidle, sizeof(keepidle));
|
||||||
|
if (ret < 0)
|
||||||
|
goto bail;
|
||||||
|
|
||||||
|
/* KEEPINTVL is the interval between successive probes. We follow
|
||||||
|
* the model in xs_tcp_finish_connecting() and re-use keepidle.
|
||||||
|
*/
|
||||||
|
ret = kernel_setsockopt(sock, IPPROTO_TCP, TCP_KEEPINTVL,
|
||||||
|
(char *)&keepidle, sizeof(keepidle));
|
||||||
|
bail:
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static int rds_tcp_accept_one(struct socket *sock)
|
static int rds_tcp_accept_one(struct socket *sock)
|
||||||
{
|
{
|
||||||
struct socket *new_sock = NULL;
|
struct socket *new_sock = NULL;
|
||||||
struct rds_connection *conn;
|
struct rds_connection *conn;
|
||||||
int ret;
|
int ret;
|
||||||
struct inet_sock *inet;
|
struct inet_sock *inet;
|
||||||
|
struct rds_tcp_connection *rs_tcp;
|
||||||
|
|
||||||
ret = sock_create_lite(sock->sk->sk_family, sock->sk->sk_type,
|
ret = sock_create_lite(sock->sk->sk_family, sock->sk->sk_type,
|
||||||
sock->sk->sk_protocol, &new_sock);
|
sock->sk->sk_protocol, &new_sock);
|
||||||
|
@ -63,6 +96,10 @@ static int rds_tcp_accept_one(struct socket *sock)
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
ret = rds_tcp_keepalive(new_sock);
|
||||||
|
if (ret < 0)
|
||||||
|
goto out;
|
||||||
|
|
||||||
rds_tcp_tune(new_sock);
|
rds_tcp_tune(new_sock);
|
||||||
|
|
||||||
inet = inet_sk(new_sock->sk);
|
inet = inet_sk(new_sock->sk);
|
||||||
|
@ -77,6 +114,15 @@ static int rds_tcp_accept_one(struct socket *sock)
|
||||||
ret = PTR_ERR(conn);
|
ret = PTR_ERR(conn);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
/* An incoming SYN request came in, and TCP just accepted it.
|
||||||
|
* We always create a new conn for listen side of TCP, and do not
|
||||||
|
* add it to the c_hash_list.
|
||||||
|
*
|
||||||
|
* If the client reboots, this conn will need to be cleaned up.
|
||||||
|
* rds_tcp_state_change() will do that cleanup
|
||||||
|
*/
|
||||||
|
rs_tcp = (struct rds_tcp_connection *)conn->c_transport_data;
|
||||||
|
WARN_ON(!rs_tcp || rs_tcp->t_sock);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* see the comment above rds_queue_delayed_reconnect()
|
* see the comment above rds_queue_delayed_reconnect()
|
||||||
|
|
Loading…
Reference in New Issue