mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	RDS: Fix send locking issue
Fix a deadlock between rds_rdma_send_complete() and rds_send_remove_from_sock() when rds socket lock and rds message lock are acquired out-of-order. Signed-off-by: Tina Yang <Tina.Yang@oracle.com> Signed-off-by: Andy Grover <andy.grover@oracle.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									2e7b3b9945
								
							
						
					
					
						commit
						048c15e641
					
				
					 1 changed files with 2 additions and 3 deletions
				
			
		| 
						 | 
					@ -533,14 +533,13 @@ void rds_send_remove_from_sock(struct list_head *messages, int status)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (rs != rm->m_rs) {
 | 
							if (rs != rm->m_rs) {
 | 
				
			||||||
			if (rs) {
 | 
								if (rs) {
 | 
				
			||||||
				spin_unlock(&rs->rs_lock);
 | 
					 | 
				
			||||||
				rds_wake_sk_sleep(rs);
 | 
									rds_wake_sk_sleep(rs);
 | 
				
			||||||
				sock_put(rds_rs_to_sk(rs));
 | 
									sock_put(rds_rs_to_sk(rs));
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
			rs = rm->m_rs;
 | 
								rs = rm->m_rs;
 | 
				
			||||||
			spin_lock(&rs->rs_lock);
 | 
					 | 
				
			||||||
			sock_hold(rds_rs_to_sk(rs));
 | 
								sock_hold(rds_rs_to_sk(rs));
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
							spin_lock(&rs->rs_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (test_and_clear_bit(RDS_MSG_ON_SOCK, &rm->m_flags)) {
 | 
							if (test_and_clear_bit(RDS_MSG_ON_SOCK, &rm->m_flags)) {
 | 
				
			||||||
			struct rds_rdma_op *ro = rm->m_rdma_op;
 | 
								struct rds_rdma_op *ro = rm->m_rdma_op;
 | 
				
			||||||
| 
						 | 
					@ -560,6 +559,7 @@ void rds_send_remove_from_sock(struct list_head *messages, int status)
 | 
				
			||||||
			rds_message_put(rm);
 | 
								rds_message_put(rm);
 | 
				
			||||||
			rm->m_rs = NULL;
 | 
								rm->m_rs = NULL;
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
							spin_unlock(&rs->rs_lock);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
unlock_and_drop:
 | 
					unlock_and_drop:
 | 
				
			||||||
		spin_unlock(&rm->m_rs_lock);
 | 
							spin_unlock(&rm->m_rs_lock);
 | 
				
			||||||
| 
						 | 
					@ -567,7 +567,6 @@ void rds_send_remove_from_sock(struct list_head *messages, int status)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (rs) {
 | 
						if (rs) {
 | 
				
			||||||
		spin_unlock(&rs->rs_lock);
 | 
					 | 
				
			||||||
		rds_wake_sk_sleep(rs);
 | 
							rds_wake_sk_sleep(rs);
 | 
				
			||||||
		sock_put(rds_rs_to_sk(rs));
 | 
							sock_put(rds_rs_to_sk(rs));
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue