Compare commits

...

5 commits

Author SHA1 Message Date
Hennadii Stepanov
ffb6c63efc
Merge d838f1bac6 into 66aa6a47bd 2025-01-08 20:45:17 +01:00
glozow
66aa6a47bd
Merge bitcoin/bitcoin#30391: BlockAssembler: return selected packages virtual size and fee
Some checks failed
CI / test each commit (push) Has been cancelled
CI / macOS 14 native, arm64, no depends, sqlite only, gui (push) Has been cancelled
CI / macOS 14 native, arm64, fuzz (push) Has been cancelled
CI / Win64 native, VS 2022 (push) Has been cancelled
CI / Win64 native fuzz, VS 2022 (push) Has been cancelled
CI / ASan + LSan + UBSan + integer, no depends, USDT (push) Has been cancelled
7c123c08dd  miner: add package feerate vector to CBlockTemplate (ismaelsadeeq)

Pull request description:

  This PR enables `BlockAssembler` to add all selected packages' fee and virtual size to a vector, and then return the vector as a member of `CBlockTemplate` struct.

  This PR is the first step in the https://github.com/bitcoin/bitcoin/issues/30392 project.

  The packages' vsize and fee are used in #30157 to select a percentile fee rate of the top block in the mempool.

ACKs for top commit:
  rkrux:
    tACK 7c123c08dd
  ryanofsky:
    Code review ACK 7c123c08dd. Changes since last review are rebasing due to a test conflict, giving the new field a better name and description, resolving the test conflict, and renaming a lot of test variables. The actual code change is still one-line change.
  glozow:
    reACK 7c123c08dd

Tree-SHA512: 767b0b3d4273cf1589fd2068d729a66c7414c0f9574b15989fbe293f8c85cd6c641dd783cde55bfabab32cd047d7d8a071d6897b06ed4295c0d071e588de0861
2025-01-08 13:01:23 -05:00
ismaelsadeeq
7c123c08dd
miner: add package feerate vector to CBlockTemplate
- The package feerates are ordered by the sequence in which
  packages are selected for inclusion in the block template.

- The commit also tests this new behaviour.

Co-authored-by: willcl-ark <will@256k1.dev>
2025-01-07 15:29:17 -05:00
Hennadii Stepanov
d838f1bac6
ci, iwyu: Treat warnings as errors for specific targets
Currently, this applies only to the `bitcoin_clientversion` and
`bitcoin_crypto` targets.
2024-12-12 10:43:53 +00:00
Hennadii Stepanov
6df1a0ce23
refactor: Fix includes for libbitcoin_crypto 2024-12-12 10:43:46 +00:00
23 changed files with 79 additions and 28 deletions

View file

@ -172,6 +172,7 @@ if [ "${RUN_TIDY}" = "true" ]; then
echo "^^^ ⚠️ Failure generated from clang-tidy" echo "^^^ ⚠️ Failure generated from clang-tidy"
false false
fi fi
# Filter out: # Filter out:
# * qt qrc and moc generated files # * qt qrc and moc generated files
jq 'map(select(.file | test("src/qt/qrc_.*\\.cpp$|/moc_.*\\.cpp$") | not))' "${BASE_BUILD_DIR}/compile_commands.json" > tmp.json jq 'map(select(.file | test("src/qt/qrc_.*\\.cpp$|/moc_.*\\.cpp$") | not))' "${BASE_BUILD_DIR}/compile_commands.json" > tmp.json
@ -182,6 +183,13 @@ if [ "${RUN_TIDY}" = "true" ]; then
-- -Xiwyu --cxx17ns -Xiwyu --mapping_file="${BASE_ROOT_DIR}/contrib/devtools/iwyu/bitcoin.core.imp" \ -- -Xiwyu --cxx17ns -Xiwyu --mapping_file="${BASE_ROOT_DIR}/contrib/devtools/iwyu/bitcoin.core.imp" \
-Xiwyu --max_line_length=160 \ -Xiwyu --max_line_length=160 \
2>&1 | tee /tmp/iwyu_ci.out 2>&1 | tee /tmp/iwyu_ci.out
# TODO: Expand the following list to all targets.
TARGETS_WITH_FORCED_IWYU="bitcoin_clientversion bitcoin_crypto"
cd "${BASE_BUILD_DIR}"
bash -c "cmake -S ${BASE_ROOT_DIR} -DCMAKE_CXX_INCLUDE_WHAT_YOU_USE=\"include-what-you-use;-Xiwyu;--cxx17ns;-Xiwyu;--mapping_file=${BASE_ROOT_DIR}/contrib/devtools/iwyu/bitcoin.core.imp;-Xiwyu;--error\""
bash -c "cmake --build . ${MAKEJOBS} --clean-first --target ${TARGETS_WITH_FORCED_IWYU}"
cd "${BASE_ROOT_DIR}/src" cd "${BASE_ROOT_DIR}/src"
python3 "/include-what-you-use/fix_includes.py" --nosafe_headers < /tmp/iwyu_ci.out python3 "/include-what-you-use/fix_includes.py" --nosafe_headers < /tmp/iwyu_ci.out
git --no-pager diff git --no-pager diff

