From: Sowmini Varadhan Date: Fri, 16 Jun 2017 19:08:29 +0000 (-0700) Subject: rds: tcp: Take explicit refcounts on struct net X-Git-Tag: v4.1.12-104.0.20170618_1145~13 X-Git-Url: https://www.infradead.org/git/?a=commitdiff_plain;h=d0e7eab11c39f362737cc9b7921221501bc859ff;p=users%2Fjedix%2Flinux-maple.git rds: tcp: Take explicit refcounts on struct net It is incorrect for the rds_connection to piggyback on the sock_net() refcount for the netns because this gives rise to a chicken-and-egg problem during rds_conn_destroy. Instead explicitly take a ref on the net, and hold the netns down till the connection tear-down is complete. Orabug: 26289770 (Cherry-pick of upstream 8edc3affc0770886c7bfb3436b0fdd09bce13167) Reported-by: Dmitry Vyukov Signed-off-by: David S. Miller Signed-off-by: Sowmini Varadhan --- diff --git a/net/rds/connection.c b/net/rds/connection.c index 6cd8b0e7ce79..35714ce80680 100644 --- a/net/rds/connection.c +++ b/net/rds/connection.c @@ -522,6 +522,7 @@ void rds_conn_destroy(struct rds_connection *conn, int shutdown) */ rds_cong_remove_conn(conn); + put_net(conn->c_net); kmem_cache_free(rds_conn_slab, conn); spin_lock_irqsave(&rds_conn_lock, flags); diff --git a/net/rds/rds.h b/net/rds/rds.h index 511f1f583d08..c0678c726719 100644 --- a/net/rds/rds.h +++ b/net/rds/rds.h @@ -318,7 +318,7 @@ struct rds_connection { /* Protocol version */ unsigned int c_proposed_version; unsigned int c_version; - possible_net_t c_net; + struct net *c_net; /* Qos support */ u8 c_tos; @@ -337,13 +337,13 @@ struct rds_connection { static inline struct net *rds_conn_net(struct rds_connection *conn) { - return read_pnet(&conn->c_net); + return conn->c_net; } static inline void rds_conn_net_set(struct rds_connection *conn, struct net *net) { - write_pnet(&conn->c_net, net); + conn->c_net = get_net(net); } #define RDS_FLAG_CONG_BITMAP 0x01 diff --git a/net/rds/tcp.c b/net/rds/tcp.c index ced67c57821a..db4d5ad590d6 100644 --- a/net/rds/tcp.c +++ b/net/rds/tcp.c @@ -527,7 +527,7 @@ static void rds_tcp_kill_sock(struct net *net) flush_work(&rtn->rds_tcp_accept_w); spin_lock_irq(&rds_tcp_conn_lock); list_for_each_entry_safe(tc, _tc, &rds_tcp_conn_list, t_tcp_node) { - struct net *c_net = read_pnet(&tc->t_cpath->cp_conn->c_net); + struct net *c_net = tc->t_cpath->cp_conn->c_net; if (net != c_net || !tc->t_sock) continue; @@ -582,7 +582,7 @@ static void rds_tcp_sysctl_reset(struct net *net) spin_lock_irq(&rds_tcp_conn_lock); list_for_each_entry_safe(tc, _tc, &rds_tcp_conn_list, t_tcp_node) { - struct net *c_net = read_pnet(&tc->t_cpath->cp_conn->c_net); + struct net *c_net = tc->t_cpath->cp_conn->c_net; if (net != c_net || !tc->t_sock) continue;