mirror of
https://github.com/dashpay/dash.git
synced 2024-12-28 05:23:01 +01:00
51911388f2
ffe33dfbd4c3b11e3475b022b6c1dd077613de79 chainparams: drop versionbits threshold to 90% for mainnnet and signet (Anthony Towns)
f054f6bcd2c2ce5fea84cf8681013f85a444e7ea versionbits: simplify state transitions (Anthony Towns)
55ac5f568a3b73d6f1ef4654617fb76e8bcbccdf versionbits: Add explicit NEVER_ACTIVE deployments (Anthony Towns)
dd07e6da48040dc7eae46bc7941db48d98a669fd fuzz: test versionbits delayed activation (Anthony Towns)
dd85d5411c1702c8ae259610fe55050ba212e21e tests: test versionbits delayed activation (Anthony Towns)
73d4a706393e6dbd6b6d6b6428f8d3233ac0a2d8 versionbits: Add support for delayed activation (Anthony Towns)
9e6b65f6fa205eee5c3b99343988adcb8d320460 tests: clean up versionbits test (Anthony Towns)
593274445004506c921d5d851361aefb3434d744 tests: test ComputeBlockVersion for all deployments (Anthony Towns)
63879f0a4760c0c0f784029849cb5d21ee088abb tests: pull ComputeBlockVersion test into its own function (Anthony Towns)
Pull request description:
BIP9-based implementation of "speedy trial" activation specification, see https://lists.linuxfoundation.org/pipermail/bitcoin-dev/2021-March/018583.html
Edge cases are tested by fuzzing added in #21380.
ACKs for top commit:
instagibbs:
tACK ffe33dfbd4
jnewbery:
utACK ffe33dfbd4c3b11e3475b022b6c1dd077613de79
MarcoFalke:
review ACK ffe33dfbd4c3b11e3475b022b6c1dd077613de79 💈
achow101:
re-ACK ffe33dfbd4c3b11e3475b022b6c1dd077613de79
gmaxwell:
ACK ffe33dfbd4c3b11e3475b022b6c1dd077613de79
benthecarman:
ACK ffe33dfbd4c3b11e3475b022b6c1dd077613de79
Sjors:
ACK ffe33dfbd4c3b11e3475b022b6c1dd077613de79
jonatack:
Initial approach ACK ffe33dfbd4c3b11e3475b022b6c1dd077613de79 after a first pass of review, building and testing each commit, mostly looking at the changes and diffs. Will do a more high-level review iteration. A few minor comments follow to pick/choose/ignore.
ariard:
Code Review ACK ffe33df
Tree-SHA512: f79a7146b2450057ee92155cbbbcec12cd64334236d9239c6bd7d31b32eec145a9781c320f178da7b44ababdb8808b84d9d22a40e0851e229ba6d224e3be747c
332 lines
15 KiB
C++
332 lines
15 KiB
C++
// Copyright (c) 2021-2023 The Dash Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#include <test/util/setup_common.h>
|
|
|
|
#include <bls/bls.h>
|
|
#include <chainparams.h>
|
|
#include <consensus/validation.h>
|
|
#include <deploymentstatus.h>
|
|
#include <messagesigner.h>
|
|
#include <miner.h>
|
|
#include <netbase.h>
|
|
#include <script/interpreter.h>
|
|
#include <script/sign.h>
|
|
#include <script/signingprovider.h>
|
|
#include <script/standard.h>
|
|
#include <spork.h>
|
|
#include <validation.h>
|
|
|
|
#include <evo/deterministicmns.h>
|
|
#include <evo/mnhftx.h>
|
|
#include <evo/providertx.h>
|
|
#include <evo/specialtx.h>
|
|
#include <governance/governance.h>
|
|
#include <llmq/blockprocessor.h>
|
|
#include <llmq/chainlocks.h>
|
|
#include <llmq/context.h>
|
|
#include <llmq/instantsend.h>
|
|
#include <masternode/payments.h>
|
|
#include <util/enumerate.h>
|
|
#include <util/irange.h>
|
|
|
|
#include <boost/test/unit_test.hpp>
|
|
|
|
#include <map>
|
|
#include <vector>
|
|
|
|
using SimpleUTXOMap = std::map<COutPoint, std::pair<int, CAmount>>;
|
|
|
|
struct TestChainBRRBeforeActivationSetup : public TestChainSetup
|
|
{
|
|
// Force fast DIP3 activation
|
|
TestChainBRRBeforeActivationSetup() : TestChainSetup(497, {"-dip3params=30:50", "-vbparams=mn_rr:0:999999999999:0:20:16:12:5:1"}) {}
|
|
};
|
|
|
|
static SimpleUTXOMap BuildSimpleUtxoMap(const std::vector<CTransactionRef>& txs)
|
|
{
|
|
SimpleUTXOMap utxos;
|
|
for (auto [i, tx] : enumerate(txs)) {
|
|
for (auto [j, output] : enumerate(tx->vout)) {
|
|
utxos.try_emplace(COutPoint(tx->GetHash(), j), std::make_pair((int)i + 1, output.nValue));
|
|
}
|
|
}
|
|
return utxos;
|
|
}
|
|
|
|
static std::vector<COutPoint> SelectUTXOs(SimpleUTXOMap& utoxs, CAmount amount, CAmount& changeRet)
|
|
{
|
|
changeRet = 0;
|
|
|
|
std::vector<COutPoint> selectedUtxos;
|
|
CAmount selectedAmount = 0;
|
|
while (!utoxs.empty()) {
|
|
bool found = false;
|
|
for (auto it = utoxs.begin(); it != utoxs.end(); ++it) {
|
|
if (::ChainActive().Height() - it->second.first < 101) {
|
|
continue;
|
|
}
|
|
|
|
found = true;
|
|
selectedAmount += it->second.second;
|
|
selectedUtxos.emplace_back(it->first);
|
|
utoxs.erase(it);
|
|
break;
|
|
}
|
|
BOOST_ASSERT(found);
|
|
if (selectedAmount >= amount) {
|
|
changeRet = selectedAmount - amount;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return selectedUtxos;
|
|
}
|
|
|
|
static void FundTransaction(CMutableTransaction& tx, SimpleUTXOMap& utoxs, const CScript& scriptPayout, CAmount amount)
|
|
{
|
|
CAmount change;
|
|
auto inputs = SelectUTXOs(utoxs, amount, change);
|
|
for (const auto& input : inputs) {
|
|
tx.vin.emplace_back(input);
|
|
}
|
|
tx.vout.emplace_back(amount, scriptPayout);
|
|
if (change != 0) {
|
|
tx.vout.emplace_back(change, scriptPayout);
|
|
}
|
|
}
|
|
|
|
static void SignTransaction(const CTxMemPool& mempool, CMutableTransaction& tx, const CKey& coinbaseKey)
|
|
{
|
|
FillableSigningProvider tempKeystore;
|
|
tempKeystore.AddKeyPubKey(coinbaseKey, coinbaseKey.GetPubKey());
|
|
|
|
for (auto [i, input] : enumerate(tx.vin)) {
|
|
uint256 hashBlock;
|
|
CTransactionRef txFrom = GetTransaction(/* block_index */ nullptr, &mempool, input.prevout.hash, Params().GetConsensus(), hashBlock);
|
|
BOOST_ASSERT(txFrom);
|
|
BOOST_ASSERT(SignSignature(tempKeystore, *txFrom, tx, i, SIGHASH_ALL));
|
|
}
|
|
}
|
|
|
|
static CMutableTransaction CreateProRegTx(const CTxMemPool& mempool, SimpleUTXOMap& utxos, int port, const CScript& scriptPayout, const CKey& coinbaseKey, CKey& ownerKeyRet, CBLSSecretKey& operatorKeyRet)
|
|
{
|
|
ownerKeyRet.MakeNewKey(true);
|
|
operatorKeyRet.MakeNewKey();
|
|
|
|
CProRegTx proTx;
|
|
proTx.nVersion = CProRegTx::GetVersion(!bls::bls_legacy_scheme);
|
|
proTx.collateralOutpoint.n = 0;
|
|
proTx.addr = LookupNumeric("1.1.1.1", port);
|
|
proTx.keyIDOwner = ownerKeyRet.GetPubKey().GetID();
|
|
proTx.pubKeyOperator.Set(operatorKeyRet.GetPublicKey(), bls::bls_legacy_scheme.load());
|
|
proTx.keyIDVoting = ownerKeyRet.GetPubKey().GetID();
|
|
proTx.scriptPayout = scriptPayout;
|
|
|
|
CMutableTransaction tx;
|
|
tx.nVersion = 3;
|
|
tx.nType = TRANSACTION_PROVIDER_REGISTER;
|
|
FundTransaction(tx, utxos, scriptPayout, dmn_types::Regular.collat_amount);
|
|
proTx.inputsHash = CalcTxInputsHash(CTransaction(tx));
|
|
SetTxPayload(tx, proTx);
|
|
SignTransaction(mempool, tx, coinbaseKey);
|
|
|
|
return tx;
|
|
}
|
|
|
|
static CScript GenerateRandomAddress()
|
|
{
|
|
CKey key;
|
|
key.MakeNewKey(false);
|
|
return GetScriptForDestination(PKHash(key.GetPubKey()));
|
|
}
|
|
|
|
BOOST_AUTO_TEST_SUITE(block_reward_reallocation_tests)
|
|
|
|
BOOST_FIXTURE_TEST_CASE(block_reward_reallocation, TestChainBRRBeforeActivationSetup)
|
|
{
|
|
auto& dmnman = *Assert(m_node.dmnman);
|
|
const auto& consensus_params = Params().GetConsensus();
|
|
|
|
CScript coinbasePubKey = CScript() << ToByteVector(coinbaseKey.GetPubKey()) << OP_CHECKSIG;
|
|
|
|
BOOST_ASSERT(DeploymentDIP0003Enforced(WITH_LOCK(cs_main, return m_node.chainman->ActiveChain().Height()), consensus_params));
|
|
|
|
// Register one MN
|
|
CKey ownerKey;
|
|
CBLSSecretKey operatorKey;
|
|
auto utxos = BuildSimpleUtxoMap(m_coinbase_txns);
|
|
auto tx = CreateProRegTx(*m_node.mempool, utxos, 1, GenerateRandomAddress(), coinbaseKey, ownerKey, operatorKey);
|
|
|
|
CreateAndProcessBlock({tx}, coinbaseKey);
|
|
|
|
{
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
dmnman.UpdatedBlockTip(tip);
|
|
|
|
BOOST_ASSERT(dmnman.GetListAtChainTip().HasMN(tx.GetHash()));
|
|
|
|
BOOST_CHECK_EQUAL(tip->nHeight, 498);
|
|
BOOST_CHECK(tip->nHeight < Params().GetConsensus().BRRHeight);
|
|
}
|
|
|
|
CreateAndProcessBlock({}, coinbaseKey);
|
|
|
|
{
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
BOOST_CHECK_EQUAL(tip->nHeight, 499);
|
|
dmnman.UpdatedBlockTip(tip);
|
|
BOOST_ASSERT(dmnman.GetListAtChainTip().HasMN(tx.GetHash()));
|
|
BOOST_CHECK(tip->nHeight < Params().GetConsensus().BRRHeight);
|
|
// Creating blocks by different ways
|
|
const auto pblocktemplate = BlockAssembler(m_node.chainman->ActiveChainstate(), m_node, *m_node.mempool, Params()).CreateNewBlock(coinbasePubKey);
|
|
}
|
|
for ([[maybe_unused]] auto _ : irange::range(1999)) {
|
|
CreateAndProcessBlock({}, coinbaseKey);
|
|
LOCK(cs_main);
|
|
dmnman.UpdatedBlockTip(m_node.chainman->ActiveChain().Tip());
|
|
}
|
|
BOOST_CHECK(m_node.chainman->ActiveChain().Height() < Params().GetConsensus().BRRHeight);
|
|
CreateAndProcessBlock({}, coinbaseKey);
|
|
|
|
{
|
|
// Advance to ACTIVE at height = (BRRHeight - 1)
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
BOOST_CHECK_EQUAL(tip->nHeight, Params().GetConsensus().BRRHeight - 1);
|
|
dmnman.UpdatedBlockTip(tip);
|
|
BOOST_ASSERT(dmnman.GetListAtChainTip().HasMN(tx.GetHash()));
|
|
}
|
|
|
|
{
|
|
// Reward split should stay ~50/50 before the first superblock after activation.
|
|
// This applies even if reallocation was activated right at superblock height like it does here.
|
|
// next block should be signaling by default
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
const bool isV20Active{DeploymentActiveAfter(tip, consensus_params, Consensus::DEPLOYMENT_V20)};
|
|
dmnman.UpdatedBlockTip(tip);
|
|
BOOST_ASSERT(dmnman.GetListAtChainTip().HasMN(tx.GetHash()));
|
|
const CAmount block_subsidy = GetBlockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
const CAmount masternode_payment = GetMasternodePayment(tip->nHeight, block_subsidy, isV20Active);
|
|
const auto pblocktemplate = BlockAssembler(m_node.chainman->ActiveChainstate(), m_node, *m_node.mempool, Params()).CreateNewBlock(coinbasePubKey);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[0].nValue, masternode_payment);
|
|
}
|
|
|
|
for ([[maybe_unused]] auto _ : irange::range(consensus_params.nSuperblockCycle - 1)) {
|
|
CreateAndProcessBlock({}, coinbaseKey);
|
|
}
|
|
|
|
{
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
const bool isV20Active{DeploymentActiveAfter(tip, consensus_params, Consensus::DEPLOYMENT_V20)};
|
|
const CAmount block_subsidy = GetBlockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
const CAmount masternode_payment = GetMasternodePayment(tip->nHeight, block_subsidy, isV20Active);
|
|
const auto pblocktemplate = BlockAssembler(m_node.chainman->ActiveChainstate(), m_node, *m_node.mempool, Params()).CreateNewBlock(coinbasePubKey);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->block.vtx[0]->GetValueOut(), 122209530);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[0].nValue, masternode_payment);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[0].nValue, 61104762); // 0.4999999755
|
|
}
|
|
|
|
// Reallocation should kick-in with the superblock mined at height = 2010,
|
|
// there will be 19 adjustments, 3 superblocks long each
|
|
for ([[maybe_unused]] auto i : irange::range(19)) {
|
|
for ([[maybe_unused]] auto j : irange::range(3)) {
|
|
for ([[maybe_unused]] auto k : irange::range(consensus_params.nSuperblockCycle)) {
|
|
CreateAndProcessBlock({}, coinbaseKey);
|
|
}
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
const bool isV20Active{DeploymentActiveAfter(tip, consensus_params, Consensus::DEPLOYMENT_V20)};
|
|
const CAmount block_subsidy = GetBlockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
const CAmount masternode_payment = GetMasternodePayment(tip->nHeight, block_subsidy, isV20Active);
|
|
const auto pblocktemplate = BlockAssembler(m_node.chainman->ActiveChainstate(), m_node, *m_node.mempool, Params()).CreateNewBlock(coinbasePubKey);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[0].nValue, masternode_payment);
|
|
}
|
|
}
|
|
BOOST_CHECK(DeploymentActiveAfter(m_node.chainman->ActiveChain().Tip(), consensus_params, Consensus::DEPLOYMENT_V20));
|
|
// Allocation of block subsidy is 60% MN, 20% miners and 20% treasury
|
|
{
|
|
// Reward split should reach ~75/25 after reallocation is done
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
const bool isV20Active{DeploymentActiveAfter(tip, consensus_params, Consensus::DEPLOYMENT_V20)};
|
|
const CAmount block_subsidy = GetBlockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
const CAmount block_subsidy_sb = GetSuperblockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
CAmount block_subsidy_potential = block_subsidy + block_subsidy_sb;
|
|
BOOST_CHECK_EQUAL(block_subsidy_potential, 84437941);
|
|
CAmount expected_block_reward = block_subsidy_potential - block_subsidy_potential / 5;
|
|
|
|
const CAmount masternode_payment = GetMasternodePayment(tip->nHeight, block_subsidy, isV20Active);
|
|
const auto pblocktemplate = BlockAssembler(m_node.chainman->ActiveChainstate(), m_node, *m_node.mempool, Params()).CreateNewBlock(coinbasePubKey);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->block.vtx[0]->GetValueOut(), expected_block_reward);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->block.vtx[0]->GetValueOut(), 67550353);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[0].nValue, masternode_payment);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[0].nValue, 50662764); // 0.75
|
|
}
|
|
BOOST_CHECK(!DeploymentActiveAfter(m_node.chainman->ActiveChain().Tip(), consensus_params, Consensus::DEPLOYMENT_MN_RR));
|
|
|
|
// Activate EHF "MN_RR"
|
|
{
|
|
LOCK(cs_main);
|
|
m_node.mnhf_manager->AddSignal(m_node.chainman->ActiveChain().Tip(), 10);
|
|
}
|
|
|
|
// Reward split should stay ~75/25 after reallocation is done,
|
|
// check 10 next superblocks
|
|
for ([[maybe_unused]] auto i : irange::range(10)) {
|
|
for ([[maybe_unused]] auto k : irange::range(consensus_params.nSuperblockCycle)) {
|
|
CreateAndProcessBlock({}, coinbaseKey);
|
|
}
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
const bool isV20Active{DeploymentActiveAfter(tip, consensus_params, Consensus::DEPLOYMENT_V20)};
|
|
const bool isMNRewardReallocated{DeploymentActiveAfter(tip, consensus_params, Consensus::DEPLOYMENT_MN_RR)};
|
|
const CAmount block_subsidy = GetBlockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
CAmount masternode_payment = GetMasternodePayment(tip->nHeight, block_subsidy, isV20Active);
|
|
const auto pblocktemplate = BlockAssembler(m_node.chainman->ActiveChainstate(), m_node, *m_node.mempool, Params()).CreateNewBlock(coinbasePubKey);
|
|
|
|
if (isMNRewardReallocated) {
|
|
const CAmount platform_payment = PlatformShare(masternode_payment);
|
|
masternode_payment -= platform_payment;
|
|
}
|
|
size_t payment_index = isMNRewardReallocated ? 1 : 0;
|
|
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[payment_index].nValue, masternode_payment);
|
|
}
|
|
|
|
BOOST_CHECK(DeploymentActiveAfter(m_node.chainman->ActiveChain().Tip(), consensus_params, Consensus::DEPLOYMENT_MN_RR));
|
|
{ // At this moment Masternode reward should be reallocated to platform
|
|
// Allocation of block subsidy is 60% MN, 20% miners and 20% treasury
|
|
LOCK(cs_main);
|
|
const CBlockIndex* const tip{m_node.chainman->ActiveChain().Tip()};
|
|
const bool isV20Active{DeploymentActiveAfter(tip, consensus_params, Consensus::DEPLOYMENT_V20)};
|
|
const CAmount block_subsidy = GetBlockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
const CAmount block_subsidy_sb = GetSuperblockSubsidyInner(tip->nBits, tip->nHeight, consensus_params, isV20Active);
|
|
CAmount masternode_payment = GetMasternodePayment(tip->nHeight, block_subsidy, isV20Active);
|
|
const CAmount platform_payment = PlatformShare(masternode_payment);
|
|
masternode_payment -= platform_payment;
|
|
const auto pblocktemplate = BlockAssembler(m_node.chainman->ActiveChainstate(), m_node, *m_node.mempool, Params()).CreateNewBlock(coinbasePubKey);
|
|
|
|
CAmount block_subsidy_potential = block_subsidy + block_subsidy_sb;
|
|
BOOST_CHECK_EQUAL(tip->nHeight, 3858);
|
|
BOOST_CHECK_EQUAL(block_subsidy_potential, 78406660);
|
|
// Treasury is 20% since MNRewardReallocation
|
|
CAmount expected_block_reward = block_subsidy_potential - block_subsidy_potential / 5;
|
|
// Since MNRewardReallocation, MN reward share is 75% of the block reward
|
|
CAmount expected_masternode_reward = expected_block_reward * 3 / 4;
|
|
CAmount expected_mn_platform_payment = PlatformShare(expected_masternode_reward);
|
|
CAmount expected_mn_core_payment = expected_masternode_reward - expected_mn_platform_payment;
|
|
|
|
BOOST_CHECK_EQUAL(pblocktemplate->block.vtx[0]->GetValueOut(), expected_block_reward);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[1].nValue, masternode_payment);
|
|
BOOST_CHECK_EQUAL(pblocktemplate->voutMasternodePayments[1].nValue, expected_mn_core_payment);
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|