View file

@ -1,3 +1,11 @@
# Nothing for now.
[ [
# Compiler intrinsics
{ include: [ "<emmintrin.h>", private, "<immintrin.h>", public ] },
{ include: [ "<smmintrin.h>", private, "<immintrin.h>", public ] },
{ include: [ "<tmmintrin.h>", private, "<immintrin.h>", public ] },
# False positive warnings
# See: https://github.com/include-what-you-use/include-what-you-use/issues/1616
{ include: [ "<iterator>", public, "<utility>", public ] },
{ include: [ "<tuple>", public, "<array>", public ] },
] ]

View file

@ -12,7 +12,7 @@
#include <algorithm> #include <algorithm>
#include <bit> #include <bit>
#include <string.h> #include <cassert>
#define QUARTERROUND(a,b,c,d) \ #define QUARTERROUND(a,b,c,d) \
a += b; d = std::rotl(d ^ a, 16); \ a += b; d = std::rotl(d ^ a, 16); \

View file

@ -9,7 +9,6 @@
#include <array> #include <array>
#include <cstddef> #include <cstddef>
#include <cstdlib>
#include <stdint.h> #include <stdint.h>
#include <utility> #include <utility>

View file

@ -4,7 +4,10 @@
#include <crypto/hex_base.h> #include <crypto/hex_base.h>
#include <span.h>
#include <array> #include <array>
#include <cassert>
#include <cstring> #include <cstring>
#include <string> #include <string>

View file

