/* * This file is part of the Monero P2Pool * Copyright (c) 2021-2024 SChernykh * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, version 3. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #pragma once #include "uv_util.h" #define TEST_MEMPOOL_PICKING_ALGORITHM 0 namespace p2pool { class SideChain; class RandomX_Hasher_Base; class Mempool; class Wallet; struct PoolBlock; struct MinerShare; class BlockTemplate { public: explicit BlockTemplate(SideChain* sidechain, RandomX_Hasher_Base* hasher); ~BlockTemplate(); BlockTemplate(const BlockTemplate& b); BlockTemplate& operator=(const BlockTemplate& b); void update(const MinerData& data, const Mempool& mempool, const Wallet* miner_wallet); uint64_t last_updated() const { return m_lastUpdated.load(); } bool get_difficulties(const uint32_t template_id, uint64_t& height, uint64_t& sidechain_height, difficulty_type& mainchain_difficulty, difficulty_type& aux_diff, difficulty_type& sidechain_difficulty) const; uint32_t get_hashing_blob(const uint32_t template_id, uint32_t extra_nonce, uint8_t (&blob)[128], uint64_t& height, difficulty_type& difficulty, difficulty_type& aux_diff, difficulty_type& sidechain_difficulty, hash& seed_hash, size_t& nonce_offset) const; uint32_t get_hashing_blob(uint32_t extra_nonce, uint8_t (&blob)[128], uint64_t& height, uint64_t& sidechain_height, difficulty_type& difficulty, difficulty_type& aux_diff, difficulty_type& sidechain_difficulty, hash& seed_hash, size_t& nonce_offset, uint32_t& template_id) const; uint32_t get_hashing_blobs(uint32_t extra_nonce_start, uint32_t count, std::vector& blobs, uint64_t& height, difficulty_type& difficulty, difficulty_type& aux_diff, difficulty_type& sidechain_difficulty, hash& seed_hash, size_t& nonce_offset, uint32_t& template_id) const; std::vector get_aux_chains(const uint32_t template_id) const; bool get_aux_proof(const uint32_t template_id, uint32_t extra_nonce, const hash& h, std::vector& proof, uint32_t& path) const; std::vector get_block_template_blob(uint32_t template_id, uint32_t sidechain_extra_nonce, size_t& nonce_offset, size_t& extra_nonce_offset, size_t& merkle_root_offset, hash& merge_mining_root, const BlockTemplate** pThis) const; FORCEINLINE uint64_t height() const { return m_height; } FORCEINLINE difficulty_type difficulty() const { return m_difficulty; } bool submit_sidechain_block(uint32_t template_id, uint32_t nonce, uint32_t extra_nonce); FORCEINLINE const std::vector& shares() const { return m_shares; } FORCEINLINE uint64_t get_reward() const { return m_finalReward; } FORCEINLINE std::vector get_coinbase_merkle_proof() const { ReadLock lock(m_lock); return m_merkleTreeMainBranch; } #ifdef P2POOL_UNIT_TESTS FORCEINLINE const PoolBlock* pool_block_template() const { return m_poolBlockTemplate; } FORCEINLINE std::mt19937_64& rng() { return m_rng; } #endif private: SideChain* m_sidechain; RandomX_Hasher_Base* m_hasher; private: int create_miner_tx(const MinerData& data, const std::vector& shares, uint64_t max_reward_amounts_weight, bool dry_run); hash calc_sidechain_hash(uint32_t sidechain_extra_nonce) const; hash calc_miner_tx_hash(uint32_t extra_nonce) const; void calc_merkle_tree_main_branch(); uint32_t get_hashing_blob_nolock(uint32_t extra_nonce, uint8_t* blob) const; mutable uv_rwlock_t m_lock; uint32_t m_templateId; std::atomic m_lastUpdated; std::vector m_blockTemplateBlob; std::vector m_fullDataBlob; std::vector m_merkleTreeMainBranch; size_t m_blockHeaderSize; size_t m_minerTxOffsetInTemplate; size_t m_minerTxSize; size_t m_nonceOffset; size_t m_extraNonceOffsetInTemplate; size_t m_numTransactionHashes; hash m_prevId; std::atomic m_height; difficulty_type m_difficulty; difficulty_type m_auxDifficulty; hash m_seedHash; uint64_t m_timestamp; PoolBlock* m_poolBlockTemplate; BlockTemplate* m_oldTemplates[4] = {}; std::atomic m_finalReward; // Temp vectors, will be cleaned up after use and skipped in copy constructor/assignment operators std::vector m_minerTx; std::array m_minerTxKeccakState; size_t m_minerTxKeccakStateInputLength; std::vector m_sidechainHashBlob; std::array m_sidechainHashKeccakState; size_t m_sidechainHashInputLength; std::vector m_blockHeader; std::vector m_minerTxExtra; std::vector m_transactionHashes; unordered_set m_transactionHashesSet; std::vector m_rewards; std::vector m_mempoolTxs; std::vector m_mempoolTxsOrder; std::vector m_mempoolTxsOrder2; std::vector m_shares; std::mt19937_64 m_rng; void shuffle_tx_order(); #if TEST_MEMPOOL_PICKING_ALGORITHM void fill_optimal_knapsack(const MinerData& data, uint64_t base_reward, uint64_t miner_tx_weight, uint64_t& best_reward, uint64_t& final_fees, uint64_t& final_weight); std::vector m_knapsack; #endif void init_merge_mining_merkle_proof(); }; } // namespace p2pool