2019-03-28 12:41:01 -03:00
|
|
|
// Copyright (c) 2010 Satoshi Nakamoto
|
2020-04-16 13:14:08 -04:00
|
|
|
// Copyright (c) 2009-2020 The Bitcoin Core developers
|
2019-03-28 12:41:01 -03:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#include <node/coinstats.h>
|
|
|
|
|
|
|
|
#include <coins.h>
|
|
|
|
#include <hash.h>
|
|
|
|
#include <serialize.h>
|
|
|
|
#include <uint256.h>
|
|
|
|
#include <util/system.h>
|
2020-06-02 17:41:04 -04:00
|
|
|
#include <validation.h>
|
2019-03-28 12:41:01 -03:00
|
|
|
|
|
|
|
#include <map>
|
|
|
|
|
2020-06-02 17:41:04 -04:00
|
|
|
static uint64_t GetBogoSize(const CScript& scriptPubKey)
|
|
|
|
{
|
|
|
|
return 32 /* txid */ +
|
|
|
|
4 /* vout index */ +
|
|
|
|
4 /* height + coinbase */ +
|
|
|
|
8 /* amount */ +
|
|
|
|
2 /* scriptPubKey len */ +
|
|
|
|
scriptPubKey.size() /* scriptPubKey */;
|
|
|
|
}
|
|
|
|
|
2020-06-02 17:52:34 -04:00
|
|
|
static void ApplyStats(CCoinsStats& stats, CHashWriter& ss, const uint256& hash, const std::map<uint32_t, Coin>& outputs)
|
2019-03-28 12:41:01 -03:00
|
|
|
{
|
|
|
|
assert(!outputs.empty());
|
|
|
|
ss << hash;
|
|
|
|
ss << VARINT(outputs.begin()->second.nHeight * 2 + outputs.begin()->second.fCoinBase ? 1u : 0u);
|
|
|
|
stats.nTransactions++;
|
|
|
|
for (const auto& output : outputs) {
|
|
|
|
ss << VARINT(output.first + 1);
|
|
|
|
ss << output.second.out.scriptPubKey;
|
2020-02-07 00:57:32 -03:00
|
|
|
ss << VARINT_MODE(output.second.out.nValue, VarIntMode::NONNEGATIVE_SIGNED);
|
2019-03-28 12:41:01 -03:00
|
|
|
stats.nTransactionOutputs++;
|
|
|
|
stats.nTotalAmount += output.second.out.nValue;
|
2020-06-02 17:41:04 -04:00
|
|
|
stats.nBogoSize += GetBogoSize(output.second.out.scriptPubKey);
|
2019-03-28 12:41:01 -03:00
|
|
|
}
|
|
|
|
ss << VARINT(0u);
|
|
|
|
}
|
|
|
|
|
2020-06-02 17:56:28 -04:00
|
|
|
static void ApplyStats(CCoinsStats& stats, std::nullptr_t, const uint256& hash, const std::map<uint32_t, Coin>& outputs)
|
|
|
|
{
|
|
|
|
assert(!outputs.empty());
|
|
|
|
stats.nTransactions++;
|
|
|
|
for (const auto& output : outputs) {
|
|
|
|
stats.nTransactionOutputs++;
|
|
|
|
stats.nTotalAmount += output.second.out.nValue;
|
|
|
|
stats.nBogoSize += GetBogoSize(output.second.out.scriptPubKey);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-28 12:41:01 -03:00
|
|
|
//! Calculate statistics about the unspent transaction output set
|
2020-06-02 17:52:34 -04:00
|
|
|
template <typename T>
|
|
|
|
static bool GetUTXOStats(CCoinsView* view, CCoinsStats& stats, T hash_obj, const std::function<void()>& interruption_point)
|
2019-03-28 12:41:01 -03:00
|
|
|
{
|
2019-03-28 13:09:06 -03:00
|
|
|
stats = CCoinsStats();
|
2019-03-28 12:41:01 -03:00
|
|
|
std::unique_ptr<CCoinsViewCursor> pcursor(view->Cursor());
|
|
|
|
assert(pcursor);
|
|
|
|
|
|
|
|
stats.hashBlock = pcursor->GetBestBlock();
|
|
|
|
{
|
|
|
|
LOCK(cs_main);
|
|
|
|
stats.nHeight = LookupBlockIndex(stats.hashBlock)->nHeight;
|
|
|
|
}
|
2020-06-02 17:52:34 -04:00
|
|
|
|
|
|
|
PrepareHash(hash_obj, stats);
|
|
|
|
|
2019-03-28 12:41:01 -03:00
|
|
|
uint256 prevkey;
|
|
|
|
std::map<uint32_t, Coin> outputs;
|
|
|
|
while (pcursor->Valid()) {
|
2020-05-22 16:09:34 -04:00
|
|
|
interruption_point();
|
2019-03-28 12:41:01 -03:00
|
|
|
COutPoint key;
|
|
|
|
Coin coin;
|
|
|
|
if (pcursor->GetKey(key) && pcursor->GetValue(coin)) {
|
|
|
|
if (!outputs.empty() && key.hash != prevkey) {
|
2020-06-02 17:52:34 -04:00
|
|
|
ApplyStats(stats, hash_obj, prevkey, outputs);
|
2019-03-28 12:41:01 -03:00
|
|
|
outputs.clear();
|
|
|
|
}
|
|
|
|
prevkey = key.hash;
|
|
|
|
outputs[key.n] = std::move(coin);
|
2019-03-28 13:09:06 -03:00
|
|
|
stats.coins_count++;
|
2019-03-28 12:41:01 -03:00
|
|
|
} else {
|
|
|
|
return error("%s: unable to read value", __func__);
|
|
|
|
}
|
|
|
|
pcursor->Next();
|
|
|
|
}
|
|
|
|
if (!outputs.empty()) {
|
2020-06-02 17:52:34 -04:00
|
|
|
ApplyStats(stats, hash_obj, prevkey, outputs);
|
2019-03-28 12:41:01 -03:00
|
|
|
}
|
2020-06-02 17:52:34 -04:00
|
|
|
|
|
|
|
FinalizeHash(hash_obj, stats);
|
|
|
|
|
2019-03-28 12:41:01 -03:00
|
|
|
stats.nDiskSize = view->EstimateSize();
|
|
|
|
return true;
|
|
|
|
}
|
2020-06-02 17:52:34 -04:00
|
|
|
|
|
|
|
bool GetUTXOStats(CCoinsView* view, CCoinsStats& stats, CoinStatsHashType hash_type, const std::function<void()>& interruption_point)
|
|
|
|
{
|
|
|
|
switch (hash_type) {
|
|
|
|
case(CoinStatsHashType::HASH_SERIALIZED): {
|
|
|
|
CHashWriter ss(SER_GETHASH, PROTOCOL_VERSION);
|
|
|
|
return GetUTXOStats(view, stats, ss, interruption_point);
|
|
|
|
}
|
2020-06-02 17:56:28 -04:00
|
|
|
case(CoinStatsHashType::NONE): {
|
|
|
|
return GetUTXOStats(view, stats, nullptr, interruption_point);
|
|
|
|
}
|
2020-06-02 17:52:34 -04:00
|
|
|
} // no default case, so the compiler can warn about missing cases
|
|
|
|
assert(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// The legacy hash serializes the hashBlock
|
|
|
|
static void PrepareHash(CHashWriter& ss, CCoinsStats& stats)
|
|
|
|
{
|
|
|
|
ss << stats.hashBlock;
|
|
|
|
}
|
2020-06-02 17:56:28 -04:00
|
|
|
static void PrepareHash(std::nullptr_t, CCoinsStats& stats) {}
|
2020-06-02 17:52:34 -04:00
|
|
|
|
|
|
|
static void FinalizeHash(CHashWriter& ss, CCoinsStats& stats)
|
|
|
|
{
|
|
|
|
stats.hashSerialized = ss.GetHash();
|
|
|
|
}
|
2020-06-02 17:56:28 -04:00
|
|
|
static void FinalizeHash(std::nullptr_t, CCoinsStats& stats) {}
|