@ -4,8 +4,9 @@
#include <crypto/hkdf_sha256_32.h> #include <crypto/hkdf_sha256_32.h>
#include <assert.h> #include <crypto/hmac_sha256.h>
#include <string.h>
#include <cassert>
CHKDF_HMAC_SHA256_L32::CHKDF_HMAC_SHA256_L32(const unsigned char* ikm, size_t ikmlen, const std::string& salt) CHKDF_HMAC_SHA256_L32::CHKDF_HMAC_SHA256_L32(const unsigned char* ikm, size_t ikmlen, const std::string& salt)
{ {

View file

@ -5,10 +5,8 @@
#ifndef BITCOIN_CRYPTO_HKDF_SHA256_32_H #ifndef BITCOIN_CRYPTO_HKDF_SHA256_32_H
#define BITCOIN_CRYPTO_HKDF_SHA256_32_H #define BITCOIN_CRYPTO_HKDF_SHA256_32_H
#include <crypto/hmac_sha256.h>
#include <cstdlib> #include <cstdlib>
#include <stdint.h> #include <string>
/** A rfc5869 HKDF implementation with HMAC_SHA256 and fixed key output length of 32 bytes (L=32) */ /** A rfc5869 HKDF implementation with HMAC_SHA256 and fixed key output length of 32 bytes (L=32) */
class CHKDF_HMAC_SHA256_L32 class CHKDF_HMAC_SHA256_L32

View file

@ -4,6 +4,7 @@
#include <crypto/hmac_sha256.h> #include <crypto/hmac_sha256.h>
#include <crypto/sha256.h>
#include <string.h> #include <string.h>
CHMAC_SHA256::CHMAC_SHA256(const unsigned char* key, size_t keylen) CHMAC_SHA256::CHMAC_SHA256(const unsigned char* key, size_t keylen)

View file

@ -8,7 +8,6 @@
#include <crypto/sha256.h> #include <crypto/sha256.h>
#include <cstdlib> #include <cstdlib>
#include <stdint.h>
/** A hasher class for HMAC-SHA-256. */ /** A hasher class for HMAC-SHA-256. */
class CHMAC_SHA256 class CHMAC_SHA256

View file

@ -4,6 +4,7 @@
#include <crypto/hmac_sha512.h> #include <crypto/hmac_sha512.h>
#include <crypto/sha512.h>
#include <string.h> #include <string.h>
CHMAC_SHA512::CHMAC_SHA512(const unsigned char* key, size_t keylen) CHMAC_SHA512::CHMAC_SHA512(const unsigned char* key, size_t keylen)

View file

@ -8,7 +8,6 @@
#include <crypto/sha512.h> #include <crypto/sha512.h>
#include <cstdlib> #include <cstdlib>
#include <stdint.h>
/** A hasher class for HMAC-SHA-512. */ /** A hasher class for HMAC-SHA-512. */
class CHMAC_SHA512 class CHMAC_SHA512

View file

@ -7,9 +7,10 @@
#include <crypto/chacha20.h> #include <crypto/chacha20.h>
#include <crypto/common.h> #include <crypto/common.h>
#include <hash.h> #include <hash.h>
#include <span.h>
#include <uint256.h>
#include <cassert> #include <cassert>
#include <cstdio>
#include <limits> #include <limits>
namespace { namespace {

View file

@ -6,9 +6,13 @@
#define BITCOIN_CRYPTO_MUHASH_H #define BITCOIN_CRYPTO_MUHASH_H
#include <serialize.h> #include <serialize.h>
#include <uint256.h>
#include <stdint.h> #include <cstddef>
#include <cstdint>
class uint256;
template <typename C>
class Span;
class Num3072 class Num3072
{ {

View file

@ -5,8 +5,6 @@
#include <crypto/common.h> #include <crypto/common.h>
#include <crypto/poly1305.h> #include <crypto/poly1305.h>
#include <string.h>
namespace poly1305_donna { namespace poly1305_donna {
// Based on the public domain implementation by Andrew Moon // Based on the public domain implementation by Andrew Moon

View file

@ -8,7 +8,7 @@
#include <span.h> #include <span.h>
#include <cassert> #include <cassert>
#include <cstdlib> #include <cstddef>
#include <stdint.h> #include <stdint.h>
#define POLY1305_BLOCK_SIZE 16 #define POLY1305_BLOCK_SIZE 16

View file

@ -8,7 +8,8 @@
#if defined(ENABLE_SSE41) && defined(ENABLE_X86_SHANI) #if defined(ENABLE_SSE41) && defined(ENABLE_X86_SHANI)
#include <stdint.h> #include <cstddef>
#include <cstdint>
#include <immintrin.h> #include <immintrin.h>
#include <attributes.h> #include <attributes.h>

View file

@ -10,10 +10,9 @@
#include <span.h> #include <span.h>
#include <algorithm> #include <algorithm>
#include <array> // For std::begin and std::end.
#include <bit> #include <bit>
#include <cassert>
#include <stdint.h> #include <iterator>
void KeccakF(uint64_t (&st)[25]) void KeccakF(uint64_t (&st)[25])
{ {

View file

@ -5,10 +5,11 @@
#ifndef BITCOIN_CRYPTO_SHA3_H #ifndef BITCOIN_CRYPTO_SHA3_H
#define BITCOIN_CRYPTO_SHA3_H #define BITCOIN_CRYPTO_SHA3_H
#include <span.h> #include <cstdint>
#include <cstdlib> #include <cstdlib>
#include <stdint.h>
template <typename C>
class Span;
//! The Keccak-f[1600] transform. //! The Keccak-f[1600] transform.
void KeccakF(uint64_t (&st)[25]); void KeccakF(uint64_t (&st)[25]);

View file

@ -4,7 +4,11 @@
#include <crypto/siphash.h> #include <crypto/siphash.h>
#include <span.h>
#include <uint256.h>
#include <bit> #include <bit>
#include <cassert>
#define SIPROUND do { \ #define SIPROUND do { \
v0 += v1; v1 = std::rotl(v1, 13); v1 ^= v0; \ v0 += v1; v1 = std::rotl(v1, 13); v1 ^= v0; \

View file

@ -7,8 +7,9 @@
#include <stdint.h> #include <stdint.h>
#include <span.h> class uint256;
#include <uint256.h> template <typename C>
class Span;
/** SipHash-2-4 */ /** SipHash-2-4 */
class CSipHasher class CSipHasher

View file

@ -421,6 +421,7 @@ void BlockAssembler::addPackageTxs(int& nPackagesSelected, int& nDescendantsUpda
} }
++nPackagesSelected; ++nPackagesSelected;
pblocktemplate->m_package_feerates.emplace_back(packageFees, static_cast<int32_t>(packageSize));
// Update transactions that depend on each of these // Update transactions that depend on each of these
nDescendantsUpdated += UpdatePackagesForAdded(mempool, ancestors, mapModifiedTx); nDescendantsUpdated += UpdatePackagesForAdded(mempool, ancestors, mapModifiedTx);

View file

@ -10,6 +10,7 @@
#include <policy/policy.h> #include <policy/policy.h>
#include <primitives/block.h> #include <primitives/block.h>
#include <txmempool.h> #include <txmempool.h>
#include <util/feefrac.h>
#include <memory> #include <memory>
#include <optional> #include <optional>
@ -39,6 +40,9 @@ struct CBlockTemplate
std::vector<CAmount> vTxFees; std::vector<CAmount> vTxFees;
std::vector<int64_t> vTxSigOpsCost; std::vector<int64_t> vTxSigOpsCost;
std::vector<unsigned char> vchCoinbaseCommitment; std::vector<unsigned char> vchCoinbaseCommitment;
/* A vector of package fee rates, ordered by the sequence in which
* packages are selected for inclusion in the block template.*/
std::vector<FeeFrac> m_package_feerates;
}; };
// Container for tracking updates to ancestor feerate as we include (parent) // Container for tracking updates to ancestor feerate as we include (parent)

View file

@ -16,6 +16,7 @@
#include <txmempool.h> #include <txmempool.h>
#include <uint256.h> #include <uint256.h>
#include <util/check.h> #include <util/check.h>
#include <util/feefrac.h>
#include <util/strencodings.h> #include <util/strencodings.h>
#include <util/time.h> #include <util/time.h>
#include <util/translation.h> #include <util/translation.h>
@ -25,6 +26,7 @@
#include <test/util/setup_common.h> #include <test/util/setup_common.h>
#include <memory> #include <memory>
#include <vector>
#include <boost/test/unit_test.hpp> #include <boost/test/unit_test.hpp>
@ -123,19 +125,22 @@ void MinerTestingSetup::TestPackageSelection(const CScript& scriptPubKey, const
tx.vout[0].nValue = 5000000000LL - 1000; tx.vout[0].nValue = 5000000000LL - 1000;
// This tx has a low fee: 1000 satoshis // This tx has a low fee: 1000 satoshis
Txid hashParentTx = tx.GetHash(); // save this txid for later use Txid hashParentTx = tx.GetHash(); // save this txid for later use
AddToMempool(tx_mempool, entry.Fee(1000).Time(Now<NodeSeconds>()).SpendsCoinbase(true).FromTx(tx)); const auto parent_tx{entry.Fee(1000).Time(Now<NodeSeconds>()).SpendsCoinbase(true).FromTx(tx)};
AddToMempool(tx_mempool, parent_tx);
// This tx has a medium fee: 10000 satoshis // This tx has a medium fee: 10000 satoshis
tx.vin[0].prevout.hash = txFirst[1]->GetHash(); tx.vin[0].prevout.hash = txFirst[1]->GetHash();
tx.vout[0].nValue = 5000000000LL - 10000; tx.vout[0].nValue = 5000000000LL - 10000;
Txid hashMediumFeeTx = tx.GetHash(); Txid hashMediumFeeTx = tx.GetHash();
AddToMempool(tx_mempool, entry.Fee(10000).Time(Now<NodeSeconds>()).SpendsCoinbase(true).FromTx(tx)); const auto medium_fee_tx{entry.Fee(10000).Time(Now<NodeSeconds>()).SpendsCoinbase(true).FromTx(tx)};
AddToMempool(tx_mempool, medium_fee_tx);
// This tx has a high fee, but depends on the first transaction // This tx has a high fee, but depends on the first transaction
tx.vin[0].prevout.hash = hashParentTx; tx.vin[0].prevout.hash = hashParentTx;
tx.vout[0].nValue = 5000000000LL - 1000 - 50000; // 50k satoshi fee tx.vout[0].nValue = 5000000000LL - 1000 - 50000; // 50k satoshi fee
Txid hashHighFeeTx = tx.GetHash(); Txid hashHighFeeTx = tx.GetHash();
AddToMempool(tx_mempool, entry.Fee(50000).Time(Now<NodeSeconds>()).SpendsCoinbase(false).FromTx(tx)); const auto high_fee_tx{entry.Fee(50000).Time(Now<NodeSeconds>()).SpendsCoinbase(false).FromTx(tx)};
AddToMempool(tx_mempool, high_fee_tx);
std::unique_ptr<BlockTemplate> block_template = mining->createNewBlock(options); std::unique_ptr<BlockTemplate> block_template = mining->createNewBlock(options);
BOOST_REQUIRE(block_template); BOOST_REQUIRE(block_template);
@ -145,6 +150,21 @@ void MinerTestingSetup::TestPackageSelection(const CScript& scriptPubKey, const
BOOST_CHECK(block.vtx[2]->GetHash() == hashHighFeeTx); BOOST_CHECK(block.vtx[2]->GetHash() == hashHighFeeTx);
BOOST_CHECK(block.vtx[3]->GetHash() == hashMediumFeeTx); BOOST_CHECK(block.vtx[3]->GetHash() == hashMediumFeeTx);
// Test the inclusion of package feerates in the block template and ensure they are sequential.
const auto block_package_feerates = BlockAssembler{m_node.chainman->ActiveChainstate(), &tx_mempool, options}.CreateNewBlock()->m_package_feerates;
BOOST_CHECK(block_package_feerates.size() == 2);
// parent_tx and high_fee_tx are added to the block as a package.
const auto combined_txs_fee = parent_tx.GetFee() + high_fee_tx.GetFee();
const auto combined_txs_size = parent_tx.GetTxSize() + high_fee_tx.GetTxSize();
FeeFrac package_feefrac{combined_txs_fee, combined_txs_size};
// The package should be added first.
BOOST_CHECK(block_package_feerates[0] == package_feefrac);
// The medium_fee_tx should be added next.
FeeFrac medium_tx_feefrac{medium_fee_tx.GetFee(), medium_fee_tx.GetTxSize()};
BOOST_CHECK(block_package_feerates[1] == medium_tx_feefrac);
// Test that a package below the block min tx fee doesn't get included // Test that a package below the block min tx fee doesn't get included
tx.vin[0].prevout.hash = hashHighFeeTx; tx.vin[0].prevout.hash = hashHighFeeTx;
tx.vout[0].nValue = 5000000000LL - 1000 - 50000; // 0 fee tx.vout[0].nValue = 5000000000LL - 1000 - 50000; // 0 fee