LCOV - code coverage report
Current view: top level - src - txdb.cpp (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 87.7 % 106 93
Test Date: 2025-01-19 05:08:01 Functions: 94.4 % 18 17
Branches: 44.8 % 116 52

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2009-2010 Satoshi Nakamoto
       2                 :             : // Copyright (c) 2009-2022 The Bitcoin Core developers
       3                 :             : // Distributed under the MIT software license, see the accompanying
       4                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       5                 :             : 
       6                 :             : #include <txdb.h>
       7                 :             : 
       8                 :             : #include <coins.h>
       9                 :             : #include <dbwrapper.h>
      10                 :             : #include <logging.h>
      11                 :             : #include <primitives/transaction.h>
      12                 :             : #include <random.h>
      13                 :             : #include <serialize.h>
      14                 :             : #include <uint256.h>
      15                 :             : #include <util/vector.h>
      16                 :             : 
      17                 :             : #include <cassert>
      18                 :             : #include <cstdlib>
      19                 :             : #include <iterator>
      20                 :             : #include <utility>
      21                 :             : 
      22                 :             : static constexpr uint8_t DB_COIN{'C'};
      23                 :             : static constexpr uint8_t DB_BEST_BLOCK{'B'};
      24                 :             : static constexpr uint8_t DB_HEAD_BLOCKS{'H'};
      25                 :             : // Keys used in previous version that might still be found in the DB:
      26                 :             : static constexpr uint8_t DB_COINS{'c'};
      27                 :             : 
      28                 :        1107 : bool CCoinsViewDB::NeedsUpgrade()
      29                 :             : {
      30         [ +  - ]:        1107 :     std::unique_ptr<CDBIterator> cursor{m_db->NewIterator()};
      31                 :             :     // DB_COINS was deprecated in v0.15.0, commit
      32                 :             :     // 1088b02f0ccd7358d2b7076bb9e122d59d502d02
      33         [ +  - ]:        1107 :     cursor->Seek(std::make_pair(DB_COINS, uint256{}));
      34         [ +  - ]:        2214 :     return cursor->Valid();
      35                 :        1107 : }
      36                 :             : 
      37                 :             : namespace {
      38                 :             : 
      39                 :             : struct CoinEntry {
      40                 :             :     COutPoint* outpoint;
      41                 :             :     uint8_t key;
      42         [ +  - ]:     5458615 :     explicit CoinEntry(const COutPoint* ptr) : outpoint(const_cast<COutPoint*>(ptr)), key(DB_COIN)  {}
      43                 :             : 
      44                 :    11805920 :     SERIALIZE_METHODS(CoinEntry, obj) { READWRITE(obj.key, obj.outpoint->hash, VARINT(obj.outpoint->n)); }
      45                 :             : };
      46                 :             : 
      47                 :             : } // namespace
      48                 :             : 
      49                 :        1152 : CCoinsViewDB::CCoinsViewDB(DBParams db_params, CoinsViewOptions options) :
      50                 :        1152 :     m_db_params{std::move(db_params)},
      51                 :        1152 :     m_options{std::move(options)},
      52         [ +  + ]:        1154 :     m_db{std::make_unique<CDBWrapper>(m_db_params)} { }
      53                 :             : 
      54                 :         122 : void CCoinsViewDB::ResizeCache(size_t new_cache_size)
      55                 :             : {
      56                 :             :     // We can't do this operation with an in-memory DB since we'll lose all the coins upon
      57                 :             :     // reset.
      58         [ +  + ]:         122 :     if (!m_db_params.memory_only) {
      59                 :             :         // Have to do a reset first to get the original `m_db` state to release its
      60                 :             :         // filesystem lock.
      61         [ +  - ]:         114 :         m_db.reset();
      62                 :         114 :         m_db_params.cache_bytes = new_cache_size;
      63                 :         114 :         m_db_params.wipe_data = false;
      64                 :         114 :         m_db = std::make_unique<CDBWrapper>(m_db_params);
      65                 :             :     }
      66                 :         122 : }
      67                 :             : 
      68                 :     5458571 : std::optional<Coin> CCoinsViewDB::GetCoin(const COutPoint& outpoint) const
      69                 :             : {
      70   [ +  -  +  + ]:     5458571 :     if (Coin coin; m_db->Read(CoinEntry(&outpoint), coin)) return coin;
      71                 :     5354268 :     return std::nullopt;
      72                 :             : }
      73                 :             : 
      74                 :          44 : bool CCoinsViewDB::HaveCoin(const COutPoint &outpoint) const {
      75                 :          44 :     return m_db->Exists(CoinEntry(&outpoint));
      76                 :             : }
      77                 :             : 
      78                 :        7121 : uint256 CCoinsViewDB::GetBestBlock() const {
      79                 :        7121 :     uint256 hashBestChain;
      80         [ +  + ]:        7121 :     if (!m_db->Read(DB_BEST_BLOCK, hashBestChain))
      81                 :        1685 :         return uint256();
      82                 :        5436 :     return hashBestChain;
      83                 :             : }
      84                 :             : 
      85                 :        1433 : std::vector<uint256> CCoinsViewDB::GetHeadBlocks() const {
      86                 :        1433 :     std::vector<uint256> vhashHeadBlocks;
      87   [ +  -  +  - ]:        1433 :     if (!m_db->Read(DB_HEAD_BLOCKS, vhashHeadBlocks)) {
      88                 :        1433 :         return std::vector<uint256>();
      89                 :             :     }
      90                 :           0 :     return vhashHeadBlocks;
      91                 :        1433 : }
      92                 :             : 
      93                 :        3272 : bool CCoinsViewDB::BatchWrite(CoinsViewCacheCursor& cursor, const uint256 &hashBlock) {
      94                 :        3272 :     CDBBatch batch(*m_db);
      95                 :        3272 :     size_t count = 0;
      96                 :        3272 :     size_t changed = 0;
      97         [ -  + ]:        3272 :     assert(!hashBlock.IsNull());
      98                 :             : 
      99         [ +  - ]:        3272 :     uint256 old_tip = GetBestBlock();
     100         [ +  + ]:        3272 :     if (old_tip.IsNull()) {
     101                 :             :         // We may be in the middle of replaying.
     102         [ +  - ]:         327 :         std::vector<uint256> old_heads = GetHeadBlocks();
     103         [ -  + ]:         327 :         if (old_heads.size() == 2) {
     104         [ #  # ]:           0 :             if (old_heads[0] != hashBlock) {
     105   [ #  #  #  #  :           0 :                 LogPrintLevel(BCLog::COINDB, BCLog::Level::Error, "The coins database detected an inconsistent state, likely due to a previous crash or shutdown. You will need to restart bitcoind with the -reindex-chainstate or -reindex configuration option.\n");
                   #  # ]
     106                 :             :             }
     107         [ #  # ]:           0 :             assert(old_heads[0] == hashBlock);
     108                 :           0 :             old_tip = old_heads[1];
     109                 :             :         }
     110                 :         327 :     }
     111                 :             : 
     112                 :             :     // In the first batch, mark the database as being in the middle of a
     113                 :             :     // transition from old_tip to hashBlock.
     114                 :             :     // A vector is used for future extensibility, as we may want to support
     115                 :             :     // interrupting after partial writes from multiple independent reorgs.
     116         [ +  - ]:        3272 :     batch.Erase(DB_BEST_BLOCK);
     117   [ +  -  +  - ]:        3272 :     batch.Write(DB_HEAD_BLOCKS, Vector(hashBlock, old_tip));
     118                 :             : 
     119         [ +  + ]:      263398 :     for (auto it{cursor.Begin()}; it != cursor.End();) {
     120         [ +  - ]:      260126 :         if (it->second.IsDirty()) {
     121                 :      260126 :             CoinEntry entry(&it->first);
     122         [ +  + ]:      260126 :             if (it->second.coin.IsSpent())
     123         [ +  - ]:       30139 :                 batch.Erase(entry);
     124                 :             :             else
     125         [ +  - ]:      229987 :                 batch.Write(entry, it->second.coin);
     126                 :      260126 :             changed++;
     127                 :             :         }
     128                 :      260126 :         count++;
     129                 :      260126 :         it = cursor.NextAndMaybeErase(*it);
     130         [ -  + ]:      260126 :         if (batch.SizeEstimate() > m_options.batch_write_bytes) {
     131   [ #  #  #  #  :           0 :             LogDebug(BCLog::COINDB, "Writing partial batch of %.2f MiB\n", batch.SizeEstimate() * (1.0 / 1048576.0));
                   #  # ]
     132         [ #  # ]:           0 :             m_db->WriteBatch(batch);
     133         [ #  # ]:           0 :             batch.Clear();
     134         [ #  # ]:           0 :             if (m_options.simulate_crash_ratio) {
     135   [ #  #  #  # ]:           0 :                 static FastRandomContext rng;
     136         [ #  # ]:           0 :                 if (rng.randrange(m_options.simulate_crash_ratio) == 0) {
     137         [ #  # ]:           0 :                     LogPrintf("Simulating a crash. Goodbye.\n");
     138                 :           0 :                     _Exit(0);
     139                 :             :                 }
     140                 :             :             }
     141                 :             :         }
     142                 :             :     }
     143                 :             : 
     144                 :             :     // In the last batch, mark the database as consistent with hashBlock again.
     145         [ +  - ]:        3272 :     batch.Erase(DB_HEAD_BLOCKS);
     146         [ +  - ]:        3272 :     batch.Write(DB_BEST_BLOCK, hashBlock);
     147                 :             : 
     148   [ +  -  +  -  :        3272 :     LogDebug(BCLog::COINDB, "Writing final batch of %.2f MiB\n", batch.SizeEstimate() * (1.0 / 1048576.0));
                   +  - ]
     149         [ +  - ]:        3272 :     bool ret = m_db->WriteBatch(batch);
     150   [ +  -  +  -  :        3272 :     LogDebug(BCLog::COINDB, "Committed %u changed transaction outputs (out of %u) to coin database...\n", (unsigned int)changed, (unsigned int)count);
                   +  - ]
     151                 :        3272 :     return ret;
     152                 :        3272 : }
     153                 :             : 
     154                 :          97 : size_t CCoinsViewDB::EstimateSize() const
     155                 :             : {
     156                 :          97 :     return m_db->EstimateSize(DB_COIN, uint8_t(DB_COIN + 1));
     157                 :             : }
     158                 :             : 
     159                 :             : /** Specialization of CCoinsViewCursor to iterate over a CCoinsViewDB */
     160                 :             : class CCoinsViewDBCursor: public CCoinsViewCursor
     161                 :             : {
     162                 :             : public:
     163                 :             :     // Prefer using CCoinsViewDB::Cursor() since we want to perform some
     164                 :             :     // cache warmup on instantiation.
     165                 :        1133 :     CCoinsViewDBCursor(CDBIterator* pcursorIn, const uint256&hashBlockIn):
     166                 :        1133 :         CCoinsViewCursor(hashBlockIn), pcursor(pcursorIn) {}
     167                 :        1133 :     ~CCoinsViewDBCursor() = default;
     168                 :             : 
     169                 :             :     bool GetKey(COutPoint &key) const override;
     170                 :             :     bool GetValue(Coin &coin) const override;
     171                 :             : 
     172                 :             :     bool Valid() const override;
     173                 :             :     void Next() override;
     174                 :             : 
     175                 :             : private:
     176                 :             :     std::unique_ptr<CDBIterator> pcursor;
     177                 :             :     std::pair<char, COutPoint> keyTmp;
     178                 :             : 
     179                 :             :     friend class CCoinsViewDB;
     180                 :             : };
     181                 :             : 
     182                 :        1133 : std::unique_ptr<CCoinsViewCursor> CCoinsViewDB::Cursor() const
     183                 :             : {
     184                 :        1133 :     auto i = std::make_unique<CCoinsViewDBCursor>(
     185                 :        1133 :         const_cast<CDBWrapper&>(*m_db).NewIterator(), GetBestBlock());
     186                 :             :     /* It seems that there are no "const iterators" for LevelDB.  Since we
     187                 :             :        only need read operations on it, use a const-cast to get around
     188                 :             :        that restriction.  */
     189         [ +  - ]:        1133 :     i->pcursor->Seek(DB_COIN);
     190                 :             :     // Cache key of first record
     191   [ +  -  +  + ]:        1133 :     if (i->pcursor->Valid()) {
     192         [ +  - ]:        1108 :         CoinEntry entry(&i->keyTmp.second);
     193         [ +  - ]:        1108 :         i->pcursor->GetKey(entry);
     194                 :        1108 :         i->keyTmp.first = entry.key;
     195                 :             :     } else {
     196                 :          25 :         i->keyTmp.first = 0; // Make sure Valid() and GetKey() return false
     197                 :             :     }
     198                 :        1133 :     return i;
     199                 :        1133 : }
     200                 :             : 
     201                 :      184259 : bool CCoinsViewDBCursor::GetKey(COutPoint &key) const
     202                 :             : {
     203                 :             :     // Return cached key
     204         [ +  - ]:      184259 :     if (keyTmp.first == DB_COIN) {
     205                 :      184259 :         key = keyTmp.second;
     206                 :      184259 :         return true;
     207                 :             :     }
     208                 :             :     return false;
     209                 :             : }
     210                 :             : 
     211                 :      184219 : bool CCoinsViewDBCursor::GetValue(Coin &coin) const
     212                 :             : {
     213                 :      184219 :     return pcursor->GetValue(coin);
     214                 :             : }
     215                 :             : 
     216                 :      185352 : bool CCoinsViewDBCursor::Valid() const
     217                 :             : {
     218                 :      185352 :     return keyTmp.first == DB_COIN;
     219                 :             : }
     220                 :             : 
     221                 :      184219 : void CCoinsViewDBCursor::Next()
     222                 :             : {
     223                 :      184219 :     pcursor->Next();
     224                 :      184219 :     CoinEntry entry(&keyTmp.second);
     225   [ +  +  -  + ]:      184219 :     if (!pcursor->Valid() || !pcursor->GetKey(entry)) {
     226                 :        1108 :         keyTmp.first = 0; // Invalidate cached key after last record so that Valid() and GetKey() return false
     227                 :             :     } else {
     228                 :      183111 :         keyTmp.first = entry.key;
     229                 :             :     }
     230                 :      184219 : }
        

Generated by: LCOV version 2.0-1