/* * This file is part of the Monero P2Pool * Copyright (c) 2021 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 #include #include namespace p2pool { class p2pool; struct DifficultyData; struct PoolBlock; class Wallet; struct MinerShare { FORCEINLINE MinerShare() : m_weight(0), m_wallet(nullptr) {} FORCEINLINE MinerShare(uint64_t w, Wallet* x) : m_weight(w), m_wallet(x) {} uint64_t m_weight; Wallet* m_wallet; }; class SideChain { public: SideChain(p2pool* pool, NetworkType type, const char* pool_name = nullptr); ~SideChain(); void fill_sidechain_data(PoolBlock& block, Wallet* w, const hash& txkeySec, std::vector& shares); bool block_seen(const PoolBlock& block); bool add_external_block(PoolBlock& block, std::vector& missing_blocks); void add_block(const PoolBlock& block); void get_missing_blocks(std::vector& missing_blocks); bool has_block(const hash& id); void watch_mainchain_block(const ChainMain& data, const hash& possible_id); bool get_block_blob(const hash& id, std::vector& blob); bool get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::vector& blob); void print_status(); // 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 const std::vector& consensus_id() const { return m_consensusId; } uint64_t chain_window_size() const { return m_chainWindowSize; } NetworkType network_type() const { return m_networkType; } const difficulty_type& difficulty() const { return m_curDifficulty; } difficulty_type total_hashes() const; uint64_t block_time() const { return m_targetBlockTime; } uint64_t miner_count(); time_t last_updated() const; bool is_default() const; static bool split_reward(uint64_t reward, const std::vector& shares, std::vector& rewards); private: p2pool* m_pool; NetworkType m_networkType; private: bool get_shares(PoolBlock* tip, std::vector& shares) const; bool get_difficulty(PoolBlock* tip, std::vector& difficultyData, difficulty_type& curDifficulty) const; void verify_loop(PoolBlock* block); void verify(PoolBlock* block); void update_chain_tip(PoolBlock* block); PoolBlock* get_parent(const PoolBlock* block); // Checks if "candidate" has longer (higher difficulty) chain than "block" bool is_longer_chain(const PoolBlock* block, const PoolBlock* candidate); void update_depths(PoolBlock* block); void prune_old_blocks(); bool load_config(const std::string& filename); bool check_config(); mutable uv_mutex_t m_sidechainLock; PoolBlock* m_chainTip; std::map> m_blocksByHeight; std::unordered_map m_blocksById; std::unordered_set m_seenBlocks; std::unordered_map m_seenWallets; 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; difficulty_type m_curDifficulty; ChainMain m_watchBlock; hash m_watchBlockSidechainId; }; } // namespace p2pool