2021-08-22 10:20:59 +00:00
|
|
|
/*
|
|
|
|
* This file is part of the Monero P2Pool <https://github.com/SChernykh/p2pool>
|
|
|
|
* Copyright (c) 2021 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 <map>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <unordered_set>
|
|
|
|
|
|
|
|
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:
|
2021-09-06 13:49:39 +00:00
|
|
|
SideChain(p2pool* pool, NetworkType type, const char* pool_name = nullptr);
|
2021-08-22 10:20:59 +00:00
|
|
|
~SideChain();
|
|
|
|
|
|
|
|
void fill_sidechain_data(PoolBlock& block, Wallet* w, const hash& txkeySec, std::vector<MinerShare>& shares);
|
|
|
|
|
|
|
|
bool block_seen(const PoolBlock& block);
|
|
|
|
bool add_external_block(PoolBlock& block, std::vector<hash>& missing_blocks);
|
|
|
|
void add_block(const PoolBlock& block);
|
|
|
|
void get_missing_blocks(std::vector<hash>& missing_blocks);
|
|
|
|
|
|
|
|
bool has_block(const hash& id);
|
2021-09-06 21:33:52 +00:00
|
|
|
void watch_mainchain_block(const ChainMain& data, const hash& possible_id);
|
|
|
|
|
2021-08-22 10:20:59 +00:00
|
|
|
bool get_block_blob(const hash& id, std::vector<uint8_t>& blob);
|
|
|
|
bool get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::vector<uint8_t>& 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<uint8_t>& consensus_id() const { return m_consensusId; }
|
|
|
|
uint64_t chain_window_size() const { return m_chainWindowSize; }
|
2021-08-27 09:25:25 +00:00
|
|
|
NetworkType network_type() const { return m_networkType; }
|
2021-08-30 17:45:37 +00:00
|
|
|
const difficulty_type& difficulty() const { return m_curDifficulty; }
|
2021-09-01 14:26:56 +00:00
|
|
|
difficulty_type total_hashes() const;
|
|
|
|
uint64_t block_time() const { return m_targetBlockTime; }
|
2021-09-01 18:48:03 +00:00
|
|
|
uint64_t miner_count();
|
2021-09-13 12:26:52 +00:00
|
|
|
time_t last_updated() const;
|
2021-09-07 19:30:52 +00:00
|
|
|
bool is_default() const;
|
2021-08-22 10:20:59 +00:00
|
|
|
|
|
|
|
static bool split_reward(uint64_t reward, const std::vector<MinerShare>& shares, std::vector<uint64_t>& rewards);
|
|
|
|
|
|
|
|
private:
|
|
|
|
p2pool* m_pool;
|
2021-08-27 09:25:25 +00:00
|
|
|
NetworkType m_networkType;
|
2021-08-22 10:20:59 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
bool get_shares(PoolBlock* tip, std::vector<MinerShare>& shares) const;
|
|
|
|
bool get_difficulty(PoolBlock* tip, std::vector<DifficultyData>& 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<uint64_t, std::vector<PoolBlock*>> m_blocksByHeight;
|
|
|
|
std::unordered_map<hash, PoolBlock*> m_blocksById;
|
|
|
|
std::unordered_set<hash> m_seenBlocks;
|
2021-09-01 18:48:03 +00:00
|
|
|
std::unordered_map<hash, time_t> m_seenWallets;
|
2021-08-22 10:20:59 +00:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
difficulty_type m_curDifficulty;
|
2021-09-06 21:33:52 +00:00
|
|
|
|
|
|
|
ChainMain m_watchBlock;
|
|
|
|
hash m_watchBlockSidechainId;
|
2021-08-22 10:20:59 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace p2pool
|