/ src / test / validation_flush_tests.cpp
validation_flush_tests.cpp
 1  // Copyright (c) 2019-present The Bitcoin Core developers
 2  // Distributed under the MIT software license, see the accompanying
 3  // file COPYING or http://www.opensource.org/licenses/mit-license.php.
 4  
 5  #include <sync.h>
 6  #include <test/util/coins.h>
 7  #include <test/util/random.h>
 8  #include <test/util/common.h>
 9  #include <test/util/setup_common.h>
10  #include <validation.h>
11  
12  #include <boost/test/unit_test.hpp>
13  
14  BOOST_FIXTURE_TEST_SUITE(validation_flush_tests, TestingSetup)
15  
16  //! Verify that Chainstate::GetCoinsCacheSizeState() switches from OK→LARGE→CRITICAL
17  //! at the expected utilization thresholds, first with *no* mempool head-room,
18  //! then with additional mempool head-room.
19  BOOST_AUTO_TEST_CASE(getcoinscachesizestate)
20  {
21      Chainstate& chainstate{m_node.chainman->ActiveChainstate()};
22  
23      LOCK(::cs_main);
24      CCoinsViewCache& view{chainstate.CoinsTip()};
25  
26      // Sanity: an empty cache should be ≲ 1 chunk (~ 256 KiB).
27      BOOST_CHECK_LT(view.DynamicMemoryUsage() / (256 * 1024.0), 1.1);
28  
29      constexpr size_t MAX_COINS_BYTES{8_MiB};
30      constexpr size_t MAX_MEMPOOL_BYTES{4_MiB};
31      constexpr size_t MAX_ATTEMPTS{50'000};
32  
33      // Run the same growth-path twice: first with 0 head-room, then with extra head-room
34      for (size_t max_mempool_size_bytes : {size_t{0}, MAX_MEMPOOL_BYTES}) {
35          const int64_t full_cap{int64_t(MAX_COINS_BYTES + max_mempool_size_bytes)};
36          const int64_t large_cap{LargeCoinsCacheThreshold(full_cap)};
37  
38          // OK → LARGE
39          auto state{chainstate.GetCoinsCacheSizeState(MAX_COINS_BYTES, max_mempool_size_bytes)};
40          for (size_t i{0}; i < MAX_ATTEMPTS && int64_t(view.DynamicMemoryUsage()) <= large_cap; ++i) {
41              BOOST_CHECK_EQUAL(state, CoinsCacheSizeState::OK);
42              AddTestCoin(m_rng, view);
43              state = chainstate.GetCoinsCacheSizeState(MAX_COINS_BYTES, max_mempool_size_bytes);
44          }
45  
46          // LARGE → CRITICAL
47          for (size_t i{0}; i < MAX_ATTEMPTS && int64_t(view.DynamicMemoryUsage()) <= full_cap; ++i) {
48              BOOST_CHECK_EQUAL(state, CoinsCacheSizeState::LARGE);
49              AddTestCoin(m_rng, view);
50              state = chainstate.GetCoinsCacheSizeState(MAX_COINS_BYTES, max_mempool_size_bytes);
51          }
52          BOOST_CHECK_EQUAL(state, CoinsCacheSizeState::CRITICAL);
53      }
54  
55      // Default thresholds (no explicit limits) permit many more coins.
56      for (int i{0}; i < 1'000; ++i) {
57          AddTestCoin(m_rng, view);
58          BOOST_CHECK_EQUAL(chainstate.GetCoinsCacheSizeState(), CoinsCacheSizeState::OK);
59      }
60  
61      // CRITICAL → OK via Flush
62      BOOST_CHECK_EQUAL(chainstate.GetCoinsCacheSizeState(MAX_COINS_BYTES, /*max_mempool_size_bytes=*/0), CoinsCacheSizeState::CRITICAL);
63      view.SetBestBlock(m_rng.rand256());
64      view.Flush();
65      BOOST_CHECK_EQUAL(chainstate.GetCoinsCacheSizeState(MAX_COINS_BYTES, /*max_mempool_size_bytes=*/0), CoinsCacheSizeState::OK);
66  }
67  
68  BOOST_AUTO_TEST_SUITE_END()