2018-07-26 18:36:45 -04:00
|
|
|
// Copyright (c) 2016-2018 The Bitcoin Core developers
|
2016-10-02 17:38:48 -04:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2017-11-10 13:57:53 +13:00
|
|
|
#include <bench/bench.h>
|
|
|
|
#include <coins.h>
|
|
|
|
#include <policy/policy.h>
|
|
|
|
#include <wallet/crypter.h>
|
2016-10-02 17:38:48 -04:00
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
// FIXME: Dedup with SetupDummyInputs in test/transaction_tests.cpp.
|
|
|
|
//
|
|
|
|
// Helper: create two dummy transactions, each with
|
2018-09-17 14:33:52 -04:00
|
|
|
// two outputs. The first has 11 and 50 COIN outputs
|
|
|
|
// paid to a TX_PUBKEY, the second 21 and 22 COIN outputs
|
2016-10-02 17:38:48 -04:00
|
|
|
// paid to a TX_PUBKEYHASH.
|
|
|
|
//
|
|
|
|
static std::vector<CMutableTransaction>
|
|
|
|
SetupDummyInputs(CBasicKeyStore& keystoreRet, CCoinsViewCache& coinsRet)
|
|
|
|
{
|
|
|
|
std::vector<CMutableTransaction> dummyTransactions;
|
|
|
|
dummyTransactions.resize(2);
|
|
|
|
|
|
|
|
// Add some keys to the keystore:
|
|
|
|
CKey key[4];
|
|
|
|
for (int i = 0; i < 4; i++) {
|
|
|
|
key[i].MakeNewKey(i % 2);
|
|
|
|
keystoreRet.AddKey(key[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create some dummy input transactions
|
|
|
|
dummyTransactions[0].vout.resize(2);
|
2018-09-17 14:33:52 -04:00
|
|
|
dummyTransactions[0].vout[0].nValue = 11 * COIN;
|
2016-10-02 17:38:48 -04:00
|
|
|
dummyTransactions[0].vout[0].scriptPubKey << ToByteVector(key[0].GetPubKey()) << OP_CHECKSIG;
|
2018-09-17 14:33:52 -04:00
|
|
|
dummyTransactions[0].vout[1].nValue = 50 * COIN;
|
2016-10-02 17:38:48 -04:00
|
|
|
dummyTransactions[0].vout[1].scriptPubKey << ToByteVector(key[1].GetPubKey()) << OP_CHECKSIG;
|
2017-04-25 11:29:30 -07:00
|
|
|
AddCoins(coinsRet, dummyTransactions[0], 0);
|
2016-10-02 17:38:48 -04:00
|
|
|
|
|
|
|
dummyTransactions[1].vout.resize(2);
|
2018-09-17 14:33:52 -04:00
|
|
|
dummyTransactions[1].vout[0].nValue = 21 * COIN;
|
2016-10-02 17:38:48 -04:00
|
|
|
dummyTransactions[1].vout[0].scriptPubKey = GetScriptForDestination(key[2].GetPubKey().GetID());
|
2018-09-17 14:33:52 -04:00
|
|
|
dummyTransactions[1].vout[1].nValue = 22 * COIN;
|
2016-10-02 17:38:48 -04:00
|
|
|
dummyTransactions[1].vout[1].scriptPubKey = GetScriptForDestination(key[3].GetPubKey().GetID());
|
2017-04-25 11:29:30 -07:00
|
|
|
AddCoins(coinsRet, dummyTransactions[1], 0);
|
2016-10-02 17:38:48 -04:00
|
|
|
|
|
|
|
return dummyTransactions;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Microbenchmark for simple accesses to a CCoinsViewCache database. Note from
|
|
|
|
// laanwj, "replicating the actual usage patterns of the client is hard though,
|
|
|
|
// many times micro-benchmarks of the database showed completely different
|
|
|
|
// characteristics than e.g. reindex timings. But that's not a requirement of
|
|
|
|
// every benchmark."
|
|
|
|
// (https://github.com/bitcoin/bitcoin/issues/7883#issuecomment-224807484)
|
|
|
|
static void CCoinsCaching(benchmark::State& state)
|
|
|
|
{
|
|
|
|
CBasicKeyStore keystore;
|
|
|
|
CCoinsView coinsDummy;
|
|
|
|
CCoinsViewCache coins(&coinsDummy);
|
|
|
|
std::vector<CMutableTransaction> dummyTransactions = SetupDummyInputs(keystore, coins);
|
|
|
|
|
|
|
|
CMutableTransaction t1;
|
|
|
|
t1.vin.resize(3);
|
|
|
|
t1.vin[0].prevout.hash = dummyTransactions[0].GetHash();
|
|
|
|
t1.vin[0].prevout.n = 1;
|
|
|
|
t1.vin[0].scriptSig << std::vector<unsigned char>(65, 0);
|
|
|
|
t1.vin[1].prevout.hash = dummyTransactions[1].GetHash();
|
|
|
|
t1.vin[1].prevout.n = 0;
|
|
|
|
t1.vin[1].scriptSig << std::vector<unsigned char>(65, 0) << std::vector<unsigned char>(33, 4);
|
|
|
|
t1.vin[2].prevout.hash = dummyTransactions[1].GetHash();
|
|
|
|
t1.vin[2].prevout.n = 1;
|
|
|
|
t1.vin[2].scriptSig << std::vector<unsigned char>(65, 0) << std::vector<unsigned char>(33, 4);
|
|
|
|
t1.vout.resize(2);
|
2018-09-17 14:33:52 -04:00
|
|
|
t1.vout[0].nValue = 90 * COIN;
|
2016-10-02 17:38:48 -04:00
|
|
|
t1.vout[0].scriptPubKey << OP_1;
|
|
|
|
|
|
|
|
// Benchmark.
|
|
|
|
while (state.KeepRunning()) {
|
|
|
|
bool success = AreInputsStandard(t1, coins);
|
|
|
|
assert(success);
|
|
|
|
CAmount value = coins.GetValueIn(t1);
|
2018-09-17 14:33:52 -04:00
|
|
|
assert(value == (50 + 21 + 22) * COIN);
|
2016-10-02 17:38:48 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-17 16:48:02 +02:00
|
|
|
BENCHMARK(CCoinsCaching, 170 * 1000);
|