24#include <boost/test/unit_test.hpp>
46 std::map<COutPoint, Coin> map_;
53 if (
auto it{map_.find(outpoint)}; it != map_.end() && !it->second.IsSpent())
return it->second;
62 if (it->second.IsDirty()) {
64 map_[it->first] = it->second.coin;
65 if (it->second.coin.IsSpent() && m_rng.
randrange(3) == 0) {
67 map_.erase(it->first);
72 hashBestBlock_ = block_hash;
86 for (
const auto& entry : cacheCoins) {
87 ret += entry.second.coin.DynamicMemoryUsage();
125 bool removed_all_caches =
false;
126 bool reached_4_caches =
false;
127 bool added_an_entry =
false;
128 bool added_an_unspendable_entry =
false;
129 bool removed_an_entry =
false;
130 bool updated_an_entry =
false;
131 bool found_an_entry =
false;
132 bool missed_an_entry =
false;
133 bool uncached_an_entry =
false;
134 bool flushed_without_erase =
false;
137 std::map<COutPoint, Coin> result;
140 std::vector<std::unique_ptr<CCoinsViewCacheTest>> stack;
141 stack.push_back(std::make_unique<CCoinsViewCacheTest>(base));
144 std::vector<Txid> txids;
146 for (
unsigned int i = 0; i < txids.size(); i++) {
162 bool result_havecoin = test_havecoin_before ? stack.back()->HaveCoin(
COutPoint(txid, 0)) :
false;
170 if (test_havecoin_before) {
174 if (test_havecoin_after) {
188 added_an_unspendable_entry =
true;
192 (coin.
IsSpent() ? added_an_entry : updated_an_entry) =
true;
196 stack.back()->EmplaceCoinInternalDANGER(std::move(op), std::move(newcoin));
202 removed_an_entry =
true;
212 stack[cacheid]->Uncache(
out);
213 uncached_an_entry |= !stack[cacheid]->HaveCoinInCache(
out);
218 for (
const auto& entry : result) {
219 bool have = stack.back()->HaveCoin(entry.first);
220 const Coin& coin = stack.back()->AccessCoin(entry.first);
224 missed_an_entry =
true;
226 BOOST_CHECK(stack.back()->HaveCoinInCache(entry.first));
227 found_an_entry =
true;
230 for (
const auto& test : stack) {
239 if (fake_best_block) stack[flushIndex]->SetBestBlock(
m_rng.
rand256());
241 should_erase ? stack[flushIndex]->Flush() : stack[flushIndex]->Sync();
242 flushed_without_erase |= !should_erase;
249 if (fake_best_block) stack.back()->SetBestBlock(
m_rng.
rand256());
251 should_erase ? stack.back()->Flush() : stack.back()->Sync();
252 flushed_without_erase |= !should_erase;
255 if (stack.size() == 0 || (stack.size() < 4 &&
m_rng.
randbool())) {
258 if (stack.size() > 0) {
259 tip = stack.back().get();
261 removed_all_caches =
true;
263 stack.push_back(std::make_unique<CCoinsViewCacheTest>(tip));
264 if (stack.size() == 4) {
265 reached_4_caches =
true;
290 CCoinsViewTest base{m_rng};
291 SimulationTest(&base,
false);
300 CCoinsViewDB db_base{{.path =
"test", .cache_bytes = 8_MiB, .memory_only =
true}, {}};
301 SimulationTest(&db_base,
true);
310typedef std::map<COutPoint, std::tuple<CTransaction,CTxUndo,Coin>>
UtxoData;
316 if (utxoSetIt == utxoSet.end()) {
317 utxoSetIt = utxoSet.begin();
319 auto utxoDataIt = utxoData.find(*utxoSetIt);
320 assert(utxoDataIt != utxoData.end());
335 bool spent_a_duplicate_coinbase =
false;
337 std::map<COutPoint, Coin> result;
340 CCoinsViewTest base{m_rng};
341 std::vector<std::unique_ptr<CCoinsViewCacheTest>> stack;
342 stack.push_back(std::make_unique<CCoinsViewCacheTest>(&base));
345 std::set<COutPoint> coinbase_coins;
346 std::set<COutPoint> disconnected_coins;
347 std::set<COutPoint> duplicate_coins;
348 std::set<COutPoint> utxoset;
351 uint32_t randiter = m_rng.
rand32();
354 if (randiter % 20 < 19) {
358 tx.
vout[0].nValue = i;
359 tx.
vout[0].scriptPubKey.assign(m_rng.
rand32() & 0x3F, 0);
360 const int height{int(m_rng.
rand32() >> 1)};
364 if (randiter % 20 < 2 || coinbase_coins.size() < 10) {
366 if (m_rng.
randrange(10) == 0 && coinbase_coins.size()) {
367 auto utxod = FindRandomFrom(coinbase_coins);
371 disconnected_coins.erase(utxod->first);
373 duplicate_coins.insert(utxod->first);
386 if (randiter % 20 == 2 && disconnected_coins.size()) {
387 auto utxod = FindRandomFrom(disconnected_coins);
389 prevout = tx.
vin[0].prevout;
390 if (!
CTransaction(tx).IsCoinBase() && !utxoset.contains(prevout)) {
391 disconnected_coins.erase(utxod->first);
396 if (utxoset.contains(utxod->first)) {
398 assert(duplicate_coins.contains(utxod->first));
400 disconnected_coins.erase(utxod->first);
405 auto utxod = FindRandomFrom(utxoset);
406 prevout = utxod->first;
409 tx.
vin[0].prevout = prevout;
413 old_coin = result[prevout];
415 result[prevout].
Clear();
417 utxoset.erase(prevout);
421 if (duplicate_coins.contains(prevout)) {
422 spent_a_duplicate_coinbase =
true;
436 utxoset.insert(outpoint);
439 utxoData.emplace(outpoint, std::make_tuple(tx,undo,old_coin));
440 }
else if (utxoset.size()) {
442 auto utxod = FindRandomFrom(utxoset);
445 CTxUndo &undo = std::get<1>(utxod->second);
446 Coin &orig_coin = std::get<2>(utxod->second);
450 result[utxod->first].Clear();
453 result[tx.
vin[0].prevout] = orig_coin;
459 BOOST_CHECK(stack.back()->SpendCoin(utxod->first));
467 disconnected_coins.insert(utxod->first);
470 utxoset.erase(utxod->first);
472 utxoset.insert(tx.
vin[0].prevout);
477 for (
const auto& entry : result) {
478 bool have = stack.back()->HaveCoin(entry.first);
479 const Coin& coin = stack.back()->AccessCoin(entry.first);
486 if (utxoset.size() > 1 && m_rng.
randrange(30) == 0) {
487 stack[m_rng.
rand32() % stack.size()]->Uncache(FindRandomFrom(utxoset)->first);
489 if (disconnected_coins.size() > 1 && m_rng.
randrange(30) == 0) {
490 stack[m_rng.
rand32() % stack.size()]->Uncache(FindRandomFrom(disconnected_coins)->first);
492 if (duplicate_coins.size() > 1 && m_rng.
randrange(30) == 0) {
493 stack[m_rng.
rand32() % stack.size()]->Uncache(FindRandomFrom(duplicate_coins)->first);
498 if (stack.size() > 1 && m_rng.
randbool() == 0) {
499 unsigned int flushIndex = m_rng.
randrange(stack.size() - 1);
500 stack[flushIndex]->Flush();
505 if (stack.size() > 0 && m_rng.
randbool() == 0) {
506 stack.back()->Flush();
509 if (stack.size() == 0 || (stack.size() < 4 && m_rng.
randbool())) {
511 if (stack.size() > 0) {
512 tip = stack.back().get();
514 stack.push_back(std::make_unique<CCoinsViewCacheTest>(tip));
527 SpanReader{
"97f23c835800816115944e077fe7c803cfa57f29b36bf87c1d35"_hex} >> cc1;
535 SpanReader{
"8ddf77bbd123008c988f1a4a4de2161e0f50aac7f17e7f9555caa4"_hex} >> cc2;
553 BOOST_CHECK_MESSAGE(
false,
"We should have thrown");
554 }
catch (
const std::ios_base::failure&) {
559 uint64_t x = 3000000000ULL;
565 BOOST_CHECK_MESSAGE(
false,
"We should have thrown");
566 }
catch (
const std::ios_base::failure&) {
578 enum class State { CLEAN, DIRTY, FRESH, DIRTY_FRESH };
586 friend std::ostream&
operator<<(std::ostream& os,
const CoinEntry& e) {
return os << e.value <<
", " << e.state; }
592 static constexpr State ToState(
const bool is_dirty,
const bool is_fresh) {
619constexpr auto EX_OVERWRITE_UNSPENT{
"Attempted to overwrite an unspent coin (when possible_overwrite is false)"};
627 if (value !=
SPENT) {
638 auto [iter, inserted] = map.emplace(
OUTPOINT, std::move(entry));
642 return iter->second.coin.DynamicMemoryUsage();
647 if (
auto it{map.find(outp)}; it != map.end()) {
649 it->second.coin.IsSpent() ?
SPENT : it->second.coin.out.nValue,
658 sentinel.second.SelfRef(sentinel);
660 CCoinsMap map{0, CCoinsMap::hasher{}, CCoinsMap::key_equal{}, &resource};
662 size_t dirty_count{cache_coin && cache_coin->IsDirty()};
677 cache.dirty() += cache_coin->IsDirty();
690 test.cache.SelfTest(
false);
720 test.cache.SelfTest();
749 bool possible_overwrite{coinbase};
751 if (
auto* expected_coin{std::get_if<MaybeCoin>(&expected)}) {
753 test.cache.SelfTest();
756 BOOST_CHECK_EXCEPTION(
add_coin(), std::logic_error,
HasReason(std::get<std::string>(expected)));
796 if (
auto* expected_coin{std::get_if<MaybeCoin>(&expected)}) {
798 test.cache.SelfTest(
false);
801 BOOST_CHECK_EXCEPTION(write_coins(), std::logic_error,
HasReason(std::get<std::string>(expected)));
900 CCoinsViewCacheTest* view,
902 std::vector<std::unique_ptr<CCoinsViewCacheTest>>& all_caches,
903 bool do_erasing_flush)
908 auto flush_all = [
this, &all_caches](
bool erase) {
910 for (
auto i = all_caches.rbegin(); i != all_caches.rend(); ++i) {
912 cache->SanityCheck();
916 erase ? cache->Flush() : cache->Sync();
929 view->AddCoin(outp,
Coin(coin),
false);
931 cache_usage = view->DynamicMemoryUsage();
932 cache_size = view->map().size();
956 if (do_erasing_flush) {
962 BOOST_TEST(view->DynamicMemoryUsage() <= cache_usage);
964 BOOST_TEST(view->map().size() < cache_size);
969 view->AccessCoin(outp);
976 view->AddCoin(outp,
Coin(coin),
false),
1007 all_caches[0]->AddCoin(outp, std::move(coin),
false);
1008 all_caches[0]->Sync();
1011 BOOST_CHECK(!all_caches[1]->HaveCoinInCache(outp));
1032 all_caches[0]->AddCoin(outp, std::move(coin),
false);
1040 all_caches[0]->Sync();
1044 BOOST_CHECK(!all_caches[0]->HaveCoinInCache(outp));
1052 CCoinsViewDB base{{.path =
"test", .cache_bytes = 8_MiB, .memory_only =
true}, {}};
1053 std::vector<std::unique_ptr<CCoinsViewCacheTest>> caches;
1054 caches.push_back(std::make_unique<CCoinsViewCacheTest>(&base));
1055 caches.push_back(std::make_unique<CCoinsViewCacheTest>(caches.back().get()));
1057 for (
const auto& view : caches) {
1058 TestFlushBehavior(view.get(), base, caches,
false);
1059 TestFlushBehavior(view.get(), base, caches,
true);
1069 CCoinsMap map{0, CCoinsMap::hasher{}, CCoinsMap::key_equal{}, &resource};
1078 for (
size_t i = 0; i < 1000; ++i) {
1095 cache.AddCoin(outpoint,
Coin{coin1},
false);
1112 cache.EmplaceCoinInternalDANGER(
COutPoint{outpoint},
Coin{coin1});
1116 cache.EmplaceCoinInternalDANGER(
COutPoint{outpoint},
Coin{coin2});
1124 CCoinsViewTest root{m_rng};
1127 root_cache.SetBestBlock(base_best_block);
1135 cache.EmplaceCoinInternalDANGER(
COutPoint{outpoint},
Coin{coin});
1139 cache.SetBestBlock(cache_best_block);
1142 const auto reset_guard{cache.CreateResetGuard()};
1144 BOOST_CHECK(!cache.AccessCoin(outpoint).IsSpent());
1148 BOOST_CHECK(!root_cache.HaveCoinInCache(outpoint));
1151 BOOST_CHECK(cache.AccessCoin(outpoint).IsSpent());
1155 BOOST_CHECK(!root_cache.HaveCoinInCache(outpoint));
1159 const auto reset_guard{cache.CreateResetGuard()};
1162 BOOST_CHECK(cache.AccessCoin(outpoint).IsSpent());
1166 BOOST_CHECK(!root_cache.HaveCoinInCache(outpoint));
1175 CCoinsViewTest base{m_rng};
1187 CCoinsViewCacheTest main_cache{&base};
1188 const auto fetched{main_cache.PeekCoin(outpoint)};
1191 BOOST_CHECK(!main_cache.HaveCoinInCache(outpoint));
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
int64_t CAmount
Amount in satoshis (Can be negative)
CCoinsView that adds a memory cache for transactions to another CCoinsView.
void AddCoin(const COutPoint &outpoint, Coin &&coin, bool possible_overwrite)
Add a coin.
unsigned int GetCacheSize() const
Size of the cache (in number of transaction outputs)
CoinsCachePair m_sentinel
size_t DynamicMemoryUsage() const
Calculate the size of the cache (in bytes)
void SanityCheck() const
Run an internal sanity check on the cache data structure. */.
CCoinsView backed by the coin database (chainstate/)
bool HaveCoin(const COutPoint &outpoint) const override
Just check whether a given outpoint is unspent.
Pure abstract view on the open txout dataset.
virtual void BatchWrite(CoinsViewCacheCursor &cursor, const uint256 &block_hash)=0
Do a bulk modification (multiple Coin changes + BestBlock change).
An outpoint - a combination of a transaction hash and an index n into its vout.
Serialized script, used inside transaction inputs and outputs.
bool IsUnspendable() const
Returns whether the script is guaranteed to fail at execution, regardless of the initial stack.
The basic transaction that is broadcasted on the network and contained in blocks.
const std::vector< CTxIn > vin
An output of a transaction.
Undo information for a CTransaction.
std::vector< Coin > vprevout
CTxOut out
unspent transaction output
bool IsSpent() const
Either this coin never existed (see e.g.
bool fCoinBase
whether containing transaction was a coinbase
uint32_t nHeight
at which height this containing transaction was included in the active block chain
void BatchWrite(CoinsViewCacheCursor &cursor, const uint256 &) override
Do a bulk modification (multiple Coin changes + BestBlock change).
uint256 GetBestBlock() const override
Retrieve the block hash whose state this CCoinsView currently represents.
std::optional< Coin > GetCoin(const COutPoint &) const override
Retrieve the Coin (unspent transaction output) for a given outpoint.
static CoinsViewEmpty & Get()
Double ended buffer combining vector and stream-like interfaces.
BOOST_CHECK_EXCEPTION predicates to check the specific validation error.
static void CheckAllDataAccountedFor(const PoolResource< MAX_BLOCK_SIZE_BYTES, ALIGN_BYTES > &resource)
Once all blocks are given back to the resource, tests that the freelists are consistent:
I randrange(I range) noexcept
Generate a random integer in the range [0..range), with range > 0.
uint256 rand256() noexcept
generate a random uint256.
bool randbool() noexcept
Generate a random boolean.
std::vector< B > randbytes(size_t len) noexcept
Generate random bytes.
uint32_t rand32() noexcept
Generate a random 32-bit integer.
uint64_t randbits(int bits) noexcept
Generate a random (bits)-bit integer.
CCoinsViewCacheTest cache
SingleEntryCacheTest(const CAmount base_value, const MaybeCoin &cache_coin)
Minimal stream for reading from an existing byte array by std::span.
constexpr bool IsNull() const
static constexpr unsigned int STATIC_SIZE
void assign(size_type n, const T &val)
static transaction_identifier FromUint256(const uint256 &id)
const Coin & AccessByTxid(const CCoinsViewCache &view, const Txid &txid)
Utility function to find any unspent output with a given txid.
std::pair< const COutPoint, CCoinsCacheEntry > CoinsCachePair
std::unordered_map< COutPoint, CCoinsCacheEntry, SaltedOutpointHasher, std::equal_to< COutPoint >, PoolAllocator< CoinsCachePair, sizeof(CoinsCachePair)+sizeof(void *) *4 > > CCoinsMap
PoolAllocator's MAX_BLOCK_SIZE_BYTES parameter here uses sizeof the data, and adds the size of 4 poin...
CCoinsMap::allocator_type::ResourceType CCoinsMapMemoryResource
BOOST_AUTO_TEST_CASE(ccoins_serialization)
constexpr MaybeCoin VALUE2_DIRTY
std::optional< CoinEntry > MaybeCoin
constexpr MaybeCoin VALUE2_CLEAN
constexpr MaybeCoin MISSING
static MaybeCoin GetCoinsMapEntry(const CCoinsMap &map, const COutPoint &outp=OUTPOINT)
static const COutPoint OUTPOINT
constexpr MaybeCoin VALUE2_DIRTY_FRESH
static void WriteCoinsViewEntry(CCoinsView &view, const MaybeCoin &cache_coin)
static void CheckWriteCoins(const MaybeCoin &parent, const MaybeCoin &child, const CoinOrError &expected)
int ApplyTxInUndo(Coin &&undo, CCoinsViewCache &view, const COutPoint &out)
Restore the UTXO in a Coin at a given COutPoint.
static const unsigned int NUM_SIMULATION_ITERATIONS
constexpr MaybeCoin VALUE1_CLEAN
constexpr MaybeCoin SPENT_DIRTY_FRESH
constexpr MaybeCoin SPENT_CLEAN
constexpr auto EX_OVERWRITE_UNSPENT
constexpr MaybeCoin VALUE1_DIRTY
constexpr MaybeCoin VALUE1_FRESH
static size_t InsertCoinsMapEntry(CCoinsMap &map, CoinsCachePair &sentinel, const CoinEntry &cache_coin)
static void CheckSpendCoins(const CAmount base_value, const MaybeCoin &cache_coin, const MaybeCoin &expected)
void UpdateCoins(const CTransaction &tx, CCoinsViewCache &inputs, CTxUndo &txundo, int nHeight)
constexpr MaybeCoin VALUE2_FRESH
constexpr MaybeCoin VALUE3_DIRTY_FRESH
static void CheckAccessCoin(const CAmount base_value, const MaybeCoin &cache_coin, const MaybeCoin &expected)
constexpr MaybeCoin VALUE3_DIRTY
constexpr MaybeCoin VALUE1_DIRTY_FRESH
constexpr MaybeCoin SPENT_DIRTY
constexpr auto EX_FRESH_MISAPPLIED
std::variant< MaybeCoin, std::string > CoinOrError
static void SetCoinsValue(const CAmount value, Coin &coin)
BOOST_FIXTURE_TEST_CASE(coins_cache_base_simulation_test, CacheTest)
constexpr MaybeCoin SPENT_FRESH
static void CheckAddCoin(const CAmount base_value, const MaybeCoin &cache_coin, const CAmount modify_value, const CoinOrError &expected, const bool coinbase)
BOOST_FIXTURE_TEST_SUITE(cuckoocache_tests, BasicTestingSetup)
Test Suite for CuckooCache.
BOOST_AUTO_TEST_SUITE_END()
std::string HexStr(const std::span< const uint8_t > s)
Convert a span of bytes to a lower-case hexadecimal string.
static bool sanity_check(const std::vector< CTransactionRef > &transactions, const std::map< COutPoint, CAmount > &bumpfees)
static size_t DynamicUsage(const int8_t &v)
Dynamic memory usage for built-in types is zero.
""_hex is a compile-time user-defined literal returning a std::array<std::byte>, equivalent to ParseH...
static void add_coin(const CAmount &nValue, int nInput, std::vector< OutputGroup > &set)
bool operator==(const CNetAddr &a, const CNetAddr &b)
#define BOOST_CHECK_THROW(stmt, excMatch)
#define BOOST_CHECK_EQUAL(v1, v2)
#define BOOST_CHECK(expr)
A Coin in one level of the coins database caching hierarchy.
static void SetFresh(CoinsCachePair &pair, CoinsCachePair &sentinel) noexcept
static void SetDirty(CoinsCachePair &pair, CoinsCachePair &sentinel) noexcept
A mutable version of CTransaction.
std::vector< CTxOut > vout
Txid GetHash() const
Compute the hash of this CMutableTransaction.
void SimulationTest(CCoinsView *base, bool fake_best_block)
constexpr bool IsDirty() const
friend std::ostream & operator<<(std::ostream &os, const CoinEntry &e)
bool operator==(const CoinEntry &o) const =default
constexpr bool IsDirtyFresh() const
static constexpr State ToState(const bool is_dirty, const bool is_fresh)
constexpr bool IsFresh() const
constexpr CoinEntry(const CAmount v, const State s)
Cursor for iterating over the linked list of flagged entries in CCoinsViewCache.
CoinsCachePair * NextAndMaybeErase(CoinsCachePair ¤t) noexcept
Return the next entry after current, possibly erasing current.
CoinsCachePair * Begin() const noexcept
CoinsCachePair * End() const noexcept
void TestFlushBehavior(CCoinsViewCacheTest *view, CCoinsViewDB &base, std::vector< std::unique_ptr< CCoinsViewCacheTest > > &all_caches, bool do_erasing_flush)
For CCoinsViewCache instances backed by either another cache instance or leveldb, test cache behavior...
std::map< COutPoint, std::tuple< CTransaction, CTxUndo, Coin > > UtxoData
UtxoData::iterator FindRandomFrom(const std::set< COutPoint > &utxoSet)
@ ZEROS
Seed with a compile time constant of zeros.
CAmount RandMoney(Rng &&rng)