Skip to content

Commit

Permalink
Simplify return values of GetCoin/HaveCoin(InCache)
Browse files Browse the repository at this point in the history
This removes the possibility for GetCoin/HaveCoin/HaveCoinInCache to return
true while the respective coin is spent. By doing it across all calls, some
extra checks can be eliminated.

coins_tests is modified to call HaveCoin sometimes before and sometimes
after AccessCoin. A further change is needed because the semantics for
GetCoin slightly changed, causing a pruned entry in the parent cache to not
be pulled into the child in FetchCoin.
  • Loading branch information
sipa committed Jun 26, 2017
1 parent 234ffc6 commit 21180ff
Show file tree
Hide file tree
Showing 5 changed files with 28 additions and 19 deletions.
8 changes: 6 additions & 2 deletions src/coins.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -11,11 +11,15 @@
#include <assert.h>

bool CCoinsView::GetCoin(const COutPoint &outpoint, Coin &coin) const { return false; }
bool CCoinsView::HaveCoin(const COutPoint &outpoint) const { return false; }
uint256 CCoinsView::GetBestBlock() const { return uint256(); }
bool CCoinsView::BatchWrite(CCoinsMap &mapCoins, const uint256 &hashBlock) { return false; }
CCoinsViewCursor *CCoinsView::Cursor() const { return 0; }

bool CCoinsView::HaveCoin(const COutPoint &outpoint) const
{
Coin coin;
return GetCoin(outpoint, coin);
}

CCoinsViewBacked::CCoinsViewBacked(CCoinsView *viewIn) : base(viewIn) { }
bool CCoinsViewBacked::GetCoin(const COutPoint &outpoint, Coin &coin) const { return base->GetCoin(outpoint, coin); }
Expand Down Expand Up @@ -55,7 +59,7 @@ bool CCoinsViewCache::GetCoin(const COutPoint &outpoint, Coin &coin) const {
CCoinsMap::const_iterator it = FetchCoin(outpoint);
if (it != cacheCoins.end()) {
coin = it->second.coin;
return true;
return !coin.IsSpent();
}
return false;
}
Expand Down
8 changes: 5 additions & 3 deletions src/coins.h
Original file line number Diff line number Diff line change
Expand Up @@ -145,11 +145,13 @@ class CCoinsViewCursor
class CCoinsView
{
public:
//! Retrieve the Coin (unspent transaction output) for a given outpoint.
/** Retrieve the Coin (unspent transaction output) for a given outpoint.
* Returns true only when an unspent coin was found, which is returned in coin.
* When false is returned, coin's value is unspecified.
*/
virtual bool GetCoin(const COutPoint &outpoint, Coin &coin) const;

//! Just check whether we have data for a given outpoint.
//! This may (but cannot always) return true for spent outputs.
//! Just check whether a given outpoint is unspent.
virtual bool HaveCoin(const COutPoint &outpoint) const;

//! Retrieve the block hash whose state this CCoinsView currently represents
Expand Down
22 changes: 15 additions & 7 deletions src/test/coins_tests.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -50,12 +50,6 @@ class CCoinsViewTest : public CCoinsView
return true;
}

bool HaveCoin(const COutPoint& outpoint) const override
{
Coin coin;
return GetCoin(outpoint, coin);
}

uint256 GetBestBlock() const override { return hashBestBlock_; }

bool BatchWrite(CCoinsMap& mapCoins, const uint256& hashBlock) override
Expand Down Expand Up @@ -147,8 +141,22 @@ BOOST_AUTO_TEST_CASE(coins_cache_simulation_test)
{
uint256 txid = txids[InsecureRandRange(txids.size())]; // txid we're going to modify in this iteration.
Coin& coin = result[COutPoint(txid, 0)];

// Determine whether to test HaveCoin before or after Access* (or both). As these functions
// can influence each other's behaviour by pulling things into the cache, all combinations
// are tested.
bool test_havecoin_before = InsecureRandBits(2) == 0;
bool test_havecoin_after = InsecureRandBits(2) == 0;

bool result_havecoin = test_havecoin_before ? stack.back()->HaveCoin(COutPoint(txid, 0)) : false;
const Coin& entry = (InsecureRandRange(500) == 0) ? AccessByTxid(*stack.back(), txid) : stack.back()->AccessCoin(COutPoint(txid, 0));
BOOST_CHECK(coin == entry);
BOOST_CHECK(!test_havecoin_before || result_havecoin == !entry.IsSpent());

if (test_havecoin_after) {
bool ret = stack.back()->HaveCoin(COutPoint(txid, 0));
BOOST_CHECK(ret == !entry.IsSpent());
}

if (InsecureRandRange(5) == 0 || coin.IsSpent()) {
Coin newcoin;
Expand Down Expand Up @@ -628,7 +636,7 @@ BOOST_AUTO_TEST_CASE(ccoins_access)
CheckAccessCoin(ABSENT, VALUE2, VALUE2, FRESH , FRESH );
CheckAccessCoin(ABSENT, VALUE2, VALUE2, DIRTY , DIRTY );
CheckAccessCoin(ABSENT, VALUE2, VALUE2, DIRTY|FRESH, DIRTY|FRESH);
CheckAccessCoin(PRUNED, ABSENT, PRUNED, NO_ENTRY , FRESH );
CheckAccessCoin(PRUNED, ABSENT, ABSENT, NO_ENTRY , NO_ENTRY );
CheckAccessCoin(PRUNED, PRUNED, PRUNED, 0 , 0 );
CheckAccessCoin(PRUNED, PRUNED, PRUNED, FRESH , FRESH );
CheckAccessCoin(PRUNED, PRUNED, PRUNED, DIRTY , DIRTY );
Expand Down
6 changes: 1 addition & 5 deletions src/txmempool.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -903,11 +903,7 @@ bool CCoinsViewMemPool::GetCoin(const COutPoint &outpoint, Coin &coin) const {
return false;
}
}
return (base->GetCoin(outpoint, coin) && !coin.IsSpent());
}

bool CCoinsViewMemPool::HaveCoin(const COutPoint &outpoint) const {
return mempool.exists(outpoint) || base->HaveCoin(outpoint);
return base->GetCoin(outpoint, coin);
}

size_t CTxMemPool::DynamicMemoryUsage() const {
Expand Down
3 changes: 1 addition & 2 deletions src/txmempool.h
Original file line number Diff line number Diff line change
Expand Up @@ -684,8 +684,7 @@ class CCoinsViewMemPool : public CCoinsViewBacked

public:
CCoinsViewMemPool(CCoinsView* baseIn, const CTxMemPool& mempoolIn);
bool GetCoin(const COutPoint &outpoint, Coin &coin) const;
bool HaveCoin(const COutPoint &outpoint) const;
bool GetCoin(const COutPoint &outpoint, Coin &coin) const override;
};

/**
Expand Down

0 comments on commit 21180ff

Please sign in to comment.