utxo_snapshot.h
1 // Copyright (c) 2009-2010 Satoshi Nakamoto 2 // Copyright (c) 2009-present 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 #ifndef BITCOIN_NODE_UTXO_SNAPSHOT_H 7 #define BITCOIN_NODE_UTXO_SNAPSHOT_H 8 9 #include <kernel/chainparams.h> 10 #include <kernel/cs_main.h> 11 #include <kernel/messagestartchars.h> 12 #include <sync.h> 13 #include <tinyformat.h> 14 #include <uint256.h> 15 #include <util/chaintype.h> 16 #include <util/fs.h> 17 18 #include <algorithm> 19 #include <array> 20 #include <cstdint> 21 #include <ios> 22 #include <optional> 23 #include <set> 24 #include <string> 25 #include <string_view> 26 27 // UTXO set snapshot magic bytes 28 static constexpr std::array<uint8_t, 5> SNAPSHOT_MAGIC_BYTES = {'u', 't', 'x', 'o', 0xff}; 29 30 class Chainstate; 31 32 namespace node { 33 //! Metadata describing a serialized version of a UTXO set from which an 34 //! assumeutxo Chainstate can be constructed. 35 //! All metadata fields come from an untrusted file, so must be validated 36 //! before being used. Thus, new fields should be added only if needed. 37 class SnapshotMetadata 38 { 39 inline static const uint16_t VERSION{2}; 40 const std::set<uint16_t> m_supported_versions{VERSION}; 41 const MessageStartChars m_network_magic; 42 public: 43 //! The hash of the block that reflects the tip of the chain for the 44 //! UTXO set contained in this snapshot. 45 uint256 m_base_blockhash; 46 47 48 //! The number of coins in the UTXO set contained in this snapshot. Used 49 //! during snapshot load to estimate progress of UTXO set reconstruction. 50 uint64_t m_coins_count = 0; 51 52 SnapshotMetadata( 53 const MessageStartChars network_magic) : 54 m_network_magic(network_magic) { } 55 SnapshotMetadata( 56 const MessageStartChars network_magic, 57 const uint256& base_blockhash, 58 uint64_t coins_count) : 59 m_network_magic(network_magic), 60 m_base_blockhash(base_blockhash), 61 m_coins_count(coins_count) { } 62 63 template <typename Stream> 64 inline void Serialize(Stream& s) const { 65 s << SNAPSHOT_MAGIC_BYTES; 66 s << VERSION; 67 s << m_network_magic; 68 s << m_base_blockhash; 69 s << m_coins_count; 70 } 71 72 template <typename Stream> 73 inline void Unserialize(Stream& s) { 74 // Read the snapshot magic bytes 75 std::array<uint8_t, SNAPSHOT_MAGIC_BYTES.size()> snapshot_magic; 76 s >> snapshot_magic; 77 if (snapshot_magic != SNAPSHOT_MAGIC_BYTES) { 78 throw std::ios_base::failure("Invalid UTXO set snapshot magic bytes. Please check if this is indeed a snapshot file or if you are using an outdated snapshot format."); 79 } 80 81 // Read the version 82 uint16_t version; 83 s >> version; 84 if (!m_supported_versions.contains(version)) { 85 throw std::ios_base::failure(strprintf("Version of snapshot %s does not match any of the supported versions.", version)); 86 } 87 88 // Read the network magic (pchMessageStart) 89 MessageStartChars message; 90 s >> message; 91 if (!std::equal(message.begin(), message.end(), m_network_magic.data())) { 92 auto metadata_network{GetNetworkForMagic(message)}; 93 if (metadata_network) { 94 std::string network_string{ChainTypeToString(metadata_network.value())}; 95 auto node_network{GetNetworkForMagic(m_network_magic)}; 96 std::string node_network_string{ChainTypeToString(node_network.value())}; 97 throw std::ios_base::failure(strprintf("The network of the snapshot (%s) does not match the network of this node (%s).", network_string, node_network_string)); 98 } else { 99 throw std::ios_base::failure("This snapshot has been created for an unrecognized network. This could be a custom signet, a new testnet or possibly caused by data corruption."); 100 } 101 } 102 103 s >> m_base_blockhash; 104 s >> m_coins_count; 105 } 106 }; 107 108 //! The file in the snapshot chainstate dir which stores the base blockhash. This is 109 //! needed to reconstruct snapshot chainstates on init. 110 //! 111 //! Because we only allow loading a single snapshot at a time, there will only be one 112 //! chainstate directory with this filename present within it. 113 const fs::path SNAPSHOT_BLOCKHASH_FILENAME{"base_blockhash"}; 114 115 //! Write out the blockhash of the snapshot base block that was used to construct 116 //! this chainstate. This value is read in during subsequent initializations and 117 //! used to reconstruct snapshot-based chainstates. 118 bool WriteSnapshotBaseBlockhash(Chainstate& snapshot_chainstate) 119 EXCLUSIVE_LOCKS_REQUIRED(::cs_main); 120 121 //! Read the blockhash of the snapshot base block that was used to construct the 122 //! chainstate. 123 std::optional<uint256> ReadSnapshotBaseBlockhash(fs::path chaindir) 124 EXCLUSIVE_LOCKS_REQUIRED(::cs_main); 125 126 //! Suffix appended to the chainstate (leveldb) dir when created based upon 127 //! a snapshot. 128 constexpr std::string_view SNAPSHOT_CHAINSTATE_SUFFIX = "_snapshot"; 129 130 131 //! Return a path to the snapshot-based chainstate dir, if one exists. 132 std::optional<fs::path> FindAssumeutxoChainstateDir(const fs::path& data_dir); 133 134 } // namespace node 135 136 #endif // BITCOIN_NODE_UTXO_SNAPSHOT_H