From a3c9b478e2261002e47bdf3fcad51037512128b6 Mon Sep 17 00:00:00 2001 From: Antioch Peverell Date: Mon, 22 Feb 2021 15:18:09 +0000 Subject: [PATCH] [5.0.x] inefficient locking on recv of peers lists can result in failure to get peers lock (#3566) (#3570) * fix for: inefficient locking of peers lists can result in failure to get peers lock * dont hold the peers Vec lock while writing to the peers lmdb Co-authored-by: Blade Doyle --- p2p/src/peers.rs | 53 +++++++++++++++++++++++++++++++----------------- p2p/src/store.rs | 9 ++++++++ 2 files changed, 43 insertions(+), 19 deletions(-) diff --git a/p2p/src/peers.rs b/p2p/src/peers.rs index e045baca0..f81134f7d 100644 --- a/p2p/src/peers.rs +++ b/p2p/src/peers.rs @@ -59,23 +59,29 @@ impl Peers { /// Adds the peer to our internal peer mapping. Note that the peer is still /// returned so the server can run it. pub fn add_connected(&self, peer: Arc) -> Result<(), Error> { - let mut peers = self.peers.try_write_for(LOCK_TIMEOUT).ok_or_else(|| { - error!("add_connected: failed to get peers lock"); - Error::Timeout - })?; - let peer_data = PeerData { - addr: peer.info.addr, - capabilities: peer.info.capabilities, - user_agent: peer.info.user_agent.clone(), - flags: State::Healthy, - last_banned: 0, - ban_reason: ReasonForBan::None, - last_connected: Utc::now().timestamp(), - }; + let peer_data: PeerData; + { + // Scope for peers vector lock - dont hold the peers lock while adding to lmdb + let mut peers = self.peers.try_write_for(LOCK_TIMEOUT).ok_or_else(|| { + error!("add_connected: failed to get peers lock"); + Error::Timeout + })?; + peer_data = PeerData { + addr: peer.info.addr, + capabilities: peer.info.capabilities, + user_agent: peer.info.user_agent.clone(), + flags: State::Healthy, + last_banned: 0, + ban_reason: ReasonForBan::None, + last_connected: Utc::now().timestamp(), + }; + debug!("Adding newly connected peer {}.", peer_data.addr); + peers.insert(peer_data.addr, peer); + } debug!("Saving newly connected peer {}.", peer_data.addr); - self.save_peer(&peer_data)?; - peers.insert(peer_data.addr, peer); - + if let Err(e) = self.save_peer(&peer_data) { + error!("Could not save connected peer address: {:?}", e); + } Ok(()) } @@ -136,8 +142,10 @@ impl Peers { } /// Ban a peer, disconnecting it if we're currently connected pub fn ban_peer(&self, peer_addr: PeerAddr, ban_reason: ReasonForBan) -> Result<(), Error> { + // Update the peer in peers db self.update_state(peer_addr, State::Banned)?; + // Update the peer in the peers Vec match self.get_connected_peer(peer_addr) { Some(peer) => { debug!("Banning peer {}", peer_addr); @@ -295,6 +303,11 @@ impl Peers { self.store.save_peer(p).map_err(From::from) } + /// Saves updated information about mulitple peers in batch + pub fn save_peers(&self, p: Vec) -> Result<(), Error> { + self.store.save_peers(p).map_err(From::from) + } + /// Updates the state of a peer in store pub fn update_state(&self, peer_addr: PeerAddr, new_state: State) -> Result<(), Error> { self.store @@ -667,6 +680,7 @@ impl NetAdapter for Peers { /// A list of peers has been received from one of our peers. fn peer_addrs_received(&self, peer_addrs: Vec) { trace!("Received {} peer addrs, saving.", peer_addrs.len()); + let mut to_save: Vec = Vec::new(); for pa in peer_addrs { if let Ok(e) = self.exists_peer(pa) { if e { @@ -682,9 +696,10 @@ impl NetAdapter for Peers { ban_reason: ReasonForBan::None, last_connected: Utc::now().timestamp(), }; - if let Err(e) = self.save_peer(&peer) { - error!("Could not save received peer address: {:?}", e); - } + to_save.push(peer); + } + if let Err(e) = self.save_peers(to_save) { + error!("Could not save received peer addresses: {:?}", e); } } diff --git a/p2p/src/store.rs b/p2p/src/store.rs index 686ceeff2..17e4211ee 100644 --- a/p2p/src/store.rs +++ b/p2p/src/store.rs @@ -127,6 +127,15 @@ impl PeerStore { batch.commit() } + pub fn save_peers(&self, p: Vec) -> Result<(), Error> { + let batch = self.db.batch()?; + for pd in p { + debug!("save_peers: {:?} marked {:?}", pd.addr, pd.flags); + batch.put_ser(&peer_key(pd.addr)[..], &pd)?; + } + batch.commit() + } + pub fn get_peer(&self, peer_addr: PeerAddr) -> Result { option_to_not_found(self.db.get_ser(&peer_key(peer_addr)[..]), || { format!("Peer at address: {}", peer_addr)