Skip to content

Commit 03c8c69

Browse files
author
MarcoFalke
committed
Merge bitcoin/bitcoin#24177: validation, refactor: add missing thread safety lock assertions
f485a07 Add missing thread safety lock assertions in validation.h (Jon Atack) 37af8a2 Add missing thread safety lock assertions in validation.cpp (Jon Atack) Pull request description: A number of functions in validation.{h,cpp} have a thread safety lock annotation in the declaration but are missing the corresponding run-time lock assertion in the definition. ACKs for top commit: hebasto: re-ACK f485a07, only suggested change since my [previous](bitcoin/bitcoin#24177 (review)) review. vasild: ACK f485a07 Tree-SHA512: c86c0c0e8fe6ec7ae9ed9890f1dd7d042aa482ecf99feb6679a670aa004f6e9a99f7bc047205a34968fab7f1f841898c59b48c3ed6245c166e3b5abbf0867445
2 parents b304b65 + f485a07 commit 03c8c69

File tree

2 files changed

+54
-11
lines changed

2 files changed

+54
-11
lines changed

src/validation.cpp

+44-7
Original file line numberDiff line numberDiff line change
@@ -286,8 +286,10 @@ bool CheckSequenceLocks(CBlockIndex* tip,
286286
static unsigned int GetBlockScriptFlags(const CBlockIndex* pindex, const Consensus::Params& chainparams);
287287

288288
static void LimitMempoolSize(CTxMemPool& pool, CCoinsViewCache& coins_cache, size_t limit, std::chrono::seconds age)
289-
EXCLUSIVE_LOCKS_REQUIRED(pool.cs, ::cs_main)
289+
EXCLUSIVE_LOCKS_REQUIRED(::cs_main, pool.cs)
290290
{
291+
AssertLockHeld(::cs_main);
292+
AssertLockHeld(pool.cs);
291293
int expired = pool.Expire(GetTime<std::chrono::seconds>() - age);
292294
if (expired != 0) {
293295
LogPrint(BCLog::MEMPOOL, "Expired %i transactions from the memory pool\n", expired);
@@ -628,8 +630,10 @@ class MemPoolAccept
628630
EXCLUSIVE_LOCKS_REQUIRED(cs_main, m_pool.cs);
629631

630632
// Compare a package's feerate against minimum allowed.
631-
bool CheckFeeRate(size_t package_size, CAmount package_fee, TxValidationState& state) EXCLUSIVE_LOCKS_REQUIRED(cs_main, m_pool.cs)
633+
bool CheckFeeRate(size_t package_size, CAmount package_fee, TxValidationState& state) EXCLUSIVE_LOCKS_REQUIRED(::cs_main, m_pool.cs)
632634
{
635+
AssertLockHeld(::cs_main);
636+
AssertLockHeld(m_pool.cs);
633637
CAmount mempoolRejectFee = m_pool.GetMinFee(gArgs.GetIntArg("-maxmempool", DEFAULT_MAX_MEMPOOL_SIZE) * 1000000).GetFee(package_size);
634638
if (mempoolRejectFee > 0 && package_fee < mempoolRejectFee) {
635639
return state.Invalid(TxValidationResult::TX_MEMPOOL_POLICY, "mempool min fee not met", strprintf("%d < %d", package_fee, mempoolRejectFee));
@@ -663,6 +667,8 @@ class MemPoolAccept
663667

664668
bool MemPoolAccept::PreChecks(ATMPArgs& args, Workspace& ws)
665669
{
670+
AssertLockHeld(cs_main);
671+
AssertLockHeld(m_pool.cs);
666672
const CTransactionRef& ptx = ws.m_ptx;
667673
const CTransaction& tx = *ws.m_ptx;
668674
const uint256& hash = ws.m_hash;
@@ -963,6 +969,8 @@ bool MemPoolAccept::PackageMempoolChecks(const std::vector<CTransactionRef>& txn
963969

964970
bool MemPoolAccept::PolicyScriptChecks(const ATMPArgs& args, Workspace& ws)
965971
{
972+
AssertLockHeld(cs_main);
973+
AssertLockHeld(m_pool.cs);
966974
const CTransaction& tx = *ws.m_ptx;
967975
TxValidationState& state = ws.m_state;
968976

@@ -989,6 +997,8 @@ bool MemPoolAccept::PolicyScriptChecks(const ATMPArgs& args, Workspace& ws)
989997

990998
bool MemPoolAccept::ConsensusScriptChecks(const ATMPArgs& args, Workspace& ws)
991999
{
1000+
AssertLockHeld(cs_main);
1001+
AssertLockHeld(m_pool.cs);
9921002
const CTransaction& tx = *ws.m_ptx;
9931003
const uint256& hash = ws.m_hash;
9941004
TxValidationState& state = ws.m_state;
@@ -1021,6 +1031,8 @@ bool MemPoolAccept::ConsensusScriptChecks(const ATMPArgs& args, Workspace& ws)
10211031

10221032
bool MemPoolAccept::Finalize(const ATMPArgs& args, Workspace& ws)
10231033
{
1034+
AssertLockHeld(cs_main);
1035+
AssertLockHeld(m_pool.cs);
10241036
const CTransaction& tx = *ws.m_ptx;
10251037
const uint256& hash = ws.m_hash;
10261038
TxValidationState& state = ws.m_state;
@@ -1342,8 +1354,9 @@ PackageMempoolAcceptResult MemPoolAccept::AcceptPackage(const Package& package,
13421354

13431355
MempoolAcceptResult AcceptToMemoryPool(CChainState& active_chainstate, const CTransactionRef& tx,
13441356
int64_t accept_time, bool bypass_limits, bool test_accept)
1345-
EXCLUSIVE_LOCKS_REQUIRED(cs_main)
1357+
EXCLUSIVE_LOCKS_REQUIRED(::cs_main)
13461358
{
1359+
AssertLockHeld(::cs_main);
13471360
const CChainParams& chainparams{active_chainstate.m_params};
13481361
assert(active_chainstate.GetMempool() != nullptr);
13491362
CTxMemPool& pool{*active_chainstate.GetMempool()};
@@ -1421,6 +1434,7 @@ CoinsViews::CoinsViews(
14211434

14221435
void CoinsViews::InitCache()
14231436
{
1437+
AssertLockHeld(::cs_main);
14241438
m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
14251439
}
14261440

@@ -1451,6 +1465,7 @@ void CChainState::InitCoinsDB(
14511465

14521466
void CChainState::InitCoinsCache(size_t cache_size_bytes)
14531467
{
1468+
AssertLockHeld(::cs_main);
14541469
assert(m_coins_views != nullptr);
14551470
m_coinstip_cache_size_bytes = cache_size_bytes;
14561471
m_coins_views->InitCache();
@@ -1524,6 +1539,7 @@ void CChainState::CheckForkWarningConditions()
15241539
// Called both upon regular invalid block discovery *and* InvalidateBlock
15251540
void CChainState::InvalidChainFound(CBlockIndex* pindexNew)
15261541
{
1542+
AssertLockHeld(cs_main);
15271543
if (!m_chainman.m_best_invalid || pindexNew->nChainWork > m_chainman.m_best_invalid->nChainWork) {
15281544
m_chainman.m_best_invalid = pindexNew;
15291545
}
@@ -1546,6 +1562,7 @@ void CChainState::InvalidChainFound(CBlockIndex* pindexNew)
15461562
// which does its own setBlockIndexCandidates management.
15471563
void CChainState::InvalidBlockFound(CBlockIndex* pindex, const BlockValidationState& state)
15481564
{
1565+
AssertLockHeld(cs_main);
15491566
if (state.GetResult() != BlockValidationResult::BLOCK_MUTATED) {
15501567
pindex->nStatus |= BLOCK_FAILED_VALID;
15511568
m_chainman.m_failed_blocks.insert(pindex);
@@ -2210,6 +2227,7 @@ bool CChainState::ConnectBlock(const CBlock& block, BlockValidationState& state,
22102227

22112228
CoinsCacheSizeState CChainState::GetCoinsCacheSizeState()
22122229
{
2230+
AssertLockHeld(::cs_main);
22132231
return this->GetCoinsCacheSizeState(
22142232
m_coinstip_cache_size_bytes,
22152233
gArgs.GetIntArg("-maxmempool", DEFAULT_MAX_MEMPOOL_SIZE) * 1000000);
@@ -2219,6 +2237,7 @@ CoinsCacheSizeState CChainState::GetCoinsCacheSizeState(
22192237
size_t max_coins_cache_size_bytes,
22202238
size_t max_mempool_size_bytes)
22212239
{
2240+
AssertLockHeld(::cs_main);
22222241
const int64_t nMempoolUsage = m_mempool ? m_mempool->DynamicMemoryUsage() : 0;
22232242
int64_t cacheSize = CoinsTip().DynamicMemoryUsage();
22242243
int64_t nTotalSpace =
@@ -2427,6 +2446,7 @@ static void UpdateTipLog(
24272446

24282447
void CChainState::UpdateTip(const CBlockIndex* pindexNew)
24292448
{
2449+
AssertLockHeld(::cs_main);
24302450
const auto& coins_tip = this->CoinsTip();
24312451

24322452
// The remainder of the function isn't relevant if we are not acting on
@@ -2650,7 +2670,9 @@ bool CChainState::ConnectTip(BlockValidationState& state, CBlockIndex* pindexNew
26502670
* Return the tip of the chain with the most work in it, that isn't
26512671
* known to be invalid (it's however far from certain to be valid).
26522672
*/
2653-
CBlockIndex* CChainState::FindMostWorkChain() {
2673+
CBlockIndex* CChainState::FindMostWorkChain()
2674+
{
2675+
AssertLockHeld(::cs_main);
26542676
do {
26552677
CBlockIndex *pindexNew = nullptr;
26562678

@@ -2854,7 +2876,7 @@ bool CChainState::ActivateBestChain(BlockValidationState& state, std::shared_ptr
28542876
// far from a guarantee. Things in the P2P/RPC will often end up calling
28552877
// us in the middle of ProcessNewBlock - do not assume pblock is set
28562878
// sanely for performance or correctness!
2857-
AssertLockNotHeld(cs_main);
2879+
AssertLockNotHeld(::cs_main);
28582880

28592881
// ABC maintains a fair degree of expensive-to-calculate internal state
28602882
// because this function periodically releases cs_main so that it does not lock up other threads for too long
@@ -2950,6 +2972,8 @@ bool CChainState::ActivateBestChain(BlockValidationState& state, std::shared_ptr
29502972

29512973
bool CChainState::PreciousBlock(BlockValidationState& state, CBlockIndex* pindex)
29522974
{
2975+
AssertLockNotHeld(m_chainstate_mutex);
2976+
AssertLockNotHeld(::cs_main);
29532977
{
29542978
LOCK(cs_main);
29552979
if (pindex->nChainWork < m_chain.Tip()->nChainWork) {
@@ -2980,6 +3004,7 @@ bool CChainState::PreciousBlock(BlockValidationState& state, CBlockIndex* pindex
29803004
bool CChainState::InvalidateBlock(BlockValidationState& state, CBlockIndex* pindex)
29813005
{
29823006
AssertLockNotHeld(m_chainstate_mutex);
3007+
AssertLockNotHeld(::cs_main);
29833008

29843009
// Genesis block can't be invalidated
29853010
assert(pindex);
@@ -3158,6 +3183,7 @@ void CChainState::ResetBlockFailureFlags(CBlockIndex *pindex) {
31583183
/** Mark a block as having its data received and checked (up to BLOCK_VALID_TRANSACTIONS). */
31593184
void CChainState::ReceivedBlockTransactions(const CBlock& block, CBlockIndex* pindexNew, const FlatFilePos& pos)
31603185
{
3186+
AssertLockHeld(cs_main);
31613187
pindexNew->nTx = block.vtx.size();
31623188
pindexNew->nChainTx = 0;
31633189
pindexNew->nFile = pos.nFile;
@@ -3330,8 +3356,9 @@ std::vector<unsigned char> GenerateCoinbaseCommitment(CBlock& block, const CBloc
33303356
* in ConnectBlock().
33313357
* Note that -reindex-chainstate skips the validation that happens here!
33323358
*/
3333-
static bool ContextualCheckBlockHeader(const CBlockHeader& block, BlockValidationState& state, BlockManager& blockman, const CChainParams& params, const CBlockIndex* pindexPrev, int64_t nAdjustedTime) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
3359+
static bool ContextualCheckBlockHeader(const CBlockHeader& block, BlockValidationState& state, BlockManager& blockman, const CChainParams& params, const CBlockIndex* pindexPrev, int64_t nAdjustedTime) EXCLUSIVE_LOCKS_REQUIRED(::cs_main)
33343360
{
3361+
AssertLockHeld(::cs_main);
33353362
assert(pindexPrev != nullptr);
33363363
const int nHeight = pindexPrev->nHeight + 1;
33373364

@@ -3702,6 +3729,7 @@ bool ChainstateManager::ProcessNewBlock(const CChainParams& chainparams, const s
37023729

37033730
MempoolAcceptResult ChainstateManager::ProcessTransaction(const CTransactionRef& tx, bool test_accept)
37043731
{
3732+
AssertLockHeld(cs_main);
37053733
CChainState& active_chainstate = ActiveChainstate();
37063734
if (!active_chainstate.GetMempool()) {
37073735
TxValidationState state;
@@ -3915,6 +3943,7 @@ bool CVerifyDB::VerifyDB(
39153943
/** Apply the effects of a block on the utxo cache, ignoring that it may already have been applied. */
39163944
bool CChainState::RollforwardBlock(const CBlockIndex* pindex, CCoinsViewCache& inputs)
39173945
{
3946+
AssertLockHeld(cs_main);
39183947
// TODO: merge with ConnectBlock
39193948
CBlock block;
39203949
if (!ReadBlockFromDisk(block, pindex, m_params.GetConsensus())) {
@@ -4018,7 +4047,9 @@ bool CChainState::NeedsRedownload() const
40184047
return false;
40194048
}
40204049

4021-
void CChainState::UnloadBlockIndex() {
4050+
void CChainState::UnloadBlockIndex()
4051+
{
4052+
AssertLockHeld(::cs_main);
40224053
nBlockSequenceId = 1;
40234054
setBlockIndexCandidates.clear();
40244055
}
@@ -4090,6 +4121,7 @@ bool CChainState::LoadGenesisBlock()
40904121

40914122
void CChainState::LoadExternalBlockFile(FILE* fileIn, FlatFilePos* dbp)
40924123
{
4124+
AssertLockNotHeld(m_chainstate_mutex);
40934125
// Map of disk positions for blocks with unknown parent (only used for reindex)
40944126
static std::multimap<uint256, FlatFilePos> mapBlocksUnknownParent;
40954127
int64_t nStart = GetTimeMillis();
@@ -4430,6 +4462,7 @@ void CChainState::CheckBlockIndex()
44304462

44314463
std::string CChainState::ToString()
44324464
{
4465+
AssertLockHeld(::cs_main);
44334466
CBlockIndex* tip = m_chain.Tip();
44344467
return strprintf("Chainstate [%s] @ height %d (%s)",
44354468
m_from_snapshot_blockhash ? "snapshot" : "ibd",
@@ -4438,6 +4471,7 @@ std::string CChainState::ToString()
44384471

44394472
bool CChainState::ResizeCoinsCaches(size_t coinstip_size, size_t coinsdb_size)
44404473
{
4474+
AssertLockHeld(::cs_main);
44414475
if (coinstip_size == m_coinstip_cache_size_bytes &&
44424476
coinsdb_size == m_coinsdb_cache_size_bytes) {
44434477
// Cache sizes are unchanged, no need to continue.
@@ -4662,6 +4696,7 @@ std::vector<CChainState*> ChainstateManager::GetAll()
46624696
CChainState& ChainstateManager::InitializeChainstate(
46634697
CTxMemPool* mempool, const std::optional<uint256>& snapshot_blockhash)
46644698
{
4699+
AssertLockHeld(::cs_main);
46654700
bool is_snapshot = snapshot_blockhash.has_value();
46664701
std::unique_ptr<CChainState>& to_modify =
46674702
is_snapshot ? m_snapshot_chainstate : m_ibd_chainstate;
@@ -4999,6 +5034,7 @@ bool ChainstateManager::IsSnapshotActive() const
49995034

50005035
void ChainstateManager::Unload()
50015036
{
5037+
AssertLockHeld(::cs_main);
50025038
for (CChainState* chainstate : this->GetAll()) {
50035039
chainstate->m_chain.SetTip(nullptr);
50045040
chainstate->UnloadBlockIndex();
@@ -5020,6 +5056,7 @@ void ChainstateManager::Reset()
50205056

50215057
void ChainstateManager::MaybeRebalanceCaches()
50225058
{
5059+
AssertLockHeld(::cs_main);
50235060
if (m_ibd_chainstate && !m_snapshot_chainstate) {
50245061
LogPrintf("[snapshot] allocating all cache to the IBD chainstate\n");
50255062
// Allocate everything to the IBD chainstate.

src/validation.h

+10-4
Original file line numberDiff line numberDiff line change
@@ -529,7 +529,9 @@ class CChainState
529529

530530
//! @returns whether or not the CoinsViews object has been fully initialized and we can
531531
//! safely flush this object to disk.
532-
bool CanFlushToDisk() const EXCLUSIVE_LOCKS_REQUIRED(cs_main) {
532+
bool CanFlushToDisk() const EXCLUSIVE_LOCKS_REQUIRED(::cs_main)
533+
{
534+
AssertLockHeld(::cs_main);
533535
return m_coins_views && m_coins_views->m_cacheview;
534536
}
535537

@@ -557,15 +559,17 @@ class CChainState
557559
std::set<CBlockIndex*, node::CBlockIndexWorkComparator> setBlockIndexCandidates;
558560

559561
//! @returns A reference to the in-memory cache of the UTXO set.
560-
CCoinsViewCache& CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
562+
CCoinsViewCache& CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(::cs_main)
561563
{
564+
AssertLockHeld(::cs_main);
562565
assert(m_coins_views->m_cacheview);
563566
return *m_coins_views->m_cacheview.get();
564567
}
565568

566569
//! @returns A reference to the on-disk UTXO set database.
567-
CCoinsViewDB& CoinsDB() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
570+
CCoinsViewDB& CoinsDB() EXCLUSIVE_LOCKS_REQUIRED(::cs_main)
568571
{
572+
AssertLockHeld(::cs_main);
569573
return m_coins_views->m_dbview;
570574
}
571575

@@ -577,8 +581,9 @@ class CChainState
577581

578582
//! @returns A reference to a wrapped view of the in-memory UTXO set that
579583
//! handles disk read errors gracefully.
580-
CCoinsViewErrorCatcher& CoinsErrorCatcher() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
584+
CCoinsViewErrorCatcher& CoinsErrorCatcher() EXCLUSIVE_LOCKS_REQUIRED(::cs_main)
581585
{
586+
AssertLockHeld(::cs_main);
582587
return m_coins_views->m_catcherview;
583588
}
584589

@@ -924,6 +929,7 @@ class ChainstateManager
924929

925930
node::BlockMap& BlockIndex() EXCLUSIVE_LOCKS_REQUIRED(::cs_main)
926931
{
932+
AssertLockHeld(::cs_main);
927933
return m_blockman.m_block_index;
928934
}
929935

0 commit comments

Comments
 (0)