From 1a6b32edc36ea583565fc59e9c8ae061aea2520a Mon Sep 17 00:00:00 2001 From: Maksim Greshnyakov Date: Wed, 13 Nov 2024 10:20:09 +0100 Subject: [PATCH] fix(validator): remove potentially deadlock --- .../src/validator/impls/std_impl/session.rs | 27 +++++++++++++------ 1 file changed, 19 insertions(+), 8 deletions(-) diff --git a/collator/src/validator/impls/std_impl/session.rs b/collator/src/validator/impls/std_impl/session.rs index c92b3d0d8..4bf79676a 100644 --- a/collator/src/validator/impls/std_impl/session.rs +++ b/collator/src/validator/impls/std_impl/session.rs @@ -183,8 +183,8 @@ impl ValidatorSession { let state = &self.inner.state; - let entry = state.block_signatures.entry(block_id.seqno); - if let tycho_util::DashMapEntry::Occupied(_) = entry { + // first check for prevent slow operations + if state.block_signatures.contains_key(&block_id.seqno) { anyhow::bail!( "block validation is already in progress. \ session_id={}, block_id={:?}", @@ -195,10 +195,9 @@ impl ValidatorSession { let cached = state .cached_signatures - .remove(&block_id.seqno) - .map(|(_, value)| value); + .get(&block_id.seqno) + .map(|entry| entry.clone()); - // Prepare block signatures let block_signatures = { match &cached { Some(cached) => self.reuse_signatures(block_id, cached.clone()).await, @@ -207,7 +206,21 @@ impl ValidatorSession { .build(block_id, state.weight_threshold) }; - entry.or_insert(block_signatures.clone()); + // second check for prevent data races + match state.block_signatures.entry(block_id.seqno) { + tycho_util::DashMapEntry::Occupied(_) => { + anyhow::bail!( + "block validation is already in progress. \ + session_id={}, block_id={:?}", + self.inner.session_id, + block_id + ); + } + tycho_util::DashMapEntry::Vacant(signatures) => { + signatures.insert(block_signatures.clone()); + state.cached_signatures.remove(&block_id.seqno); + } + } // At this point the following is true: // - All new signatures will be stored (and validated) in the block; @@ -294,7 +307,6 @@ impl ValidatorSession { debug_assert!(prev.is_none(), "duplicate signature in result"); total_weight += validator_info.weight; } - tracing::info!(target: tracing_targets::VALIDATOR, "finished"); Ok(ValidationStatus::Complete(ValidationComplete { signatures: result, @@ -797,7 +809,6 @@ impl ExchangeSignatures for SessionState { action, "exchanged signatures" ); - Ok(result) } }