/* * 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" #include "pool_block.h" #include #include namespace p2pool { class p2pool; class P2PServer; struct MinerShare { FORCEINLINE MinerShare() : m_weight(), m_wallet(nullptr) {} FORCEINLINE MinerShare(const difficulty_type& w, const Wallet* x) : m_weight(w), m_wallet(x) {} FORCEINLINE bool operator==(const MinerShare& s) const { return *m_wallet == *s.m_wallet; } difficulty_type m_weight; const Wallet* m_wallet; }; class SideChain : public nocopy_nomove { public: SideChain(p2pool* pool, NetworkType type, const char* pool_name = nullptr); ~SideChain(); void fill_sidechain_data(PoolBlock& block, std::vector& shares) const; [[nodiscard]] bool incoming_block_seen(const PoolBlock& block); void forget_incoming_block(const PoolBlock& block); void cleanup_incoming_blocks(); [[nodiscard]] bool add_external_block(PoolBlock& block, std::vector& missing_blocks); [[nodiscard]] bool add_block(const PoolBlock& block); void get_missing_blocks(unordered_set& missing_blocks) const; [[nodiscard]] PoolBlock* find_block(const hash& id) const; [[nodiscard]] PoolBlock* find_block_by_merkle_root(const root_hash& merkle_root) const; void watch_mainchain_block(const ChainMain& data, const hash& possible_merkle_root); [[nodiscard]] const PoolBlock* get_block_blob(const hash& id, std::vector& blob) const; [[nodiscard]] bool get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::vector& blob, uv_loop_t* loop) const; void print_status(bool obtain_sidechain_lock = true) const; [[nodiscard]] double get_reward_share(const Wallet& w) const; // Consensus ID can be used to spawn independent P2Pools with their own sidechains // It's never sent over the network to avoid revealing it to the possible man in the middle // Consensus ID can therefore be used as a password to create private P2Pools [[nodiscard]] const std::vector& consensus_id() const { return m_consensusId; } [[nodiscard]] const hash& consensus_hash() const { return m_consensusHash; } [[nodiscard]] uint64_t chain_window_size() const { return m_chainWindowSize; } [[nodiscard]] static NetworkType network_type() { return s_networkType; } [[nodiscard]] static uint64_t network_major_version(uint64_t height); [[nodiscard]] FORCEINLINE difficulty_type difficulty() const { ReadLock lock(m_curDifficultyLock); return m_curDifficulty; } [[nodiscard]] difficulty_type total_hashes() const; [[nodiscard]] uint64_t block_time() const { return m_targetBlockTime; } [[nodiscard]] uint64_t miner_count(); [[nodiscard]] uint64_t last_updated() const; [[nodiscard]] bool is_default() const; [[nodiscard]] bool is_mini() const; [[nodiscard]] uint64_t bottom_height(const PoolBlock* tip) const; [[nodiscard]] const PoolBlock* chainTip() const { return m_chainTip; } [[nodiscard]] bool precalcFinished() const { return m_precalcFinished.load(); } [[nodiscard]] bool p2pool_update_available() const; #ifdef P2POOL_UNIT_TESTS difficulty_type m_testMainChainDiff; const unordered_map& blocksById() const { return m_blocksById; } #endif [[nodiscard]] static bool split_reward(uint64_t reward, const std::vector& shares, std::vector& rewards); private: p2pool* m_pool; P2PServer* p2pServer() const; static NetworkType s_networkType; private: [[nodiscard]] bool get_shares(const PoolBlock* tip, std::vector& shares, uint64_t* bottom_height = nullptr, bool quiet = false) const; [[nodiscard]] bool get_difficulty(const PoolBlock* tip, std::vector& difficultyData, difficulty_type& curDifficulty) const; void verify_loop(PoolBlock* block); void verify(PoolBlock* block); void update_chain_tip(const PoolBlock* block); [[nodiscard]] PoolBlock* get_parent(const PoolBlock* block) const; // Checks if "candidate" has longer (higher difficulty) chain than "block" [[nodiscard]] bool is_longer_chain(const PoolBlock* block, const PoolBlock* candidate, bool& is_alternative) const; void update_depths(PoolBlock* block); void prune_old_blocks(); [[nodiscard]] bool load_config(const std::string& filename); [[nodiscard]] bool check_config() const; mutable uv_rwlock_t m_sidechainLock; std::atomic m_chainTip; std::map> m_blocksByHeight; unordered_map m_blocksById; unordered_map m_blocksByMerkleRoot; uv_mutex_t m_seenWalletsLock; unordered_map m_seenWallets; uint64_t m_seenWalletsLastPruneTime; // Used to quickly cut off multiple broadcasts of the same block by different peers. Only the first broadcast will be processed. uv_mutex_t m_incomingBlocksLock; unordered_map m_incomingBlocks; std::vector m_difficultyData; std::string m_poolName; std::string m_poolPassword; uint64_t m_targetBlockTime; difficulty_type m_minDifficulty; uint64_t m_chainWindowSize; uint64_t m_unclePenalty; std::vector m_consensusId; std::string m_consensusIdDisplayStr; mutable uv_rwlock_t m_curDifficultyLock; difficulty_type m_curDifficulty; uv_rwlock_t m_watchBlockLock; ChainMain m_watchBlock; hash m_watchBlockMerkleRoot; uv_cond_t m_precalcJobsCond; uv_mutex_t m_precalcJobsMutex; std::vector m_precalcJobs; std::vector m_precalcWorkers; unordered_set* m_uniquePrecalcInputs; std::atomic m_precalcFinished; #ifdef DEV_TEST_SYNC uint64_t m_firstPruneTime; #endif hash m_consensusHash; void launch_precalc(const PoolBlock* block); void precalc_worker(); void finish_precalc(); }; } // namespace p2pool namespace robin_hood { template<> struct hash { FORCEINLINE size_t operator()(const p2pool::MinerShare& value) const noexcept { return hash_bytes(value.m_wallet->spend_public_key().h, p2pool::HASH_SIZE); } }; } // namespace robin_hood