diff options
Diffstat (limited to 'net/tipc/server.c')
| -rw-r--r-- | net/tipc/server.c | 48 | 
1 files changed, 21 insertions, 27 deletions
| diff --git a/net/tipc/server.c b/net/tipc/server.c index 215849ce453d..3cd6402e812c 100644 --- a/net/tipc/server.c +++ b/net/tipc/server.c @@ -86,12 +86,12 @@ struct outqueue_entry {  static void tipc_recv_work(struct work_struct *work);  static void tipc_send_work(struct work_struct *work);  static void tipc_clean_outqueues(struct tipc_conn *con); -static void tipc_sock_release(struct tipc_conn *con);  static void tipc_conn_kref_release(struct kref *kref)  {  	struct tipc_conn *con = container_of(kref, struct tipc_conn, kref); -	struct sockaddr_tipc *saddr = con->server->saddr; +	struct tipc_server *s = con->server; +	struct sockaddr_tipc *saddr = s->saddr;  	struct socket *sock = con->sock;  	struct sock *sk; @@ -103,9 +103,13 @@ static void tipc_conn_kref_release(struct kref *kref)  		}  		saddr->scope = -TIPC_NODE_SCOPE;  		kernel_bind(sock, (struct sockaddr *)saddr, sizeof(*saddr)); -		tipc_sock_release(con);  		sock_release(sock);  		con->sock = NULL; + +		spin_lock_bh(&s->idr_lock); +		idr_remove(&s->conn_idr, con->conid); +		s->idr_in_use--; +		spin_unlock_bh(&s->idr_lock);  	}  	tipc_clean_outqueues(con); @@ -128,8 +132,10 @@ static struct tipc_conn *tipc_conn_lookup(struct tipc_server *s, int conid)  	spin_lock_bh(&s->idr_lock);  	con = idr_find(&s->conn_idr, conid); -	if (con) +	if (con && test_bit(CF_CONNECTED, &con->flags))  		conn_get(con); +	else +		con = NULL;  	spin_unlock_bh(&s->idr_lock);  	return con;  } @@ -186,26 +192,15 @@ static void tipc_unregister_callbacks(struct tipc_conn *con)  	write_unlock_bh(&sk->sk_callback_lock);  } -static void tipc_sock_release(struct tipc_conn *con) -{ -	struct tipc_server *s = con->server; - -	if (con->conid) -		s->tipc_conn_release(con->conid, con->usr_data); - -	tipc_unregister_callbacks(con); -} -  static void tipc_close_conn(struct tipc_conn *con)  {  	struct tipc_server *s = con->server;  	if (test_and_clear_bit(CF_CONNECTED, &con->flags)) { +		tipc_unregister_callbacks(con); -		spin_lock_bh(&s->idr_lock); -		idr_remove(&s->conn_idr, con->conid); -		s->idr_in_use--; -		spin_unlock_bh(&s->idr_lock); +		if (con->conid) +			s->tipc_conn_release(con->conid, con->usr_data);  		/* We shouldn't flush pending works as we may be in the  		 * thread. In fact the races with pending rx/tx work structs @@ -458,6 +453,11 @@ int tipc_conn_sendmsg(struct tipc_server *s, int conid,  	if (!con)  		return -EINVAL; +	if (!test_bit(CF_CONNECTED, &con->flags)) { +		conn_put(con); +		return 0; +	} +  	e = tipc_alloc_entry(data, len);  	if (!e) {  		conn_put(con); @@ -471,12 +471,8 @@ int tipc_conn_sendmsg(struct tipc_server *s, int conid,  	list_add_tail(&e->list, &con->outqueue);  	spin_unlock_bh(&con->outqueue_lock); -	if (test_bit(CF_CONNECTED, &con->flags)) { -		if (!queue_work(s->send_wq, &con->swork)) -			conn_put(con); -	} else { +	if (!queue_work(s->send_wq, &con->swork))  		conn_put(con); -	}  	return 0;  } @@ -500,7 +496,7 @@ static void tipc_send_to_sock(struct tipc_conn *con)  	int ret;  	spin_lock_bh(&con->outqueue_lock); -	while (1) { +	while (test_bit(CF_CONNECTED, &con->flags)) {  		e = list_entry(con->outqueue.next, struct outqueue_entry,  			       list);  		if ((struct list_head *) e == &con->outqueue) @@ -623,14 +619,12 @@ int tipc_server_start(struct tipc_server *s)  void tipc_server_stop(struct tipc_server *s)  {  	struct tipc_conn *con; -	int total = 0;  	int id;  	spin_lock_bh(&s->idr_lock); -	for (id = 0; total < s->idr_in_use; id++) { +	for (id = 0; s->idr_in_use; id++) {  		con = idr_find(&s->conn_idr, id);  		if (con) { -			total++;  			spin_unlock_bh(&s->idr_lock);  			tipc_close_conn(con);  			spin_lock_bh(&s->idr_lock); | 
