mirror of
https://github.com/SChernykh/p2pool.git
synced 2025-01-08 03:39:24 +00:00
fd0a7ea0c7
Some checks are pending
C/C++ CI / build-windows-msbuild (map[grpc:OFF os:2019 rx:OFF tls:ON upnp:OFF vs:Visual Studio 16 2019 vspath:C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise]) (push) Waiting to run
C/C++ CI / build-windows-msbuild (map[grpc:OFF os:2019 rx:OFF tls:ON upnp:ON vs:Visual Studio 16 2019 vspath:C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise]) (push) Waiting to run
C/C++ CI / build-windows-msbuild (map[grpc:OFF os:2019 rx:ON tls:ON upnp:ON vs:Visual Studio 16 2019 vspath:C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise]) (push) Waiting to run
C/C++ CI / build-windows-msbuild (map[grpc:ON os:2019 rx:ON tls:ON upnp:ON vs:Visual Studio 16 2019 vspath:C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise]) (push) Waiting to run
C/C++ CI / build-windows-msbuild (map[grpc:ON os:2022 rx:ON tls:ON upnp:ON vs:Visual Studio 17 2022 vspath:C:\Program Files\Microsoft Visual Studio\2022\Enterprise]) (push) Waiting to run
C/C++ CI / build-macos (push) Waiting to run
C/C++ CI / build-macos-aarch64 (push) Waiting to run
C/C++ CI / build-freebsd (map[architecture:x86-64 host:ubuntu-latest name:freebsd version:13.3]) (push) Waiting to run
C/C++ CI / build-openbsd (map[architecture:x86-64 host:ubuntu-latest name:openbsd version:7.4]) (push) Waiting to run
C/C++ CI / build-alpine-static (map[arch:aarch64 branch:latest-stable flags:-ffunction-sections -Wno-error=inline -mfix-cortex-a53-835769 -mfix-cortex-a53-843419]) (push) Waiting to run
C/C++ CI / build-alpine-static (map[arch:riscv64 branch:edge flags:-ffunction-sections -Wno-error=inline]) (push) Waiting to run
C/C++ CI / build-alpine-static (map[arch:x86_64 branch:latest-stable flags:-ffunction-sections -Wno-error=inline]) (push) Waiting to run
C/C++ CI / build-ubuntu (map[c:gcc-11 cpp:g++-11 flags: os:ubuntu-20.04]) (push) Waiting to run
C/C++ CI / build-ubuntu (map[c:gcc-12 cpp:g++-12 flags: os:ubuntu-22.04]) (push) Waiting to run
C/C++ CI / build-ubuntu (map[c:gcc-8 cpp:g++-8 flags: os:ubuntu-20.04]) (push) Waiting to run
C/C++ CI / build-ubuntu-static-libs (map[flags:-fuse-linker-plugin -ffunction-sections -Wno-error=inline]) (push) Waiting to run
C/C++ CI / build-ubuntu-aarch64 (map[flags:-fuse-linker-plugin -ffunction-sections -mfix-cortex-a53-835769 -mfix-cortex-a53-843419 os:ubuntu-20.04]) (push) Waiting to run
C/C++ CI / build-ubuntu-aarch64 (map[flags:-fuse-linker-plugin -ffunction-sections -mfix-cortex-a53-835769 -mfix-cortex-a53-843419 os:ubuntu-22.04]) (push) Waiting to run
C/C++ CI / build-windows-msys2 (map[c:clang cxx:clang++ flags:-fuse-ld=lld -Wno-unused-command-line-argument -Wno-nan-infinity-disabled]) (push) Waiting to run
C/C++ CI / build-windows-msys2 (map[c:gcc cxx:g++ flags:-ffunction-sections -Wno-error=maybe-uninitialized -Wno-error=attributes]) (push) Waiting to run
C/C++ CI / build-windows-msbuild (map[grpc:OFF os:2019 rx:OFF tls:OFF upnp:OFF vs:Visual Studio 16 2019 vspath:C:\Program Files (x86)\Microsoft Visual Studio\2019\Enterprise]) (push) Waiting to run
clang-tidy / clang-tidy (push) Waiting to run
CodeQL / Analyze (push) Waiting to run
cppcheck / cppcheck-ubuntu (push) Waiting to run
cppcheck / cppcheck-windows (push) Waiting to run
Microsoft C++ Code Analysis / Analyze (push) Waiting to run
source-snapshot / source-snapshot (push) Waiting to run
Sync test / sync-test-ubuntu-asan (push) Waiting to run
Sync test / sync-test-ubuntu-tsan (push) Waiting to run
Sync test / sync-test-ubuntu-msan (push) Waiting to run
Sync test / sync-test-ubuntu-ubsan (push) Waiting to run
Sync test / sync-test-macos (map[flags: os:macos-13]) (push) Waiting to run
Sync test / sync-test-macos (map[flags:-target arm64-apple-macos-11 os:macos-14]) (push) Waiting to run
Sync test / sync-test-windows-debug-asan (push) Waiting to run
Sync test / sync-test-windows-leaks (push) Waiting to run
0.5-1 seconds saved on verification after removing the repeated `get_shares` calls
183 lines
6.6 KiB
C++
183 lines
6.6 KiB
C++
/*
|
|
* This file is part of the Monero P2Pool <https://github.com/SChernykh/p2pool>
|
|
* Copyright (c) 2021-2024 SChernykh <https://github.com/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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "uv_util.h"
|
|
#include "pool_block.h"
|
|
#include <map>
|
|
#include <thread>
|
|
|
|
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<MinerShare>& 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<hash>& missing_blocks);
|
|
[[nodiscard]] bool add_block(const PoolBlock& block);
|
|
void get_missing_blocks(unordered_set<hash>& 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<uint8_t>& blob) const;
|
|
[[nodiscard]] bool get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::vector<uint8_t>& 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<uint8_t>& 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<hash, PoolBlock*>& blocksById() const { return m_blocksById; }
|
|
#endif
|
|
|
|
[[nodiscard]] static bool split_reward(uint64_t reward, const std::vector<MinerShare>& shares, std::vector<uint64_t>& rewards);
|
|
|
|
private:
|
|
p2pool* m_pool;
|
|
P2PServer* p2pServer() const;
|
|
static NetworkType s_networkType;
|
|
|
|
private:
|
|
[[nodiscard]] bool get_shares(const PoolBlock* tip, std::vector<MinerShare>& shares, uint64_t* bottom_height = nullptr, bool quiet = false) const;
|
|
[[nodiscard]] bool get_difficulty(const PoolBlock* tip, std::vector<DifficultyData>& 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<PoolBlock*> m_chainTip;
|
|
std::map<uint64_t, std::vector<PoolBlock*>> m_blocksByHeight;
|
|
unordered_map<hash, PoolBlock*> m_blocksById;
|
|
unordered_map<root_hash, PoolBlock*> m_blocksByMerkleRoot;
|
|
|
|
uv_mutex_t m_seenWalletsLock;
|
|
unordered_map<hash, uint64_t> 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<PoolBlock::full_id, uint64_t> m_incomingBlocks;
|
|
|
|
std::vector<DifficultyData> 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<uint8_t> 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<const PoolBlock*> m_precalcJobs;
|
|
std::vector<std::thread> m_precalcWorkers;
|
|
unordered_set<size_t>* m_uniquePrecalcInputs;
|
|
|
|
std::atomic<bool> 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<p2pool::MinerShare>
|
|
{
|
|
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
|