mirror of
https://github.com/SChernykh/p2pool.git
synced 2025-01-18 16:44:31 +00:00
7bad1e05a4
Shows progress in slices of PPNLS window. Shares move left to right as they age. A dot "." indicates no shares, 1-9 for that number in shares, for 9+ it shows a "+". Will not show shares or uncles lines unless you have one on window.
1655 lines
53 KiB
C++
1655 lines
53 KiB
C++
/*
|
|
* 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/>.
|
|
*/
|
|
|
|
#include "common.h"
|
|
#include "p2pool.h"
|
|
#include "side_chain.h"
|
|
#include "pool_block.h"
|
|
#include "wallet.h"
|
|
#include "block_template.h"
|
|
#include "randomx.h"
|
|
#include "dataset.hpp"
|
|
#include "configuration.h"
|
|
#include "intrin_portable.h"
|
|
#include "keccak.h"
|
|
#include "p2p_server.h"
|
|
#include "params.h"
|
|
#include "json_parsers.h"
|
|
#include <rapidjson/document.h>
|
|
#include <rapidjson/istreamwrapper.h>
|
|
#include <fstream>
|
|
#include <iterator>
|
|
#include <numeric>
|
|
#include <array>
|
|
|
|
// Only uncomment it to debug issues with uncle/orphan blocks
|
|
//#define DEBUG_BROADCAST_DELAY_MS 100
|
|
|
|
#ifdef DEBUG_BROADCAST_DELAY_MS
|
|
#include <thread>
|
|
#endif
|
|
|
|
static constexpr char log_category_prefix[] = "SideChain ";
|
|
|
|
constexpr uint64_t MIN_DIFFICULTY = 100000;
|
|
constexpr size_t UNCLE_BLOCK_DEPTH = 3;
|
|
|
|
static_assert(1 <= UNCLE_BLOCK_DEPTH && UNCLE_BLOCK_DEPTH <= 10, "Invalid UNCLE_BLOCK_DEPTH");
|
|
|
|
namespace p2pool {
|
|
|
|
SideChain::SideChain(p2pool* pool, NetworkType type, const char* pool_name)
|
|
: m_pool(pool)
|
|
, m_networkType(type)
|
|
, m_chainTip(nullptr)
|
|
, m_poolName(pool_name ? pool_name : "mainnet test 2")
|
|
, m_targetBlockTime(10)
|
|
, m_minDifficulty(MIN_DIFFICULTY, 0)
|
|
, m_chainWindowSize(2160)
|
|
, m_unclePenalty(20)
|
|
, m_curDifficulty(m_minDifficulty)
|
|
{
|
|
LOGINFO(1, log::LightCyan() << "network type = " << m_networkType);
|
|
|
|
if (m_pool && !load_config(m_pool->params().m_config)) {
|
|
panic();
|
|
}
|
|
|
|
if (!check_config()) {
|
|
panic();
|
|
}
|
|
|
|
uv_mutex_init_checked(&m_sidechainLock);
|
|
|
|
m_difficultyData.reserve(m_chainWindowSize);
|
|
|
|
LOGINFO(1, "generating consensus ID");
|
|
|
|
const randomx_flags flags = randomx_get_flags();
|
|
randomx_cache* cache = randomx_alloc_cache(flags | RANDOMX_FLAG_LARGE_PAGES);
|
|
if (!cache) {
|
|
LOGWARN(1, "couldn't allocate RandomX cache using large pages");
|
|
cache = randomx_alloc_cache(flags);
|
|
if (!cache) {
|
|
LOGERR(1, "couldn't allocate RandomX cache, aborting");
|
|
panic();
|
|
}
|
|
}
|
|
|
|
{
|
|
char consensus_str[log::Stream::BUF_SIZE + 1];
|
|
log::Stream s(consensus_str);
|
|
|
|
s << m_networkType << '\0'
|
|
<< m_poolName << '\0'
|
|
<< m_poolPassword << '\0'
|
|
<< m_targetBlockTime << '\0'
|
|
<< m_minDifficulty << '\0'
|
|
<< m_chainWindowSize << '\0'
|
|
<< m_unclePenalty << '\0';
|
|
|
|
randomx_init_cache(cache, consensus_str, s.m_pos);
|
|
}
|
|
|
|
// Intentionally not a power of 2
|
|
constexpr size_t scratchpad_size = 1009;
|
|
|
|
rx_vec_i128* scratchpad = reinterpret_cast<rx_vec_i128*>(cache->memory);
|
|
rx_vec_i128* scratchpad_end = scratchpad + scratchpad_size;
|
|
rx_vec_i128* scratchpad_ptr = scratchpad;
|
|
rx_vec_i128* cache_ptr = scratchpad_end;
|
|
|
|
for (uint64_t i = scratchpad_size, n = RANDOMX_ARGON_MEMORY * 1024 / sizeof(rx_vec_i128); i < n; ++i) {
|
|
*scratchpad_ptr = rx_xor_vec_i128(*scratchpad_ptr, *cache_ptr);
|
|
++cache_ptr;
|
|
++scratchpad_ptr;
|
|
if (scratchpad_ptr == scratchpad_end) {
|
|
scratchpad_ptr = scratchpad;
|
|
}
|
|
}
|
|
|
|
hash id;
|
|
keccak(reinterpret_cast<uint8_t*>(scratchpad), static_cast<int>(scratchpad_size * sizeof(rx_vec_i128)), id.h, HASH_SIZE);
|
|
|
|
randomx_release_cache(cache);
|
|
|
|
m_consensusId.assign(id.h, id.h + HASH_SIZE);
|
|
|
|
char buf[HASH_SIZE * 2 + 1];
|
|
log::Stream s(buf);
|
|
s << id << '\0';
|
|
|
|
// Hide most consensus ID bytes, we only want it on screen to show that we're on the right sidechain
|
|
memset(buf + 8, '*', HASH_SIZE * 2 - 16);
|
|
LOGINFO(1, "consensus ID = " << log::LightCyan() << buf);
|
|
}
|
|
|
|
SideChain::~SideChain()
|
|
{
|
|
uv_mutex_destroy(&m_sidechainLock);
|
|
for (auto& it : m_blocksById) {
|
|
delete it.second;
|
|
}
|
|
}
|
|
|
|
void SideChain::fill_sidechain_data(PoolBlock& block, Wallet* w, const hash& txkeySec, std::vector<MinerShare>& shares)
|
|
{
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
block.m_minerWallet = *w;
|
|
block.m_txkeySec = txkeySec;
|
|
block.m_uncles.clear();
|
|
|
|
if (!m_chainTip) {
|
|
block.m_parent = {};
|
|
block.m_sidechainHeight = 0;
|
|
block.m_difficulty = m_minDifficulty;
|
|
block.m_cumulativeDifficulty = m_minDifficulty;
|
|
|
|
get_shares(&block, shares);
|
|
return;
|
|
}
|
|
|
|
block.m_parent = m_chainTip->m_sidechainId;
|
|
block.m_sidechainHeight = m_chainTip->m_sidechainHeight + 1;
|
|
|
|
// Collect uncles from 3 previous block heights
|
|
|
|
// First get a list of already mined blocks at these heights
|
|
std::vector<hash> mined_blocks;
|
|
mined_blocks.reserve(UNCLE_BLOCK_DEPTH * 2 + 1);
|
|
|
|
PoolBlock* tmp = m_chainTip;
|
|
for (uint64_t i = 0, n = std::min<uint64_t>(UNCLE_BLOCK_DEPTH, m_chainTip->m_sidechainHeight + 1); tmp && (i < n); ++i) {
|
|
mined_blocks.push_back(tmp->m_sidechainId);
|
|
mined_blocks.insert(mined_blocks.end(), tmp->m_uncles.begin(), tmp->m_uncles.end());
|
|
tmp = get_parent(tmp);
|
|
}
|
|
|
|
for (uint64_t i = 0, n = std::min<uint64_t>(UNCLE_BLOCK_DEPTH, m_chainTip->m_sidechainHeight + 1); i < n; ++i) {
|
|
for (PoolBlock* uncle : m_blocksByHeight[m_chainTip->m_sidechainHeight - i]) {
|
|
// Only add verified and valid blocks
|
|
if (!uncle || !uncle->m_verified || uncle->m_invalid) {
|
|
continue;
|
|
}
|
|
|
|
// Only add it if it hasn't been mined already
|
|
if (std::find(mined_blocks.begin(), mined_blocks.end(), uncle->m_sidechainId) != mined_blocks.end()) {
|
|
continue;
|
|
}
|
|
|
|
// Only add it if it's on the same chain
|
|
bool same_chain = false;
|
|
do {
|
|
tmp = m_chainTip;
|
|
while (tmp->m_sidechainHeight > uncle->m_sidechainHeight) {
|
|
tmp = get_parent(tmp);
|
|
if (!tmp) {
|
|
break;
|
|
}
|
|
}
|
|
if (!tmp || (tmp->m_sidechainHeight < uncle->m_sidechainHeight)) {
|
|
break;
|
|
}
|
|
PoolBlock* tmp2 = uncle;
|
|
for (size_t j = 0; (j < UNCLE_BLOCK_DEPTH) && tmp && tmp2 && (tmp->m_sidechainHeight + UNCLE_BLOCK_DEPTH >= block.m_sidechainHeight); ++j) {
|
|
if (tmp->m_parent == tmp2->m_parent) {
|
|
same_chain = true;
|
|
break;
|
|
}
|
|
tmp = get_parent(tmp);
|
|
tmp2 = get_parent(tmp2);
|
|
}
|
|
} while (0);
|
|
|
|
if (same_chain) {
|
|
block.m_uncles.emplace_back(uncle->m_sidechainId);
|
|
LOGINFO(4, "block template at height " << block.m_sidechainHeight <<
|
|
": added " << uncle->m_sidechainId <<
|
|
" (height " << uncle->m_sidechainHeight <<
|
|
") as an uncle block, depth " << block.m_sidechainHeight - uncle->m_sidechainHeight);
|
|
}
|
|
else {
|
|
LOGINFO(4, "block template at height " << block.m_sidechainHeight <<
|
|
": uncle block " << uncle->m_sidechainId <<
|
|
" (height " << uncle->m_sidechainHeight <<
|
|
") is not on the same chain, depth " << block.m_sidechainHeight - uncle->m_sidechainHeight);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Sort uncles and remove duplicates
|
|
if (block.m_uncles.size() > 1) {
|
|
std::sort(block.m_uncles.begin(), block.m_uncles.end());
|
|
block.m_uncles.erase(std::unique(block.m_uncles.begin(), block.m_uncles.end()), block.m_uncles.end());
|
|
}
|
|
|
|
block.m_difficulty = m_curDifficulty;
|
|
block.m_cumulativeDifficulty = m_chainTip->m_cumulativeDifficulty + block.m_difficulty;
|
|
|
|
for (const hash& uncle_id : block.m_uncles) {
|
|
auto it = m_blocksById.find(uncle_id);
|
|
if (it == m_blocksById.end()) {
|
|
LOGERR(1, "block template has an unknown uncle block " << uncle_id << ". Fix the code!");
|
|
continue;
|
|
}
|
|
block.m_cumulativeDifficulty += it->second->m_difficulty;
|
|
}
|
|
|
|
get_shares(&block, shares);
|
|
}
|
|
|
|
bool SideChain::get_shares(PoolBlock* tip, std::vector<MinerShare>& shares) const
|
|
{
|
|
shares.clear();
|
|
shares.reserve(m_chainWindowSize * 2);
|
|
|
|
// Collect shares from each block in the PPLNS window, starting from the "tip"
|
|
|
|
uint64_t block_depth = 0;
|
|
PoolBlock* cur = tip;
|
|
do {
|
|
MinerShare cur_share{ cur->m_difficulty.lo, &cur->m_minerWallet };
|
|
|
|
for (const hash& uncle_id : cur->m_uncles) {
|
|
auto it = m_blocksById.find(uncle_id);
|
|
if (it == m_blocksById.end()) {
|
|
LOGWARN(4, "get_shares: can't find uncle block at height = " << cur->m_sidechainHeight << ", id = " << uncle_id);
|
|
LOGWARN(4, "get_shares: can't calculate shares for block at height = " << tip->m_sidechainHeight << ", id = " << tip->m_sidechainId << ", mainchain height = " << tip->m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
PoolBlock* uncle = it->second;
|
|
|
|
// Skip uncles which are already out of PPLNS window
|
|
if (tip->m_sidechainHeight - uncle->m_sidechainHeight >= m_chainWindowSize) {
|
|
continue;
|
|
}
|
|
|
|
// Take some % of uncle's weight into this share
|
|
uint64_t product[2];
|
|
product[0] = umul128(uncle->m_difficulty.lo, m_unclePenalty, &product[1]);
|
|
|
|
uint64_t rem;
|
|
const uint64_t uncle_penalty = udiv128(product[1], product[0], 100, &rem);
|
|
|
|
cur_share.m_weight += uncle_penalty;
|
|
shares.emplace_back(uncle->m_difficulty.lo - uncle_penalty, &uncle->m_minerWallet);
|
|
}
|
|
|
|
shares.push_back(cur_share);
|
|
|
|
++block_depth;
|
|
if (block_depth >= m_chainWindowSize) {
|
|
break;
|
|
}
|
|
|
|
// Reached the genesis block so we're done
|
|
if (cur->m_sidechainHeight == 0) {
|
|
break;
|
|
}
|
|
|
|
auto it = m_blocksById.find(cur->m_parent);
|
|
if (it == m_blocksById.end()) {
|
|
LOGWARN(4, "get_shares: can't find parent block at height = " << cur->m_sidechainHeight - 1 << ", id = " << cur->m_parent);
|
|
LOGWARN(4, "get_shares: can't calculate shares for block at height = " << tip->m_sidechainHeight << ", id = " << tip->m_sidechainId << ", mainchain height = " << tip->m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
cur = it->second;
|
|
} while (block_depth < m_chainWindowSize);
|
|
|
|
// Combine shares with the same wallet addresses
|
|
std::sort(shares.begin(), shares.end(), [](const auto& a, const auto& b) { return *a.m_wallet < *b.m_wallet; });
|
|
|
|
size_t k = 0;
|
|
for (size_t i = 1, n = shares.size(); i < n; ++i)
|
|
{
|
|
if (*shares[i].m_wallet == *shares[k].m_wallet) {
|
|
shares[k].m_weight += shares[i].m_weight;
|
|
}
|
|
else {
|
|
++k;
|
|
shares[k].m_weight = shares[i].m_weight;
|
|
shares[k].m_wallet = shares[i].m_wallet;
|
|
}
|
|
}
|
|
|
|
shares.resize(k + 1);
|
|
|
|
LOGINFO(6, "get_shares: " << k + 1 << " unique wallets in PPLNS window");
|
|
return true;
|
|
}
|
|
|
|
bool SideChain::block_seen(const PoolBlock& block)
|
|
{
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
// Check if it's some old block
|
|
if (m_chainTip && m_chainTip->m_sidechainHeight > block.m_sidechainHeight + m_chainWindowSize * 2 &&
|
|
block.m_cumulativeDifficulty < m_chainTip->m_cumulativeDifficulty) {
|
|
return true;
|
|
}
|
|
|
|
// Check if it was received before
|
|
return !m_seenBlocks.insert(block.m_sidechainId).second;
|
|
}
|
|
|
|
extern const char* BLOCK_FOUND;
|
|
|
|
bool SideChain::add_external_block(PoolBlock& block, std::vector<hash>& missing_blocks)
|
|
{
|
|
if (block.m_difficulty < m_minDifficulty) {
|
|
LOGWARN(3, "add_external_block: block has invalid difficulty " << block.m_difficulty << ", expected >= " << m_minDifficulty);
|
|
return false;
|
|
}
|
|
|
|
difficulty_type min_accepted_diff;
|
|
{
|
|
MutexLock lock(m_sidechainLock);
|
|
if (m_blocksById.find(block.m_sidechainId) != m_blocksById.end()) {
|
|
LOGINFO(4, "add_external_block: block " << block.m_sidechainId << " is already added");
|
|
return true;
|
|
}
|
|
|
|
// Find the minimum difficulty in the current PPLNS window
|
|
min_accepted_diff = m_curDifficulty;
|
|
for (PoolBlock* tmp = m_chainTip; tmp && (tmp->m_sidechainHeight + m_chainWindowSize > m_chainTip->m_sidechainHeight); tmp = get_parent(tmp)) {
|
|
if (tmp->m_difficulty < min_accepted_diff) {
|
|
min_accepted_diff = tmp->m_difficulty;
|
|
}
|
|
}
|
|
}
|
|
|
|
LOGINFO(4, "add_external_block: height = " << block.m_sidechainHeight << ", id = " << block.m_sidechainId << ", mainchain height = " << block.m_txinGenHeight);
|
|
|
|
// Reduce it by 50% to account for alternative chains. This is mainly an anti-spam measure, not an actual verification step
|
|
min_accepted_diff.lo = (min_accepted_diff.lo >> 1) | (min_accepted_diff.hi << 63);
|
|
min_accepted_diff.hi >>= 1;
|
|
|
|
if (block.m_difficulty < min_accepted_diff) {
|
|
LOGWARN(4, "add_external_block: block has too low difficulty " << block.m_difficulty << ", expected >= " << min_accepted_diff << ". Ignoring it.");
|
|
return true;
|
|
}
|
|
|
|
// This check is not always possible to perform because of mainchain reorgs
|
|
ChainMain data;
|
|
if (m_pool->chainmain_get_by_hash(block.m_prevId, data)) {
|
|
if (data.height + 1 != block.m_txinGenHeight) {
|
|
LOGWARN(3, "add_external_block: wrong mainchain height " << block.m_txinGenHeight << ", expected " << data.height + 1);
|
|
return false;
|
|
}
|
|
}
|
|
else {
|
|
LOGWARN(3, "add_external_block: block is built on top of an unknown mainchain block " << block.m_prevId << ", mainchain reorg might've happened");
|
|
}
|
|
|
|
hash seed;
|
|
if (!m_pool->get_seed(block.m_txinGenHeight, seed)) {
|
|
LOGWARN(3, "add_external_block: couldn't get seed hash for mainchain height " << block.m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
hash pow_hash;
|
|
if (!block.get_pow_hash(m_pool->hasher(), seed, pow_hash)) {
|
|
LOGWARN(3, "add_external_block: couldn't get PoW hash for height = " << block.m_sidechainHeight << ", mainchain height " << block.m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
// Check if it has the correct parent and difficulty to go right to monerod for checking
|
|
const MinerData& miner_data = m_pool->miner_data();
|
|
if ((block.m_prevId == miner_data.prev_id) && miner_data.difficulty.check_pow(pow_hash)) {
|
|
LOGINFO(0, log::LightGreen() << "add_external_block: block " << block.m_sidechainId << " has enough PoW for Monero network, submitting it");
|
|
m_pool->submit_block_async(block.m_mainChainData);
|
|
}
|
|
|
|
if (!block.m_difficulty.check_pow(pow_hash)) {
|
|
LOGWARN(3, "add_external_block: not enough PoW for height = " << block.m_sidechainHeight << ", mainchain height " << block.m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
bool block_found = false;
|
|
|
|
missing_blocks.clear();
|
|
{
|
|
MutexLock lock(m_sidechainLock);
|
|
if (!block.m_parent.empty() && (m_blocksById.find(block.m_parent) == m_blocksById.end())) {
|
|
missing_blocks.push_back(block.m_parent);
|
|
}
|
|
|
|
for (const hash& h : block.m_uncles) {
|
|
if (!h.empty() && (m_blocksById.find(h) == m_blocksById.end())) {
|
|
missing_blocks.push_back(h);
|
|
}
|
|
}
|
|
|
|
if (block.m_sidechainId == m_watchBlockSidechainId) {
|
|
LOGINFO(0, log::LightGreen() << "BLOCK FOUND: main chain block at height " << m_watchBlock.height << " was mined by this p2pool" << BLOCK_FOUND);
|
|
m_watchBlockSidechainId = {};
|
|
data = m_watchBlock;
|
|
block_found = true;
|
|
}
|
|
}
|
|
|
|
if (block_found) {
|
|
m_pool->api_update_block_found(&data);
|
|
}
|
|
|
|
add_block(block);
|
|
return true;
|
|
}
|
|
|
|
void SideChain::add_block(const PoolBlock& block)
|
|
{
|
|
LOGINFO(3, "add_block: height = " << block.m_sidechainHeight <<
|
|
", id = " << block.m_sidechainId <<
|
|
", mainchain height = " << block.m_txinGenHeight <<
|
|
", verified = " << (block.m_verified ? 1 : 0)
|
|
);
|
|
|
|
// Save it for faster syncing on the next p2pool start
|
|
if (m_pool->p2p_server()) {
|
|
m_pool->p2p_server()->store_in_cache(block);
|
|
}
|
|
|
|
PoolBlock* new_block = new PoolBlock(block);
|
|
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
auto result = m_blocksById.insert({ new_block->m_sidechainId, new_block });
|
|
if (!result.second) {
|
|
LOGWARN(3, "add_block: trying to add the same block twice, id = "
|
|
<< new_block->m_sidechainId << ", sidechain height = "
|
|
<< new_block->m_sidechainHeight << ", height = "
|
|
<< new_block->m_txinGenHeight);
|
|
|
|
delete new_block;
|
|
return;
|
|
}
|
|
|
|
m_blocksByHeight[new_block->m_sidechainHeight].push_back(new_block);
|
|
|
|
update_depths(new_block);
|
|
|
|
if (new_block->m_verified) {
|
|
if (!new_block->m_invalid) {
|
|
update_chain_tip(new_block);
|
|
}
|
|
}
|
|
else {
|
|
verify_loop(new_block);
|
|
}
|
|
|
|
m_seenWallets[new_block->m_minerWallet.spend_public_key()] = new_block->m_localTimestamp;
|
|
}
|
|
|
|
bool SideChain::has_block(const hash& id)
|
|
{
|
|
MutexLock lock(m_sidechainLock);
|
|
return m_blocksById.find(id) != m_blocksById.end();
|
|
}
|
|
|
|
void SideChain::watch_mainchain_block(const ChainMain& data, const hash& possible_id)
|
|
{
|
|
MutexLock lock(m_sidechainLock);
|
|
m_watchBlock = data;
|
|
m_watchBlockSidechainId = possible_id;
|
|
}
|
|
|
|
bool SideChain::get_block_blob(const hash& id, std::vector<uint8_t>& blob)
|
|
{
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
PoolBlock* block = nullptr;
|
|
|
|
// Empty hash means we return current sidechain tip
|
|
if (id == hash()) {
|
|
block = m_chainTip;
|
|
}
|
|
else {
|
|
auto it = m_blocksById.find(id);
|
|
if (it != m_blocksById.end()) {
|
|
block = it->second;
|
|
}
|
|
}
|
|
|
|
if (!block) {
|
|
return false;
|
|
}
|
|
|
|
blob.reserve(block->m_mainChainData.size() + block->m_sideChainData.size());
|
|
|
|
blob = block->m_mainChainData;
|
|
blob.insert(blob.end(), block->m_sideChainData.begin(), block->m_sideChainData.end());
|
|
return true;
|
|
}
|
|
|
|
bool SideChain::get_outputs_blob(PoolBlock* block, uint64_t total_reward, std::vector<uint8_t>& blob)
|
|
{
|
|
std::vector<MinerShare> shares;
|
|
std::vector<uint64_t> rewards;
|
|
|
|
shares.reserve(m_chainWindowSize * 2);
|
|
rewards.reserve(m_chainWindowSize * 2);
|
|
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
if (!get_shares(block, shares) || !split_reward(total_reward, shares, rewards) || (rewards.size() != shares.size())) {
|
|
return false;
|
|
}
|
|
|
|
const size_t n = shares.size();
|
|
|
|
blob.clear();
|
|
blob.reserve(n * 38 + 64);
|
|
|
|
writeVarint(n, blob);
|
|
|
|
block->m_outputs.clear();
|
|
block->m_outputs.reserve(n);
|
|
|
|
hash eph_public_key;
|
|
for (size_t i = 0; i < n; ++i) {
|
|
writeVarint(rewards[i], blob);
|
|
|
|
blob.emplace_back(TXOUT_TO_KEY);
|
|
|
|
if (!shares[i].m_wallet->get_eph_public_key(block->m_txkeySec, i, eph_public_key)) {
|
|
LOGWARN(6, "get_eph_public_key failed at index " << i);
|
|
}
|
|
blob.insert(blob.end(), eph_public_key.h, eph_public_key.h + HASH_SIZE);
|
|
|
|
block->m_outputs.emplace_back(rewards[i], eph_public_key);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void SideChain::print_status()
|
|
{
|
|
std::vector<hash> blocks_in_window;
|
|
blocks_in_window.reserve(m_chainWindowSize * 9 / 8);
|
|
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
uint64_t rem;
|
|
uint64_t pool_hashrate = udiv128(m_curDifficulty.hi, m_curDifficulty.lo, m_targetBlockTime, &rem);
|
|
|
|
const difficulty_type& network_diff = m_pool->miner_data().difficulty;
|
|
uint64_t network_hashrate = udiv128(network_diff.hi, network_diff.lo, 120, &rem);
|
|
|
|
uint64_t block_depth = 0;
|
|
PoolBlock* cur = m_chainTip;
|
|
const uint64_t tip_height = m_chainTip ? m_chainTip->m_sidechainHeight : 0;
|
|
|
|
uint32_t total_blocks_in_window = 0;
|
|
uint32_t total_uncles_in_window = 0;
|
|
|
|
// each dot corresponds to m_chainWindowSize / 30 shares, with current values, 2160 / 30 = 72
|
|
std::array<uint32_t, 30> our_blocks_in_window{};
|
|
std::array<uint32_t, 30> our_uncles_in_window{};
|
|
|
|
while (cur) {
|
|
blocks_in_window.emplace_back(cur->m_sidechainId);
|
|
++total_blocks_in_window;
|
|
|
|
if (cur->m_minerWallet == m_pool->params().m_wallet) {
|
|
// this produces an integer division with quotient rounded up, avoids non-whole divisions from overflowing on total_blocks_in_window
|
|
const size_t window_index = (total_blocks_in_window - 1) / ((m_chainWindowSize + our_blocks_in_window.size() - 1) / our_blocks_in_window.size());
|
|
our_blocks_in_window[std::min(window_index, our_blocks_in_window.size() - 1)]++; // clamp window_index, even if total_blocks_in_window is not larger than m_chainWindowSize
|
|
}
|
|
|
|
++block_depth;
|
|
if (block_depth >= m_chainWindowSize) {
|
|
break;
|
|
}
|
|
|
|
for (const hash& uncle_id : cur->m_uncles) {
|
|
blocks_in_window.emplace_back(uncle_id);
|
|
auto it = m_blocksById.find(uncle_id);
|
|
if (it != m_blocksById.end()) {
|
|
PoolBlock* uncle = it->second;
|
|
if (tip_height - uncle->m_sidechainHeight < m_chainWindowSize) {
|
|
++total_uncles_in_window;
|
|
if (uncle->m_minerWallet == m_pool->params().m_wallet) {
|
|
// this produces an integer division with quotient rounded up, avoids non-whole divisions from overflowing on total_blocks_in_window
|
|
const size_t window_index = (total_blocks_in_window - 1) / ((m_chainWindowSize + our_uncles_in_window.size() - 1) / our_uncles_in_window.size());
|
|
our_uncles_in_window[std::min(window_index, our_uncles_in_window.size() - 1)]++; // clamp window_index, even if total_blocks_in_window is not larger than m_chainWindowSize
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
cur = get_parent(cur);
|
|
}
|
|
|
|
uint64_t total_orphans = 0;
|
|
uint64_t our_orphans = 0;
|
|
|
|
uint64_t your_reward = 0;
|
|
uint64_t total_reward = 0;
|
|
|
|
if (m_chainTip) {
|
|
std::sort(blocks_in_window.begin(), blocks_in_window.end());
|
|
for (uint64_t i = 0; (i < m_chainWindowSize) && (i <= tip_height); ++i) {
|
|
for (PoolBlock* block : m_blocksByHeight[tip_height - i]) {
|
|
if (!std::binary_search(blocks_in_window.begin(), blocks_in_window.end(), block->m_sidechainId)) {
|
|
LOGINFO(4, "orphan block at height " << log::Gray() << block->m_sidechainHeight << log::NoColor() << ": " << log::Gray() << block->m_sidechainId);
|
|
++total_orphans;
|
|
if (block->m_minerWallet == m_pool->params().m_wallet) {
|
|
++our_orphans;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Wallet w = m_pool->params().m_wallet;
|
|
const std::vector<PoolBlock::TxOutput>& outs = m_chainTip->m_outputs;
|
|
|
|
hash eph_public_key;
|
|
for (size_t i = 0, n = outs.size(); i < n; ++i) {
|
|
if (w.get_eph_public_key(m_chainTip->m_txkeySec, i, eph_public_key) && (outs[i].m_ephPublicKey == eph_public_key)) {
|
|
your_reward = outs[i].m_reward;
|
|
}
|
|
total_reward += outs[i].m_reward;
|
|
}
|
|
}
|
|
|
|
uint64_t product[2];
|
|
product[0] = umul128(pool_hashrate, your_reward, &product[1]);
|
|
const uint64_t hashrate_est = total_reward ? udiv128(product[1], product[0], total_reward, &rem) : 0;
|
|
const double block_share = total_reward ? ((static_cast<double>(your_reward) * 100.0) / static_cast<double>(total_reward)) : 0.0;
|
|
|
|
uint32_t our_blocks_in_window_total = std::accumulate(our_blocks_in_window.begin(), our_blocks_in_window.end(), decltype(our_blocks_in_window)::value_type(0));
|
|
uint32_t our_uncles_in_window_total = std::accumulate(our_uncles_in_window.begin(), our_uncles_in_window.end(), decltype(our_uncles_in_window)::value_type(0));
|
|
|
|
std::string our_blocks_in_window_chart;
|
|
our_blocks_in_window_chart.reserve(our_blocks_in_window.size());
|
|
for(const auto& p : our_blocks_in_window){
|
|
our_blocks_in_window_chart += (p > 0 ? (p > 9 ? "+" : std::to_string(p)) : ".");
|
|
}
|
|
|
|
std::string our_uncles_in_window_chart;
|
|
our_uncles_in_window_chart.reserve(our_uncles_in_window.size());
|
|
for(const auto& p : our_uncles_in_window){
|
|
our_uncles_in_window_chart += (p > 0 ? (p > 9 ? "+" : std::to_string(p)) : ".");
|
|
}
|
|
|
|
LOGINFO(0, "status" <<
|
|
"\nMain chain height = " << m_pool->block_template().height() <<
|
|
"\nMain chain hashrate = " << log::Hashrate(network_hashrate) <<
|
|
"\nSide chain height = " << tip_height + 1 <<
|
|
"\nSide chain hashrate = " << log::Hashrate(pool_hashrate) <<
|
|
(hashrate_est ? "\nYour hashrate (pool-side) = " : "") << (hashrate_est ? log::Hashrate(hashrate_est) : log::Hashrate()) <<
|
|
"\nPPLNS window = " << total_blocks_in_window << " blocks (+" << total_uncles_in_window << " uncles, " << total_orphans << " orphans)" <<
|
|
"\nYour shares = " << our_blocks_in_window_total << " blocks (+" << our_uncles_in_window_total << " uncles, " << our_orphans << " orphans)" <<
|
|
(our_blocks_in_window_total > 0 ? "\nYour shares position = " : "") << (our_blocks_in_window_total > 0 ? "[" + our_blocks_in_window_chart + "]" : "") <<
|
|
(our_uncles_in_window_total > 0 ? "\nYour uncles position = " : "") << (our_uncles_in_window_total > 0 ? "[" + our_uncles_in_window_chart + "]" : "") <<
|
|
"\nBlock reward share = " << block_share << "% (" << log::XMRAmount(your_reward) << ')'
|
|
);
|
|
}
|
|
|
|
difficulty_type SideChain::total_hashes() const
|
|
{
|
|
return m_chainTip ? m_chainTip->m_cumulativeDifficulty : difficulty_type();
|
|
}
|
|
|
|
uint64_t SideChain::miner_count()
|
|
{
|
|
const time_t cur_time = time(nullptr);
|
|
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
// Delete wallets that weren't seen for more than 24 hours and return how many remain
|
|
for (auto it = m_seenWallets.begin(); it != m_seenWallets.end();) {
|
|
if (it->second + 24 * 60 * 60 <= cur_time) {
|
|
it = m_seenWallets.erase(it);
|
|
}
|
|
else {
|
|
++it;
|
|
}
|
|
}
|
|
|
|
return m_seenWallets.size();
|
|
}
|
|
|
|
bool SideChain::is_default() const
|
|
{
|
|
constexpr uint8_t default_consensus_id[HASH_SIZE] = {
|
|
34,175,126,231,181,11,104,146,227,153,218,107,44,108,68,39,178,81,4,212,169,4,142,0,177,110,157,240,68,7,249,24
|
|
};
|
|
|
|
return (memcmp(m_consensusId.data(), default_consensus_id, HASH_SIZE) == 0);
|
|
}
|
|
|
|
bool SideChain::split_reward(uint64_t reward, const std::vector<MinerShare>& shares, std::vector<uint64_t>& rewards)
|
|
{
|
|
const size_t num_shares = shares.size();
|
|
|
|
const uint64_t total_weight = std::accumulate(shares.begin(), shares.end(), 0ULL, [](uint64_t a, const MinerShare& b) { return a + b.m_weight; });
|
|
|
|
if (total_weight == 0) {
|
|
LOGERR(1, "total_weight is 0. Check the code!");
|
|
return false;
|
|
}
|
|
|
|
rewards.clear();
|
|
rewards.reserve(num_shares);
|
|
|
|
// Each miner gets a proportional fraction of the block reward
|
|
uint64_t w = 0;
|
|
uint64_t reward_given = 0;
|
|
for (uint64_t i = 0; i < num_shares; ++i) {
|
|
w += shares[i].m_weight;
|
|
|
|
uint64_t hi;
|
|
const uint64_t lo = umul128(w, reward, &hi);
|
|
|
|
uint64_t rem;
|
|
const uint64_t next_value = udiv128(hi, lo, total_weight, &rem);
|
|
rewards.emplace_back(next_value - reward_given);
|
|
reward_given = next_value;
|
|
}
|
|
|
|
// Double check that we gave out the exact amount
|
|
if (std::accumulate(rewards.begin(), rewards.end(), 0ULL) != reward) {
|
|
LOGERR(1, "miners got incorrect reward. This should never happen because math says so. Check the code!");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SideChain::get_difficulty(PoolBlock* tip, std::vector<DifficultyData>& difficultyData, difficulty_type& curDifficulty) const
|
|
{
|
|
difficultyData.clear();
|
|
|
|
PoolBlock* cur = tip;
|
|
uint64_t oldest_timestamp = std::numeric_limits<uint64_t>::max();
|
|
|
|
uint64_t block_depth = 0;
|
|
do {
|
|
oldest_timestamp = std::min(oldest_timestamp, cur->m_timestamp);
|
|
difficultyData.emplace_back(cur->m_timestamp, cur->m_cumulativeDifficulty);
|
|
|
|
for (const hash& uncle_id : cur->m_uncles) {
|
|
auto it = m_blocksById.find(uncle_id);
|
|
if (it == m_blocksById.end()) {
|
|
LOGWARN(4, "get_difficulty: can't find uncle block at height = " << cur->m_sidechainHeight << ", id = " << uncle_id);
|
|
LOGWARN(4, "get_difficulty: can't calculate diff for block at height = " << tip->m_sidechainHeight << ", id = " << tip->m_sidechainId << ", mainchain height = " << tip->m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
const PoolBlock* uncle = it->second;
|
|
if (tip->m_sidechainHeight - uncle->m_sidechainHeight < m_chainWindowSize) {
|
|
oldest_timestamp = std::min(oldest_timestamp, uncle->m_timestamp);
|
|
difficultyData.emplace_back(uncle->m_timestamp, uncle->m_cumulativeDifficulty);
|
|
}
|
|
}
|
|
|
|
++block_depth;
|
|
if (block_depth >= m_chainWindowSize) {
|
|
break;
|
|
}
|
|
|
|
// Reached the genesis block so we're done
|
|
if (cur->m_sidechainHeight == 0) {
|
|
break;
|
|
}
|
|
|
|
auto it = m_blocksById.find(cur->m_parent);
|
|
if (it == m_blocksById.end()) {
|
|
LOGWARN(4, "get_difficulty: can't find parent block at height = " << cur->m_sidechainHeight - 1 << ", id = " << cur->m_parent);
|
|
LOGWARN(4, "get_difficulty: can't calculate diff for block at height = " << tip->m_sidechainHeight << ", id = " << tip->m_sidechainId << ", mainchain height = " << tip->m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
cur = it->second;
|
|
} while (true);
|
|
|
|
// Discard 10% oldest and 10% newest (by timestamp) blocks
|
|
std::vector<uint32_t> tmpTimestamps;
|
|
tmpTimestamps.reserve(difficultyData.size());
|
|
|
|
std::transform(difficultyData.begin(), difficultyData.end(), std::back_inserter(tmpTimestamps),
|
|
[oldest_timestamp](const DifficultyData& d)
|
|
{
|
|
return static_cast<uint32_t>(d.m_timestamp - oldest_timestamp);
|
|
});
|
|
|
|
const uint64_t cut_size = (difficultyData.size() + 9) / 10;
|
|
const uint64_t index1 = cut_size - 1;
|
|
const uint64_t index2 = difficultyData.size() - cut_size;
|
|
|
|
std::nth_element(tmpTimestamps.begin(), tmpTimestamps.begin() + index1, tmpTimestamps.end());
|
|
const uint64_t timestamp1 = oldest_timestamp + tmpTimestamps[index1];
|
|
|
|
std::nth_element(tmpTimestamps.begin(), tmpTimestamps.begin() + index2, tmpTimestamps.end());
|
|
const uint64_t timestamp2 = oldest_timestamp + tmpTimestamps[index2];
|
|
|
|
const uint64_t delta_t = (timestamp2 > timestamp1) ? (timestamp2 - timestamp1) : 1;
|
|
|
|
difficulty_type diff1{ std::numeric_limits<uint64_t>::max(), std::numeric_limits<uint64_t>::max() };
|
|
difficulty_type diff2{ 0, 0 };
|
|
|
|
for (const DifficultyData& d : difficultyData) {
|
|
if (timestamp1 <= d.m_timestamp && d.m_timestamp <= timestamp2) {
|
|
if (d.m_cumulativeDifficulty < diff1) {
|
|
diff1 = d.m_cumulativeDifficulty;
|
|
}
|
|
if (diff2 < d.m_cumulativeDifficulty) {
|
|
diff2 = d.m_cumulativeDifficulty;
|
|
}
|
|
}
|
|
}
|
|
|
|
// This is correct as long as the difference between two 128-bit difficulties is less than 2^64, even if it wraps
|
|
const uint64_t delta_diff = diff2.lo - diff1.lo;
|
|
|
|
uint64_t product[2];
|
|
product[0] = umul128(delta_diff, m_targetBlockTime, &product[1]);
|
|
|
|
if (product[1] >= delta_t) {
|
|
LOGERR(1, "calculated difficulty is too high for block at height = " << tip->m_sidechainHeight << ", id = " << tip->m_sidechainId << ", mainchain height = " << tip->m_txinGenHeight);
|
|
return false;
|
|
}
|
|
|
|
uint64_t rem;
|
|
curDifficulty.lo = udiv128(product[1], product[0], delta_t, &rem);
|
|
curDifficulty.hi = 0;
|
|
|
|
if (curDifficulty < m_minDifficulty) {
|
|
curDifficulty = m_minDifficulty;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void SideChain::verify_loop(PoolBlock* block)
|
|
{
|
|
// PoW is already checked at this point
|
|
|
|
std::vector<PoolBlock*> blocks_to_verify(1, block);
|
|
PoolBlock* highest_block = nullptr;
|
|
|
|
while (!blocks_to_verify.empty()) {
|
|
block = blocks_to_verify.back();
|
|
blocks_to_verify.pop_back();
|
|
|
|
if (block->m_verified) {
|
|
continue;
|
|
}
|
|
|
|
verify(block);
|
|
|
|
if (!block->m_verified) {
|
|
LOGINFO(6, "can't verify block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << ": parent or uncle blocks are not available)");
|
|
continue;
|
|
}
|
|
|
|
if (block->m_invalid) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " is invalid");
|
|
}
|
|
else {
|
|
LOGINFO(3, "verified block at height = " << block->m_sidechainHeight <<
|
|
", depth = " << block->m_depth <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight);
|
|
|
|
// This block is now verified
|
|
|
|
if (is_longer_chain(highest_block, block)) {
|
|
highest_block = block;
|
|
}
|
|
else if (highest_block && (highest_block->m_sidechainHeight > block->m_sidechainHeight)) {
|
|
LOGINFO(4, "block " << highest_block->m_sidechainId <<
|
|
", height = " << highest_block->m_sidechainHeight <<
|
|
" is not a longer chain than " << block->m_sidechainId <<
|
|
", height " << block->m_sidechainHeight);
|
|
}
|
|
|
|
// If it came through a broadcast, send it to our peers
|
|
if (block->m_wantBroadcast && !block->m_broadcasted) {
|
|
block->m_broadcasted = true;
|
|
if (m_pool->p2p_server() && (block->m_depth < UNCLE_BLOCK_DEPTH)) {
|
|
m_pool->p2p_server()->broadcast(*block);
|
|
}
|
|
}
|
|
|
|
// Save it for faster syncing on the next p2pool start
|
|
if (m_pool->p2p_server()) {
|
|
m_pool->p2p_server()->store_in_cache(*block);
|
|
}
|
|
|
|
// Try to verify blocks on top of this one
|
|
for (size_t i = 1; i <= UNCLE_BLOCK_DEPTH; ++i) {
|
|
auto it = m_blocksByHeight.find(block->m_sidechainHeight + i);
|
|
if (it == m_blocksByHeight.end()) {
|
|
continue;
|
|
}
|
|
|
|
const std::vector<PoolBlock*>& next_blocks = it->second;
|
|
if (!next_blocks.empty()) {
|
|
blocks_to_verify.insert(blocks_to_verify.end(), next_blocks.begin(), next_blocks.end());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (highest_block) {
|
|
update_chain_tip(highest_block);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
void SideChain::verify(PoolBlock* block)
|
|
{
|
|
// Genesis block
|
|
if (block->m_sidechainHeight == 0) {
|
|
if (!block->m_parent.empty() ||
|
|
!block->m_uncles.empty() ||
|
|
(block->m_difficulty != m_minDifficulty) ||
|
|
(block->m_cumulativeDifficulty != m_minDifficulty))
|
|
{
|
|
block->m_invalid = true;
|
|
}
|
|
|
|
block->m_verified = true;
|
|
return;
|
|
}
|
|
|
|
// Deep block
|
|
//
|
|
// Blocks in PPLNS window (m_chainWindowSize) require up to m_chainWindowSize earlier blocks to verify
|
|
// If a block is deeper than m_chainWindowSize * 2 - 1 it can't influence blocks in PPLNS window
|
|
// Also, having so many blocks on top of this one means it was verified by the network at some point
|
|
// We skip checks in this case to make pruning possible
|
|
if (block->m_depth >= m_chainWindowSize * 2) {
|
|
LOGINFO(4, "block " << block->m_sidechainId << " skipped verification");
|
|
block->m_verified = true;
|
|
block->m_invalid = false;
|
|
return;
|
|
}
|
|
|
|
// Regular block
|
|
|
|
// Must have a parent
|
|
if (block->m_parent.empty()) {
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
// Check parent
|
|
auto it = m_blocksById.find(block->m_parent);
|
|
if ((it == m_blocksById.end()) || !it->second->m_verified) {
|
|
block->m_verified = false;
|
|
return;
|
|
}
|
|
|
|
// If it's invalid then this block is also invalid
|
|
PoolBlock* parent = it->second;
|
|
if (parent->m_invalid) {
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
const uint64_t expectedHeight = parent->m_sidechainHeight + 1;
|
|
if (block->m_sidechainHeight != expectedHeight) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight <<
|
|
" has wrong height: expected " << expectedHeight);
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
// Uncle hashes must be sorted in the ascending order to prevent cheating when the same hash is repeated multiple times
|
|
for (size_t i = 1, n = block->m_uncles.size(); i < n; ++i) {
|
|
if (!(block->m_uncles[i - 1] < block->m_uncles[i])) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " has invalid uncle order");
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
}
|
|
|
|
difficulty_type expectedCumulativeDifficulty = parent->m_cumulativeDifficulty + block->m_difficulty;
|
|
|
|
// Check uncles
|
|
|
|
// First get a list of already mined blocks at possible uncle heights
|
|
std::vector<hash> mined_blocks;
|
|
|
|
if (!block->m_uncles.empty()) {
|
|
mined_blocks.reserve(UNCLE_BLOCK_DEPTH * 2 + 1);
|
|
|
|
PoolBlock* tmp = parent;
|
|
for (uint64_t i = 0, n = std::min<uint64_t>(UNCLE_BLOCK_DEPTH, block->m_sidechainHeight + 1); tmp && (i < n); ++i) {
|
|
mined_blocks.push_back(tmp->m_sidechainId);
|
|
mined_blocks.insert(mined_blocks.end(), tmp->m_uncles.begin(), tmp->m_uncles.end());
|
|
tmp = get_parent(tmp);
|
|
}
|
|
}
|
|
|
|
for (const hash& uncle_id : block->m_uncles) {
|
|
// Empty hash is only used in the genesis block and only for its parent
|
|
// Uncles can't be empty
|
|
if (uncle_id.empty()) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " has empty uncle hash");
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
// Can't mine the same uncle block twice
|
|
if (std::find(mined_blocks.begin(), mined_blocks.end(), uncle_id) != mined_blocks.end()) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " has an uncle (" << uncle_id << ") that's already been mined");
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
it = m_blocksById.find(uncle_id);
|
|
if ((it == m_blocksById.end()) || !it->second->m_verified) {
|
|
block->m_verified = false;
|
|
return;
|
|
}
|
|
|
|
PoolBlock* uncle = it->second;
|
|
|
|
// If it's invalid then this block is also invalid
|
|
if (uncle->m_invalid) {
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
// Check that it has correct height
|
|
if ((uncle->m_sidechainHeight >= block->m_sidechainHeight) || (uncle->m_sidechainHeight + UNCLE_BLOCK_DEPTH < block->m_sidechainHeight)) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " has an uncle at the wrong height (" << uncle->m_sidechainHeight << ')');
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
// Check that uncle and parent have the same ancestor (they must be on the same chain)
|
|
PoolBlock* tmp = parent;
|
|
while (tmp->m_sidechainHeight > uncle->m_sidechainHeight) {
|
|
tmp = get_parent(tmp);
|
|
if (!tmp) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " has an uncle from a different chain (check 1 failed)");
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (tmp->m_sidechainHeight < uncle->m_sidechainHeight) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " has an uncle from a different chain (check 2 failed)");
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
bool same_chain = false;
|
|
PoolBlock* tmp2 = uncle;
|
|
for (size_t j = 0; (j < UNCLE_BLOCK_DEPTH) && tmp && tmp2 && (tmp->m_sidechainHeight + UNCLE_BLOCK_DEPTH >= block->m_sidechainHeight); ++j) {
|
|
if (tmp->m_parent == tmp2->m_parent) {
|
|
same_chain = true;
|
|
break;
|
|
}
|
|
tmp = get_parent(tmp);
|
|
tmp2 = get_parent(tmp2);
|
|
}
|
|
|
|
if (!same_chain) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight << " has an uncle from a different chain (check 3 failed)");
|
|
block->m_verified = true;
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
expectedCumulativeDifficulty += uncle->m_difficulty;
|
|
}
|
|
|
|
// We can verify this block now (all previous blocks in the window are verified and valid)
|
|
// It can still turn out to be invalid
|
|
block->m_verified = true;
|
|
|
|
if (block->m_cumulativeDifficulty != expectedCumulativeDifficulty) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight <<
|
|
" has wrong cumulative difficulty: got " << block->m_cumulativeDifficulty << ", expected " << expectedCumulativeDifficulty);
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
// Verify difficulty and miner rewards only for blocks in PPLNS window
|
|
if (block->m_depth >= m_chainWindowSize) {
|
|
LOGINFO(4, "block " << block->m_sidechainId << " skipped diff/reward verification");
|
|
block->m_invalid = false;
|
|
return;
|
|
}
|
|
|
|
difficulty_type diff;
|
|
if (!get_difficulty(parent, m_difficultyData, diff)) {
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
if (diff != block->m_difficulty) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight <<
|
|
" has wrong difficulty: got " << block->m_difficulty << ", expected " << diff);
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
std::vector<MinerShare> shares;
|
|
if (!get_shares(block, shares)) {
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
if (shares.size() != block->m_outputs.size()) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight
|
|
<< " has invalid number of outputs: got " << block->m_outputs.size() << ", expected " << shares.size());
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
uint64_t total_reward = std::accumulate(block->m_outputs.begin(), block->m_outputs.end(), 0ULL,
|
|
[](uint64_t a, const PoolBlock::TxOutput& b)
|
|
{
|
|
return a + b.m_reward;
|
|
});
|
|
|
|
std::vector<uint64_t> rewards;
|
|
split_reward(total_reward, shares, rewards);
|
|
|
|
if (rewards.size() != block->m_outputs.size()) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight
|
|
<< " has invalid number of outputs: got " << block->m_outputs.size() << ", expected " << rewards.size());
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
for (size_t i = 0, n = rewards.size(); i < n; ++i) {
|
|
if (rewards[i] != block->m_outputs[i].m_reward) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight <<
|
|
" has invalid reward at index " << i << ": got " << block->m_outputs[i].m_reward << ", expected " << rewards[i]);
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
hash eph_public_key;
|
|
if (!shares[i].m_wallet->get_eph_public_key(block->m_txkeySec, i, eph_public_key)) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight <<
|
|
" failed to eph_public_key at index " << i);
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
|
|
if (eph_public_key != block->m_outputs[i].m_ephPublicKey) {
|
|
LOGWARN(3, "block at height = " << block->m_sidechainHeight <<
|
|
", id = " << block->m_sidechainId <<
|
|
", mainchain height = " << block->m_txinGenHeight <<
|
|
" pays out to a wrong wallet at index " << i);
|
|
block->m_invalid = true;
|
|
return;
|
|
}
|
|
}
|
|
|
|
// All checks passed
|
|
block->m_invalid = false;
|
|
}
|
|
|
|
void SideChain::update_chain_tip(PoolBlock* block)
|
|
{
|
|
if (!block->m_verified || block->m_invalid) {
|
|
LOGERR(1, "trying to update chain tip to an unverified or invalid block, fix the code!");
|
|
return;
|
|
}
|
|
|
|
if (block->m_depth >= m_chainWindowSize) {
|
|
LOGINFO(5, "Trying to update chain tip to a block with depth " << block->m_depth << ". Ignoring it.");
|
|
return;
|
|
}
|
|
|
|
if (is_longer_chain(m_chainTip, block)) {
|
|
difficulty_type diff;
|
|
if (get_difficulty(block, m_difficultyData, diff)) {
|
|
m_chainTip = block;
|
|
m_curDifficulty = diff;
|
|
|
|
LOGINFO(2, "new chain tip: next height = " << log::Gray() << block->m_sidechainHeight + 1 << log::NoColor() <<
|
|
", next difficulty = " << log::Gray() << m_curDifficulty << log::NoColor() <<
|
|
", main chain height = " << log::Gray() << m_chainTip->m_txinGenHeight);
|
|
|
|
block->m_wantBroadcast = true;
|
|
m_pool->update_block_template_async();
|
|
prune_old_blocks();
|
|
}
|
|
}
|
|
else if (block->m_sidechainHeight > m_chainTip->m_sidechainHeight) {
|
|
LOGINFO(4, "block " << block->m_sidechainId <<
|
|
", height = " << block->m_sidechainHeight <<
|
|
" is not a longer chain than " << m_chainTip->m_sidechainId <<
|
|
", height " << m_chainTip->m_sidechainHeight);
|
|
}
|
|
else if (block->m_sidechainHeight + UNCLE_BLOCK_DEPTH > m_chainTip->m_sidechainHeight) {
|
|
LOGINFO(4, "possible uncle block: id = " << log::Gray() << block->m_sidechainId << log::NoColor() <<
|
|
", height = " << log::Gray() << block->m_sidechainHeight);
|
|
m_pool->update_block_template_async();
|
|
}
|
|
|
|
if (m_pool->p2p_server() && block->m_wantBroadcast && !block->m_broadcasted) {
|
|
block->m_broadcasted = true;
|
|
#ifdef DEBUG_BROADCAST_DELAY_MS
|
|
struct Work
|
|
{
|
|
uv_work_t req;
|
|
P2PServer* server;
|
|
PoolBlock* block;
|
|
};
|
|
Work* work = new Work{};
|
|
work->req.data = work;
|
|
work->server = m_pool->p2p_server();
|
|
work->block = block;
|
|
const int err = uv_queue_work(uv_default_loop(), &work->req,
|
|
[](uv_work_t*)
|
|
{
|
|
num_running_jobs.fetch_add(1);
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(DEBUG_BROADCAST_DELAY_MS));
|
|
},
|
|
[](uv_work_t* req, int)
|
|
{
|
|
Work* work = reinterpret_cast<Work*>(req->data);
|
|
work->server->broadcast(*work->block);
|
|
delete reinterpret_cast<Work*>(req->data);
|
|
num_running_jobs.fetch_sub(1);
|
|
});
|
|
if (err) {
|
|
LOGERR(1, "update_chain_tip: uv_queue_work failed, error " << uv_err_name(err));
|
|
}
|
|
#else
|
|
m_pool->p2p_server()->broadcast(*block);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
PoolBlock* SideChain::get_parent(const PoolBlock* block)
|
|
{
|
|
if (block) {
|
|
auto it = m_blocksById.find(block->m_parent);
|
|
if (it != m_blocksById.end()) {
|
|
return it->second;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
bool SideChain::is_longer_chain(const PoolBlock* block, const PoolBlock* candidate)
|
|
{
|
|
if (!candidate || !candidate->m_verified || candidate->m_invalid) {
|
|
return false;
|
|
}
|
|
|
|
if (!block) {
|
|
return true;
|
|
}
|
|
|
|
// If these two blocks are on the same chain, they must have a common ancestor
|
|
|
|
const PoolBlock* block_ancestor = block;
|
|
while (block_ancestor->m_sidechainHeight > candidate->m_sidechainHeight) {
|
|
const hash& id = block_ancestor->m_parent;
|
|
block_ancestor = get_parent(block_ancestor);
|
|
if (!block_ancestor) {
|
|
LOGINFO(4, "couldn't find ancestor " << id << " of block " << block->m_sidechainId << " at height " << block->m_sidechainHeight);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (block_ancestor) {
|
|
const PoolBlock* candidate_ancestor = candidate;
|
|
while (candidate_ancestor->m_sidechainHeight > block_ancestor->m_sidechainHeight) {
|
|
const hash& id = candidate_ancestor->m_parent;
|
|
candidate_ancestor = get_parent(candidate_ancestor);
|
|
if (!candidate_ancestor) {
|
|
LOGINFO(4, "couldn't find ancestor " << id << " of block " << candidate->m_sidechainId << " at height " << candidate->m_sidechainHeight);
|
|
break;
|
|
}
|
|
}
|
|
|
|
while (block_ancestor && candidate_ancestor) {
|
|
if (block_ancestor->m_parent == candidate_ancestor->m_parent) {
|
|
// If they are really on the same chain, we can just compare cumulative difficulties
|
|
return block->m_cumulativeDifficulty < candidate->m_cumulativeDifficulty;
|
|
}
|
|
block_ancestor = get_parent(block_ancestor);
|
|
candidate_ancestor = get_parent(candidate_ancestor);
|
|
}
|
|
}
|
|
|
|
// They're on totally different chains. Compare total difficulties over the last m_chainWindowSize blocks
|
|
difficulty_type block_total_diff;
|
|
difficulty_type candidate_total_diff;
|
|
|
|
const PoolBlock* old_chain = block;
|
|
const PoolBlock* new_chain = candidate;
|
|
|
|
uint64_t candidate_mainchain_height = 0;
|
|
hash mainchain_prev_id;
|
|
|
|
for (uint64_t i = 0; (i < m_chainWindowSize) && (old_chain || new_chain); ++i) {
|
|
if (old_chain) {
|
|
block_total_diff += old_chain->m_difficulty;
|
|
old_chain = get_parent(old_chain);
|
|
}
|
|
|
|
if (new_chain) {
|
|
candidate_total_diff += new_chain->m_difficulty;
|
|
|
|
ChainMain data;
|
|
if ((new_chain->m_prevId != mainchain_prev_id) && m_pool->chainmain_get_by_hash(new_chain->m_prevId, data)) {
|
|
mainchain_prev_id = new_chain->m_prevId;
|
|
candidate_mainchain_height = std::max(candidate_mainchain_height, data.height);
|
|
}
|
|
|
|
new_chain = get_parent(new_chain);
|
|
}
|
|
}
|
|
|
|
if (block_total_diff >= candidate_total_diff) {
|
|
return false;
|
|
}
|
|
|
|
// Final check: candidate chain must be built on top of recent mainchain blocks
|
|
if (candidate_mainchain_height + 10 < m_pool->miner_data().height) {
|
|
LOGWARN(3, "received a longer alternative chain but it's stale: height " << candidate_mainchain_height << ", current height " << m_pool->miner_data().height);
|
|
return false;
|
|
}
|
|
|
|
LOGINFO(3, "received a longer alternative chain: height " <<
|
|
log::Gray() << block->m_sidechainHeight << log::NoColor() << " -> " <<
|
|
log::Gray() << candidate->m_sidechainHeight << log::NoColor() << ", cumulative difficulty " <<
|
|
log::Gray() << block->m_cumulativeDifficulty << log::NoColor() << " -> " <<
|
|
log::Gray() << candidate->m_cumulativeDifficulty);
|
|
|
|
return true;
|
|
}
|
|
|
|
void SideChain::update_depths(PoolBlock* block)
|
|
{
|
|
for (size_t i = 1; i <= UNCLE_BLOCK_DEPTH; ++i) {
|
|
for (PoolBlock* child : m_blocksByHeight[block->m_sidechainHeight + i]) {
|
|
if (child->m_parent == block->m_sidechainId) {
|
|
if (i != 1) {
|
|
LOGERR(1, "m_blocksByHeight is inconsistent with child->m_parent. Fix the code!");
|
|
}
|
|
else {
|
|
block->m_depth = std::max(block->m_depth, child->m_depth + 1);
|
|
}
|
|
}
|
|
|
|
auto it = std::find(child->m_uncles.begin(), child->m_uncles.end(), block->m_sidechainId);
|
|
if (it != child->m_uncles.end()) {
|
|
block->m_depth = std::max(block->m_depth, child->m_depth + i);
|
|
}
|
|
}
|
|
}
|
|
|
|
std::vector<PoolBlock*> blocks_to_update(1, block);
|
|
|
|
do {
|
|
block = blocks_to_update.back();
|
|
blocks_to_update.pop_back();
|
|
|
|
// Verify this block and possibly other blocks on top of it when we're sure it will get verified
|
|
if (!block->m_verified && ((block->m_depth >= m_chainWindowSize * 2) || (block->m_sidechainHeight == 0))) {
|
|
verify_loop(block);
|
|
}
|
|
|
|
auto it = m_blocksById.find(block->m_parent);
|
|
if (it != m_blocksById.end()) {
|
|
if (it->second->m_sidechainHeight + 1 != block->m_sidechainHeight) {
|
|
LOGERR(1, "m_sidechainHeight is inconsistent with block->m_parent. Fix the code!");
|
|
}
|
|
|
|
if (it->second->m_depth < block->m_depth + 1) {
|
|
it->second->m_depth = block->m_depth + 1;
|
|
blocks_to_update.push_back(it->second);
|
|
}
|
|
}
|
|
|
|
for (const hash& uncle_id : block->m_uncles) {
|
|
it = m_blocksById.find(uncle_id);
|
|
if (it == m_blocksById.end()) {
|
|
continue;
|
|
}
|
|
|
|
if ((it->second->m_sidechainHeight >= block->m_sidechainHeight) || (it->second->m_sidechainHeight + UNCLE_BLOCK_DEPTH < block->m_sidechainHeight)) {
|
|
LOGERR(1, "m_sidechainHeight is inconsistent with block->m_uncles. Fix the code!");
|
|
}
|
|
|
|
const uint64_t d = block->m_sidechainHeight - it->second->m_sidechainHeight;
|
|
if (it->second->m_depth < block->m_depth + d) {
|
|
it->second->m_depth = block->m_depth + d;
|
|
blocks_to_update.push_back(it->second);
|
|
}
|
|
}
|
|
} while (!blocks_to_update.empty());
|
|
}
|
|
|
|
void SideChain::prune_old_blocks()
|
|
{
|
|
// Leave 2 minutes worth of spare blocks in addition to 2xPPLNS window for lagging nodes which need to sync
|
|
const uint64_t prune_distance = m_chainWindowSize * 2 + 120 / m_targetBlockTime;
|
|
|
|
// Remove old blocks from alternative unconnected chains after long enough time
|
|
const time_t prune_time = time(nullptr) - m_chainWindowSize * 4 * m_targetBlockTime;
|
|
|
|
if (m_chainTip->m_sidechainHeight < prune_distance) {
|
|
return;
|
|
}
|
|
|
|
const uint64_t h = m_chainTip->m_sidechainHeight - prune_distance;
|
|
|
|
uint64_t num_blocks_pruned = 0;
|
|
|
|
for (auto it = m_blocksByHeight.begin(); (it != m_blocksByHeight.end()) && (it->first <= h);) {
|
|
const uint64_t height = it->first;
|
|
std::vector<PoolBlock*>& v = it->second;
|
|
|
|
v.erase(std::remove_if(v.begin(), v.end(),
|
|
[this, prune_distance, prune_time, &num_blocks_pruned, height](PoolBlock* block)
|
|
{
|
|
if ((block->m_depth >= prune_distance) || (block->m_localTimestamp <= prune_time)) {
|
|
auto it2 = m_blocksById.find(block->m_sidechainId);
|
|
if (it2 != m_blocksById.end()) {
|
|
m_blocksById.erase(it2);
|
|
delete block;
|
|
++num_blocks_pruned;
|
|
}
|
|
else {
|
|
LOGERR(1, "m_blocksByHeight and m_blocksById are inconsistent at height " << height << ". Fix the code!");
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}), v.end());
|
|
|
|
if (v.empty()) {
|
|
it = m_blocksByHeight.erase(it);
|
|
}
|
|
else {
|
|
++it;
|
|
}
|
|
}
|
|
|
|
if (num_blocks_pruned) {
|
|
LOGINFO(4, "pruned " << num_blocks_pruned << " old blocks at heights <= " << h);
|
|
}
|
|
}
|
|
|
|
void SideChain::get_missing_blocks(std::vector<hash>& missing_blocks)
|
|
{
|
|
missing_blocks.clear();
|
|
|
|
MutexLock lock(m_sidechainLock);
|
|
|
|
for (auto& b : m_blocksById) {
|
|
if (b.second->m_verified) {
|
|
continue;
|
|
}
|
|
|
|
if (!b.second->m_parent.empty() && (m_blocksById.find(b.second->m_parent) == m_blocksById.end())) {
|
|
missing_blocks.push_back(b.second->m_parent);
|
|
}
|
|
|
|
for (const hash& h : b.second->m_uncles) {
|
|
if (!h.empty() && (m_blocksById.find(h) == m_blocksById.end())) {
|
|
missing_blocks.push_back(h);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool SideChain::load_config(const std::string& filename)
|
|
{
|
|
if (filename.empty()) {
|
|
LOGINFO(1, "using default config");
|
|
return true;
|
|
}
|
|
|
|
LOGINFO(1, "loading config from " << log::Gray() << filename);
|
|
|
|
std::ifstream f(filename);
|
|
if (!f.is_open()) {
|
|
LOGERR(1, "can't open " << filename);
|
|
return false;
|
|
}
|
|
|
|
rapidjson::Document doc;
|
|
rapidjson::IStreamWrapper s(f);
|
|
if (doc.ParseStream<rapidjson::kParseCommentsFlag | rapidjson::kParseTrailingCommasFlag>(s).HasParseError()) {
|
|
LOGERR(1, "failed to parse JSON data in " << filename);
|
|
return false;
|
|
}
|
|
|
|
if (!doc.IsObject()) {
|
|
LOGERR(1, "invalid JSON data in " << filename << ": top level is not an object");
|
|
return false;
|
|
}
|
|
|
|
parseValue(doc, "name", m_poolName);
|
|
parseValue(doc, "password", m_poolPassword);
|
|
parseValue(doc, "block_time", m_targetBlockTime);
|
|
|
|
uint64_t min_diff;
|
|
if (parseValue(doc, "min_diff", min_diff)) {
|
|
m_minDifficulty = { min_diff, 0 };
|
|
}
|
|
|
|
parseValue(doc, "pplns_window", m_chainWindowSize);
|
|
parseValue(doc, "uncle_penalty", m_unclePenalty);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SideChain::check_config()
|
|
{
|
|
if (m_poolName.empty()) {
|
|
LOGERR(1, "name can't be empty");
|
|
return false;
|
|
}
|
|
|
|
if (m_poolName.length() > 128) {
|
|
LOGERR(1, "name is too long (must be 128 characters max)");
|
|
return false;
|
|
}
|
|
|
|
if (m_poolPassword.length() > 128) {
|
|
LOGERR(1, "password is too long (must be 128 characters max)");
|
|
return false;
|
|
}
|
|
|
|
if ((m_targetBlockTime < 1) || (m_targetBlockTime > 120)) {
|
|
LOGERR(1, "block_time is invalid (must be between 1 and 120)");
|
|
return false;
|
|
}
|
|
|
|
const difficulty_type min_diff{ MIN_DIFFICULTY, 0 };
|
|
const difficulty_type max_diff{ 1000000000, 0 };
|
|
|
|
if ((m_minDifficulty < min_diff) || (max_diff < m_minDifficulty)) {
|
|
LOGERR(1, "min_diff is invalid (must be between " << min_diff << " and " << max_diff << ')');
|
|
return false;
|
|
}
|
|
|
|
if ((m_chainWindowSize < 60) || (m_chainWindowSize > 2160)) {
|
|
LOGERR(1, "pplns_window is invalid (must be between 60 and 2160)");
|
|
return false;
|
|
}
|
|
|
|
if ((m_unclePenalty < 1) || (m_unclePenalty > 99)) {
|
|
LOGERR(1, "uncle_penalty is invalid (must be between 1 and 99)");
|
|
return false;
|
|
}
|
|
|
|
LOGINFO(1, log::LightCyan() << "pool name = " << m_poolName);
|
|
LOGINFO(1, log::LightCyan() << "block time = " << m_targetBlockTime << " seconds");
|
|
LOGINFO(1, log::LightCyan() << "min diff = " << m_minDifficulty);
|
|
LOGINFO(1, log::LightCyan() << "PPLNS window = " << m_chainWindowSize << " blocks");
|
|
LOGINFO(1, log::LightCyan() << "uncle penalty = " << m_unclePenalty << '%');
|
|
|
|
return true;
|
|
}
|
|
|
|
} // namespace p2pool
|