mirror of
https://github.com/SChernykh/p2pool.git
synced 2025-01-05 10:19:23 +00:00
Block cache WIP and other fixes
- Block cache is implemented only on Windows for now - Tracking of background jobs - More robust sidechain syncing
This commit is contained in:
parent
86b31ea821
commit
aba3bc50b8
13 changed files with 461 additions and 57 deletions
|
@ -13,6 +13,7 @@ include(cmake/flags.cmake)
|
||||||
set(HEADERS
|
set(HEADERS
|
||||||
external/src/cryptonote/crypto-ops.h
|
external/src/cryptonote/crypto-ops.h
|
||||||
external/src/llhttp/llhttp.h
|
external/src/llhttp/llhttp.h
|
||||||
|
src/block_cache.h
|
||||||
src/block_template.h
|
src/block_template.h
|
||||||
src/common.h
|
src/common.h
|
||||||
src/console_commands.h
|
src/console_commands.h
|
||||||
|
@ -44,6 +45,7 @@ set(SOURCES
|
||||||
external/src/llhttp/api.c
|
external/src/llhttp/api.c
|
||||||
external/src/llhttp/http.c
|
external/src/llhttp/http.c
|
||||||
external/src/llhttp/llhttp.c
|
external/src/llhttp/llhttp.c
|
||||||
|
src/block_cache.cpp
|
||||||
src/block_template.cpp
|
src/block_template.cpp
|
||||||
src/console_commands.cpp
|
src/console_commands.cpp
|
||||||
src/crypto.cpp
|
src/crypto.cpp
|
||||||
|
|
161
src/block_cache.cpp
Normal file
161
src/block_cache.cpp
Normal file
|
@ -0,0 +1,161 @@
|
||||||
|
/*
|
||||||
|
* 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 "block_cache.h"
|
||||||
|
#include "pool_block.h"
|
||||||
|
#include "p2p_server.h"
|
||||||
|
|
||||||
|
static constexpr char log_category_prefix[] = "BlockCache ";
|
||||||
|
|
||||||
|
static constexpr uint32_t BLOCK_SIZE = 96 * 1024;
|
||||||
|
static constexpr uint32_t NUM_BLOCKS = 5120;
|
||||||
|
static constexpr uint32_t CACHE_SIZE = BLOCK_SIZE * NUM_BLOCKS;
|
||||||
|
static constexpr char cache_name[] = "p2pool.cache";
|
||||||
|
|
||||||
|
namespace p2pool {
|
||||||
|
|
||||||
|
struct BlockCache::Impl
|
||||||
|
{
|
||||||
|
#ifdef _WIN32
|
||||||
|
Impl()
|
||||||
|
{
|
||||||
|
m_file = CreateFile(cache_name, GENERIC_ALL, FILE_SHARE_READ, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_HIDDEN, NULL);
|
||||||
|
if (m_file == INVALID_HANDLE_VALUE) {
|
||||||
|
LOGERR(1, "couldn't open " << cache_name << ", error " << static_cast<uint32_t>(GetLastError()));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (SetFilePointer(m_file, CACHE_SIZE, NULL, FILE_BEGIN) == INVALID_SET_FILE_POINTER) {
|
||||||
|
LOGERR(1, "SetFilePointer failed, error " << static_cast<uint32_t>(GetLastError()));
|
||||||
|
CloseHandle(m_file);
|
||||||
|
m_file = INVALID_HANDLE_VALUE;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!SetEndOfFile(m_file)) {
|
||||||
|
LOGERR(1, "SetEndOfFile failed, error " << static_cast<uint32_t>(GetLastError()));
|
||||||
|
CloseHandle(m_file);
|
||||||
|
m_file = INVALID_HANDLE_VALUE;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_map = CreateFileMapping(m_file, NULL, PAGE_READWRITE, 0, CACHE_SIZE, NULL);
|
||||||
|
if (!m_map) {
|
||||||
|
LOGERR(1, "CreateFileMapping failed, error " << static_cast<uint32_t>(GetLastError()));
|
||||||
|
CloseHandle(m_file);
|
||||||
|
m_file = INVALID_HANDLE_VALUE;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_data = reinterpret_cast<uint8_t*>(MapViewOfFile(m_map, FILE_MAP_ALL_ACCESS, 0, 0, 0));
|
||||||
|
if (!m_data) {
|
||||||
|
LOGERR(1, "MapViewOfFile failed, error " << static_cast<uint32_t>(GetLastError()));
|
||||||
|
CloseHandle(m_map);
|
||||||
|
CloseHandle(m_file);
|
||||||
|
m_map = 0;
|
||||||
|
m_file = INVALID_HANDLE_VALUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~Impl()
|
||||||
|
{
|
||||||
|
if (m_data) UnmapViewOfFile(m_data);
|
||||||
|
if (m_map) CloseHandle(m_map);
|
||||||
|
if (m_file != INVALID_HANDLE_VALUE) CloseHandle(m_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
void flush()
|
||||||
|
{
|
||||||
|
if (m_data && (m_flushRunning.exchange(1) == 0)) {
|
||||||
|
FlushViewOfFile(m_data, 0);
|
||||||
|
FlushFileBuffers(m_file);
|
||||||
|
m_flushRunning.store(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
HANDLE m_file = INVALID_HANDLE_VALUE;
|
||||||
|
HANDLE m_map = 0;
|
||||||
|
|
||||||
|
#else
|
||||||
|
// TODO: Linux version is not implemented yet
|
||||||
|
void flush() {}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
uint8_t* m_data = nullptr;
|
||||||
|
std::atomic<uint32_t> m_flushRunning{ 0 };
|
||||||
|
};
|
||||||
|
|
||||||
|
BlockCache::BlockCache()
|
||||||
|
: m_impl(new Impl())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
BlockCache::~BlockCache()
|
||||||
|
{
|
||||||
|
delete m_impl;
|
||||||
|
}
|
||||||
|
|
||||||
|
void BlockCache::store(const PoolBlock& block)
|
||||||
|
{
|
||||||
|
if (!m_impl->m_data) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t* data = m_impl->m_data + (static_cast<size_t>(block.m_sidechainHeight % NUM_BLOCKS) * BLOCK_SIZE);
|
||||||
|
|
||||||
|
const size_t n1 = block.m_mainChainData.size();
|
||||||
|
const size_t n2 = block.m_sideChainData.size();
|
||||||
|
|
||||||
|
*reinterpret_cast<uint32_t*>(data) = static_cast<uint32_t>(n1 + n2);
|
||||||
|
memcpy(data + sizeof(uint32_t), block.m_mainChainData.data(), n1);
|
||||||
|
memcpy(data + sizeof(uint32_t) + n1, block.m_sideChainData.data(), n2);
|
||||||
|
}
|
||||||
|
|
||||||
|
void BlockCache::load_all(SideChain& side_chain, P2PServer& server)
|
||||||
|
{
|
||||||
|
if (!m_impl->m_data) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
LOGINFO(1, "loading cached blocks");
|
||||||
|
|
||||||
|
PoolBlock block;
|
||||||
|
uint32_t blocks_loaded = 0;
|
||||||
|
|
||||||
|
for (uint64_t i = 0; i < NUM_BLOCKS; ++i) {
|
||||||
|
const uint8_t* data = m_impl->m_data + i * BLOCK_SIZE;
|
||||||
|
const uint32_t n = *reinterpret_cast<const uint32_t*>(data);
|
||||||
|
|
||||||
|
if (!n || (n + sizeof(uint32_t) > BLOCK_SIZE)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
block.deserialize(data + sizeof(uint32_t), n, side_chain);
|
||||||
|
server.add_cached_block(block);
|
||||||
|
++blocks_loaded;
|
||||||
|
}
|
||||||
|
|
||||||
|
LOGINFO(1, "loaded " << blocks_loaded << " cached blocks");
|
||||||
|
}
|
||||||
|
|
||||||
|
void BlockCache::flush()
|
||||||
|
{
|
||||||
|
m_impl->flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace p2pool
|
41
src/block_cache.h
Normal file
41
src/block_cache.h
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
/*
|
||||||
|
* 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
|
||||||
|
|
||||||
|
namespace p2pool {
|
||||||
|
|
||||||
|
struct PoolBlock;
|
||||||
|
class SideChain;
|
||||||
|
class P2PServer;
|
||||||
|
|
||||||
|
class BlockCache
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
BlockCache();
|
||||||
|
~BlockCache();
|
||||||
|
|
||||||
|
void store(const PoolBlock& block);
|
||||||
|
void load_all(SideChain& side_chain, P2PServer& server);
|
||||||
|
void flush();
|
||||||
|
|
||||||
|
private:
|
||||||
|
struct Impl;
|
||||||
|
Impl* m_impl;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace p2pool
|
51
src/log.cpp
51
src/log.cpp
|
@ -44,7 +44,7 @@ public:
|
||||||
enum params : int
|
enum params : int
|
||||||
{
|
{
|
||||||
SLOT_SIZE = 1024,
|
SLOT_SIZE = 1024,
|
||||||
BUF_SIZE = SLOT_SIZE * 1024,
|
BUF_SIZE = SLOT_SIZE * 16384,
|
||||||
};
|
};
|
||||||
|
|
||||||
FORCEINLINE Worker()
|
FORCEINLINE Worker()
|
||||||
|
@ -240,27 +240,9 @@ NOINLINE Writer::Writer(Severity severity) : Stream(m_stackBuf)
|
||||||
m_buf[0] = static_cast<char>(severity);
|
m_buf[0] = static_cast<char>(severity);
|
||||||
m_pos = 3;
|
m_pos = 3;
|
||||||
|
|
||||||
using namespace std::chrono;
|
*this << Cyan();
|
||||||
|
writeCurrentTime();
|
||||||
const system_clock::time_point now = system_clock::now();
|
*this << NoColor() << ' ';
|
||||||
const time_t t0 = system_clock::to_time_t(now);
|
|
||||||
|
|
||||||
tm t;
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
localtime_s(&t, &t0);
|
|
||||||
#else
|
|
||||||
localtime_r(&t0, &t);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
m_numberWidth = 2;
|
|
||||||
*this << Cyan() << (t.tm_year + 1900) << '-' << (t.tm_mon + 1) << '-' << t.tm_mday << ' ' << t.tm_hour << ':' << t.tm_min << ':' << t.tm_sec << '.';
|
|
||||||
|
|
||||||
const int32_t mcs = time_point_cast<microseconds>(now).time_since_epoch().count() % 1000000;
|
|
||||||
|
|
||||||
m_numberWidth = 4;
|
|
||||||
*this << (mcs / 100) << NoColor() << ' ';
|
|
||||||
m_numberWidth = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
NOINLINE Writer::~Writer()
|
NOINLINE Writer::~Writer()
|
||||||
|
@ -283,6 +265,31 @@ void stop()
|
||||||
worker.stop();
|
worker.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NOINLINE void Stream::writeCurrentTime()
|
||||||
|
{
|
||||||
|
using namespace std::chrono;
|
||||||
|
|
||||||
|
const system_clock::time_point now = system_clock::now();
|
||||||
|
const time_t t0 = system_clock::to_time_t(now);
|
||||||
|
|
||||||
|
tm t;
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
localtime_s(&t, &t0);
|
||||||
|
#else
|
||||||
|
localtime_r(&t0, &t);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
m_numberWidth = 2;
|
||||||
|
*this << (t.tm_year + 1900) << '-' << (t.tm_mon + 1) << '-' << t.tm_mday << ' ' << t.tm_hour << ':' << t.tm_min << ':' << t.tm_sec << '.';
|
||||||
|
|
||||||
|
const int32_t mcs = time_point_cast<microseconds>(now).time_since_epoch().count() % 1000000;
|
||||||
|
|
||||||
|
m_numberWidth = 4;
|
||||||
|
*this << (mcs / 100);
|
||||||
|
m_numberWidth = 1;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace log
|
} // namespace log
|
||||||
|
|
||||||
} // namespace p2pool
|
} // namespace p2pool
|
||||||
|
|
|
@ -98,6 +98,8 @@ struct Stream
|
||||||
FORCEINLINE int getNumberWidth() const { return m_numberWidth; }
|
FORCEINLINE int getNumberWidth() const { return m_numberWidth; }
|
||||||
FORCEINLINE void setNumberWidth(int width) { m_numberWidth = width; }
|
FORCEINLINE void setNumberWidth(int width) { m_numberWidth = width; }
|
||||||
|
|
||||||
|
NOINLINE void writeCurrentTime();
|
||||||
|
|
||||||
int m_pos;
|
int m_pos;
|
||||||
int m_numberWidth;
|
int m_numberWidth;
|
||||||
char* m_buf;
|
char* m_buf;
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
#include "keccak.h"
|
#include "keccak.h"
|
||||||
#include "side_chain.h"
|
#include "side_chain.h"
|
||||||
#include "pool_block.h"
|
#include "pool_block.h"
|
||||||
|
#include "block_cache.h"
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <numeric>
|
#include <numeric>
|
||||||
|
|
||||||
|
@ -38,6 +39,8 @@ namespace p2pool {
|
||||||
P2PServer::P2PServer(p2pool* pool)
|
P2PServer::P2PServer(p2pool* pool)
|
||||||
: TCPServer(P2PClient::allocate, pool->params().m_p2pAddresses)
|
: TCPServer(P2PClient::allocate, pool->params().m_p2pAddresses)
|
||||||
, m_pool(pool)
|
, m_pool(pool)
|
||||||
|
, m_cache(new BlockCache())
|
||||||
|
, m_cacheLoaded(false)
|
||||||
, m_rd{}
|
, m_rd{}
|
||||||
, m_rng(m_rd())
|
, m_rng(m_rd())
|
||||||
, m_block(new PoolBlock())
|
, m_block(new PoolBlock())
|
||||||
|
@ -49,6 +52,7 @@ P2PServer::P2PServer(p2pool* pool)
|
||||||
uv_mutex_init_checked(&m_blockLock);
|
uv_mutex_init_checked(&m_blockLock);
|
||||||
uv_mutex_init_checked(&m_peerListLock);
|
uv_mutex_init_checked(&m_peerListLock);
|
||||||
uv_mutex_init_checked(&m_broadcastLock);
|
uv_mutex_init_checked(&m_broadcastLock);
|
||||||
|
uv_rwlock_init_checked(&m_cachedBlocksLock);
|
||||||
|
|
||||||
int err = uv_async_init(&m_loop, &m_broadcastAsync, on_broadcast);
|
int err = uv_async_init(&m_loop, &m_broadcastAsync, on_broadcast);
|
||||||
if (err) {
|
if (err) {
|
||||||
|
@ -64,6 +68,12 @@ P2PServer::P2PServer(p2pool* pool)
|
||||||
panic();
|
panic();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (m_cache) {
|
||||||
|
WriteLock lock(m_cachedBlocksLock);
|
||||||
|
m_cache->load_all(m_pool->side_chain(), *this);
|
||||||
|
m_cacheLoaded = true;
|
||||||
|
}
|
||||||
|
|
||||||
m_timer.data = this;
|
m_timer.data = this;
|
||||||
err = uv_timer_start(&m_timer, on_timer, 10000, 2000);
|
err = uv_timer_start(&m_timer, on_timer, 10000, 2000);
|
||||||
if (err) {
|
if (err) {
|
||||||
|
@ -86,8 +96,33 @@ P2PServer::~P2PServer()
|
||||||
uv_mutex_destroy(&m_blockLock);
|
uv_mutex_destroy(&m_blockLock);
|
||||||
uv_mutex_destroy(&m_peerListLock);
|
uv_mutex_destroy(&m_peerListLock);
|
||||||
uv_mutex_destroy(&m_broadcastLock);
|
uv_mutex_destroy(&m_broadcastLock);
|
||||||
|
uv_rwlock_destroy(&m_cachedBlocksLock);
|
||||||
|
|
||||||
delete m_block;
|
delete m_block;
|
||||||
|
|
||||||
|
for (auto it : m_cachedBlocks) {
|
||||||
|
delete it.second;
|
||||||
|
}
|
||||||
|
|
||||||
|
delete m_cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
void P2PServer::add_cached_block(const PoolBlock& block)
|
||||||
|
{
|
||||||
|
if (m_cacheLoaded) {
|
||||||
|
LOGERR(1, "add_cached_block can only be called on startup. Fix the code!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
PoolBlock* new_block = new PoolBlock(block);
|
||||||
|
m_cachedBlocks.insert({ new_block->m_sidechainId, new_block });
|
||||||
|
}
|
||||||
|
|
||||||
|
void P2PServer::store_in_cache(const PoolBlock& block)
|
||||||
|
{
|
||||||
|
if (m_cache) {
|
||||||
|
m_cache->store(block);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void P2PServer::connect_to_peers(const std::string& peer_list)
|
void P2PServer::connect_to_peers(const std::string& peer_list)
|
||||||
|
@ -200,13 +235,13 @@ void P2PServer::save_peer_list_async()
|
||||||
const int err = uv_queue_work(&m_loop, &work->req,
|
const int err = uv_queue_work(&m_loop, &work->req,
|
||||||
[](uv_work_t* req)
|
[](uv_work_t* req)
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_add(1);
|
bkg_jobs_tracker.start("P2PServer::save_peer_list_async");
|
||||||
reinterpret_cast<Work*>(req->data)->server->save_peer_list();
|
reinterpret_cast<Work*>(req->data)->server->save_peer_list();
|
||||||
},
|
},
|
||||||
[](uv_work_t* req, int /*status*/)
|
[](uv_work_t* req, int /*status*/)
|
||||||
{
|
{
|
||||||
delete reinterpret_cast<Work*>(req->data);
|
delete reinterpret_cast<Work*>(req->data);
|
||||||
num_running_jobs.fetch_sub(1);
|
bkg_jobs_tracker.stop("P2PServer::save_peer_list_async");
|
||||||
});
|
});
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
|
@ -384,7 +419,7 @@ void P2PServer::broadcast(const PoolBlock& block)
|
||||||
data->ancestor_hashes = block.m_uncles;
|
data->ancestor_hashes = block.m_uncles;
|
||||||
data->ancestor_hashes.push_back(block.m_parent);
|
data->ancestor_hashes.push_back(block.m_parent);
|
||||||
|
|
||||||
LOGINFO(5, "Broadcasting block " << block.m_sidechainId << ": " << data->pruned_blob.size() << '/' << data->blob.size() << " bytes (pruned/full)");
|
LOGINFO(5, "Broadcasting block " << block.m_sidechainId << " (height " << block.m_sidechainHeight << "): " << data->pruned_blob.size() << '/' << data->blob.size() << " bytes (pruned/full)");
|
||||||
|
|
||||||
{
|
{
|
||||||
MutexLock lock(m_broadcastLock);
|
MutexLock lock(m_broadcastLock);
|
||||||
|
@ -504,12 +539,46 @@ void P2PServer::print_status()
|
||||||
|
|
||||||
void P2PServer::on_timer()
|
void P2PServer::on_timer()
|
||||||
{
|
{
|
||||||
|
flush_cache();
|
||||||
download_missing_blocks();
|
download_missing_blocks();
|
||||||
update_peer_connections();
|
update_peer_connections();
|
||||||
update_peer_list();
|
update_peer_list();
|
||||||
save_peer_list_async();
|
save_peer_list_async();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void P2PServer::flush_cache()
|
||||||
|
{
|
||||||
|
if (!m_cache) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Work
|
||||||
|
{
|
||||||
|
uv_work_t req;
|
||||||
|
BlockCache* cache;
|
||||||
|
};
|
||||||
|
|
||||||
|
Work* work = new Work{};
|
||||||
|
work->req.data = work;
|
||||||
|
work->cache = m_cache;
|
||||||
|
|
||||||
|
const int err = uv_queue_work(uv_default_loop(), &work->req,
|
||||||
|
[](uv_work_t* req)
|
||||||
|
{
|
||||||
|
bkg_jobs_tracker.start("P2PServer::flush_cache");
|
||||||
|
reinterpret_cast<Work*>(req->data)->cache->flush();
|
||||||
|
},
|
||||||
|
[](uv_work_t* req, int)
|
||||||
|
{
|
||||||
|
delete reinterpret_cast<Work*>(req->data);
|
||||||
|
bkg_jobs_tracker.stop("P2PServer::flush_cache");
|
||||||
|
});
|
||||||
|
|
||||||
|
if (err) {
|
||||||
|
LOGERR(1, "flush_cache: uv_queue_work failed, error " << uv_err_name(err));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void P2PServer::download_missing_blocks()
|
void P2PServer::download_missing_blocks()
|
||||||
{
|
{
|
||||||
std::vector<hash> missing_blocks;
|
std::vector<hash> missing_blocks;
|
||||||
|
@ -822,7 +891,7 @@ void P2PServer::P2PClient::send_handshake_solution(const uint8_t (&challenge)[CH
|
||||||
const int err = uv_queue_work(&server->m_loop, &work->req,
|
const int err = uv_queue_work(&server->m_loop, &work->req,
|
||||||
[](uv_work_t* req)
|
[](uv_work_t* req)
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_add(1);
|
bkg_jobs_tracker.start("P2PServer::send_handshake_solution");
|
||||||
|
|
||||||
Work* work = reinterpret_cast<Work*>(req->data);
|
Work* work = reinterpret_cast<Work*>(req->data);
|
||||||
const std::vector<uint8_t>& consensus_id = work->server->m_pool->side_chain().consensus_id();
|
const std::vector<uint8_t>& consensus_id = work->server->m_pool->side_chain().consensus_id();
|
||||||
|
@ -879,7 +948,7 @@ void P2PServer::P2PClient::send_handshake_solution(const uint8_t (&challenge)[CH
|
||||||
[work]()
|
[work]()
|
||||||
{
|
{
|
||||||
delete work;
|
delete work;
|
||||||
num_running_jobs.fetch_sub(1);
|
bkg_jobs_tracker.stop("P2PServer::send_handshake_solution");
|
||||||
});
|
});
|
||||||
|
|
||||||
// We might've been disconnected while working on the challenge, do nothing in this case
|
// We might've been disconnected while working on the challenge, do nothing in this case
|
||||||
|
@ -1118,7 +1187,7 @@ bool P2PServer::P2PClient::on_block_response(const uint8_t* buf, uint32_t size)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return handle_incoming_block_async();
|
return handle_incoming_block_async(server->m_block);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool P2PServer::P2PClient::on_block_broadcast(const uint8_t* buf, uint32_t size)
|
bool P2PServer::P2PClient::on_block_broadcast(const uint8_t* buf, uint32_t size)
|
||||||
|
@ -1145,13 +1214,13 @@ bool P2PServer::P2PClient::on_block_broadcast(const uint8_t* buf, uint32_t size)
|
||||||
|
|
||||||
if ((server->m_block->m_prevId != server->m_pool->miner_data().prev_id) &&
|
if ((server->m_block->m_prevId != server->m_pool->miner_data().prev_id) &&
|
||||||
(server->m_block->m_txinGenHeight < server->m_pool->miner_data().height)){
|
(server->m_block->m_txinGenHeight < server->m_pool->miner_data().height)){
|
||||||
LOGINFO(4, "peer " << static_cast<char*>(m_addrString) << " broadcasted a stale block, ignoring it");
|
LOGINFO(4, "peer " << static_cast<char*>(m_addrString) << " broadcasted a stale block (mainchain height " << server->m_block->m_txinGenHeight << ", expected >= " << server->m_pool->miner_data().height << "), ignoring it");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
server->m_block->m_wantBroadcast = true;
|
server->m_block->m_wantBroadcast = true;
|
||||||
|
|
||||||
return handle_incoming_block_async();
|
return handle_incoming_block_async(server->m_block);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool P2PServer::P2PClient::on_peer_list_request(const uint8_t*)
|
bool P2PServer::P2PClient::on_peer_list_request(const uint8_t*)
|
||||||
|
@ -1245,15 +1314,17 @@ bool P2PServer::P2PClient::on_peer_list_response(const uint8_t* buf) const
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool P2PServer::P2PClient::handle_incoming_block_async()
|
bool P2PServer::P2PClient::handle_incoming_block_async(PoolBlock* block)
|
||||||
{
|
{
|
||||||
P2PServer* server = static_cast<P2PServer*>(m_owner);
|
P2PServer* server = static_cast<P2PServer*>(m_owner);
|
||||||
|
|
||||||
if (server->m_pool->side_chain().block_seen(*server->m_block)) {
|
if (server->m_pool->side_chain().block_seen(*block)) {
|
||||||
LOGINFO(5, "block " << server->m_block->m_sidechainId << " was received before, skipping it");
|
LOGINFO(5, "block " << block->m_sidechainId << " was received before, skipping it");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
server->store_in_cache(*block);
|
||||||
|
|
||||||
struct Work
|
struct Work
|
||||||
{
|
{
|
||||||
uv_work_t req;
|
uv_work_t req;
|
||||||
|
@ -1264,13 +1335,13 @@ bool P2PServer::P2PClient::handle_incoming_block_async()
|
||||||
std::vector<hash> missing_blocks;
|
std::vector<hash> missing_blocks;
|
||||||
};
|
};
|
||||||
|
|
||||||
Work* work = new Work{ {}, *server->m_block, this, server, m_resetCounter.load(), {} };
|
Work* work = new Work{ {}, *block, this, server, m_resetCounter.load(), {} };
|
||||||
work->req.data = work;
|
work->req.data = work;
|
||||||
|
|
||||||
const int err = uv_queue_work(&server->m_loop, &work->req,
|
const int err = uv_queue_work(&server->m_loop, &work->req,
|
||||||
[](uv_work_t* req)
|
[](uv_work_t* req)
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_add(1);
|
bkg_jobs_tracker.start("P2PServer::handle_incoming_block_async");
|
||||||
Work* work = reinterpret_cast<Work*>(req->data);
|
Work* work = reinterpret_cast<Work*>(req->data);
|
||||||
work->client->handle_incoming_block(work->server->m_pool, work->block, work->client_reset_counter, work->missing_blocks);
|
work->client->handle_incoming_block(work->server->m_pool, work->block, work->client_reset_counter, work->missing_blocks);
|
||||||
},
|
},
|
||||||
|
@ -1279,7 +1350,7 @@ bool P2PServer::P2PClient::handle_incoming_block_async()
|
||||||
Work* work = reinterpret_cast<Work*>(req->data);
|
Work* work = reinterpret_cast<Work*>(req->data);
|
||||||
work->client->post_handle_incoming_block(work->client_reset_counter, work->missing_blocks);
|
work->client->post_handle_incoming_block(work->client_reset_counter, work->missing_blocks);
|
||||||
delete work;
|
delete work;
|
||||||
num_running_jobs.fetch_sub(1);
|
bkg_jobs_tracker.stop("P2PServer::handle_incoming_block_async");
|
||||||
});
|
});
|
||||||
|
|
||||||
if (err != 0) {
|
if (err != 0) {
|
||||||
|
@ -1311,7 +1382,22 @@ void P2PServer::P2PClient::post_handle_incoming_block(const uint32_t reset_count
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (missing_blocks.empty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
P2PServer* server = static_cast<P2PServer*>(m_owner);
|
||||||
|
|
||||||
|
ReadLock lock(server->m_cachedBlocksLock);
|
||||||
|
|
||||||
for (const hash& id : missing_blocks) {
|
for (const hash& id : missing_blocks) {
|
||||||
|
auto it = server->m_cachedBlocks.find(id);
|
||||||
|
if (it != server->m_cachedBlocks.end()) {
|
||||||
|
LOGINFO(5, "using cached block for id = " << id);
|
||||||
|
handle_incoming_block_async(it->second);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
const bool result = m_owner->send(this,
|
const bool result = m_owner->send(this,
|
||||||
[this, &id](void* buf)
|
[this, &id](void* buf)
|
||||||
{
|
{
|
||||||
|
|
|
@ -19,11 +19,13 @@
|
||||||
|
|
||||||
#include "tcp_server.h"
|
#include "tcp_server.h"
|
||||||
#include <random>
|
#include <random>
|
||||||
|
#include <unordered_map>
|
||||||
|
|
||||||
namespace p2pool {
|
namespace p2pool {
|
||||||
|
|
||||||
class p2pool;
|
class p2pool;
|
||||||
struct PoolBlock;
|
struct PoolBlock;
|
||||||
|
class BlockCache;
|
||||||
|
|
||||||
static constexpr size_t P2P_BUF_SIZE = 128 * 1024;
|
static constexpr size_t P2P_BUF_SIZE = 128 * 1024;
|
||||||
static constexpr size_t PEER_LIST_RESPONSE_MAX_PEERS = 16;
|
static constexpr size_t PEER_LIST_RESPONSE_MAX_PEERS = 16;
|
||||||
|
@ -45,6 +47,9 @@ public:
|
||||||
explicit P2PServer(p2pool *pool);
|
explicit P2PServer(p2pool *pool);
|
||||||
~P2PServer();
|
~P2PServer();
|
||||||
|
|
||||||
|
void add_cached_block(const PoolBlock& block);
|
||||||
|
void store_in_cache(const PoolBlock& block);
|
||||||
|
|
||||||
void connect_to_peers(const std::string& peer_list);
|
void connect_to_peers(const std::string& peer_list);
|
||||||
void on_connect_failed(bool is_v6, const raw_ip& ip, int port) override;
|
void on_connect_failed(bool is_v6, const raw_ip& ip, int port) override;
|
||||||
|
|
||||||
|
@ -89,7 +94,7 @@ public:
|
||||||
bool on_peer_list_request(const uint8_t* buf);
|
bool on_peer_list_request(const uint8_t* buf);
|
||||||
bool on_peer_list_response(const uint8_t* buf) const;
|
bool on_peer_list_response(const uint8_t* buf) const;
|
||||||
|
|
||||||
bool handle_incoming_block_async();
|
bool handle_incoming_block_async(PoolBlock* block);
|
||||||
void handle_incoming_block(p2pool* pool, PoolBlock& block, const uint32_t reset_counter, std::vector<hash>& missing_blocks);
|
void handle_incoming_block(p2pool* pool, PoolBlock& block, const uint32_t reset_counter, std::vector<hash>& missing_blocks);
|
||||||
void post_handle_incoming_block(const uint32_t reset_counter, std::vector<hash>& missing_blocks);
|
void post_handle_incoming_block(const uint32_t reset_counter, std::vector<hash>& missing_blocks);
|
||||||
|
|
||||||
|
@ -113,11 +118,17 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
p2pool* m_pool;
|
p2pool* m_pool;
|
||||||
|
BlockCache* m_cache;
|
||||||
|
bool m_cacheLoaded;
|
||||||
|
|
||||||
|
uv_rwlock_t m_cachedBlocksLock;
|
||||||
|
std::unordered_map<hash, PoolBlock*> m_cachedBlocks;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static void on_timer(uv_timer_t* timer) { reinterpret_cast<P2PServer*>(timer->data)->on_timer(); }
|
static void on_timer(uv_timer_t* timer) { reinterpret_cast<P2PServer*>(timer->data)->on_timer(); }
|
||||||
void on_timer();
|
void on_timer();
|
||||||
|
|
||||||
|
void flush_cache();
|
||||||
void download_missing_blocks();
|
void download_missing_blocks();
|
||||||
void update_peer_connections();
|
void update_peer_connections();
|
||||||
void update_peer_list();
|
void update_peer_list();
|
||||||
|
|
|
@ -320,7 +320,7 @@ void p2pool::update_block_template_async()
|
||||||
const int err = uv_queue_work(uv_default_loop(), req,
|
const int err = uv_queue_work(uv_default_loop(), req,
|
||||||
[](uv_work_t* req)
|
[](uv_work_t* req)
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_add(1);
|
bkg_jobs_tracker.start("p2pool::update_block_template_async");
|
||||||
|
|
||||||
p2pool* pool = reinterpret_cast<p2pool*>(req->data);
|
p2pool* pool = reinterpret_cast<p2pool*>(req->data);
|
||||||
pool->m_blockTemplate->update(pool->m_minerData, *pool->m_mempool, &pool->m_params->m_wallet);
|
pool->m_blockTemplate->update(pool->m_minerData, *pool->m_mempool, &pool->m_params->m_wallet);
|
||||||
|
@ -330,7 +330,7 @@ void p2pool::update_block_template_async()
|
||||||
{
|
{
|
||||||
delete req;
|
delete req;
|
||||||
|
|
||||||
num_running_jobs.fetch_sub(1);
|
bkg_jobs_tracker.stop("p2pool::update_block_template_async");
|
||||||
});
|
});
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
|
@ -685,13 +685,7 @@ int p2pool::run()
|
||||||
|
|
||||||
m_stopped = true;
|
m_stopped = true;
|
||||||
|
|
||||||
const int32_t k = num_running_jobs.load();
|
bkg_jobs_tracker.wait();
|
||||||
if (k != 0) {
|
|
||||||
LOGINFO(1, "waiting for " << k << " background jobs to finish");
|
|
||||||
while (num_running_jobs != 0) {
|
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(1));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
delete m_stratumServer;
|
delete m_stratumServer;
|
||||||
delete m_p2pServer;
|
delete m_p2pServer;
|
||||||
|
|
|
@ -133,7 +133,7 @@ void RandomX_Hasher::set_seed_async(const hash& seed)
|
||||||
uv_queue_work(uv_default_loop(), &work->req,
|
uv_queue_work(uv_default_loop(), &work->req,
|
||||||
[](uv_work_t* req)
|
[](uv_work_t* req)
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_add(1);
|
bkg_jobs_tracker.start("RandomX_Hasher::set_seed_async");
|
||||||
Work* work = reinterpret_cast<Work*>(req->data);
|
Work* work = reinterpret_cast<Work*>(req->data);
|
||||||
if (!work->pool->stopped()) {
|
if (!work->pool->stopped()) {
|
||||||
work->hasher->set_seed(work->seed);
|
work->hasher->set_seed(work->seed);
|
||||||
|
@ -142,7 +142,7 @@ void RandomX_Hasher::set_seed_async(const hash& seed)
|
||||||
[](uv_work_t* req, int)
|
[](uv_work_t* req, int)
|
||||||
{
|
{
|
||||||
delete reinterpret_cast<Work*>(req->data);
|
delete reinterpret_cast<Work*>(req->data);
|
||||||
num_running_jobs.fetch_sub(1);
|
bkg_jobs_tracker.stop("RandomX_Hasher::set_seed_async");
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -166,7 +166,7 @@ void RandomX_Hasher::set_old_seed_async(const hash& seed)
|
||||||
uv_queue_work(uv_default_loop(), &work->req,
|
uv_queue_work(uv_default_loop(), &work->req,
|
||||||
[](uv_work_t* req)
|
[](uv_work_t* req)
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_add(1);
|
bkg_jobs_tracker.start("RandomX_Hasher::set_old_seed_async");
|
||||||
Work* work = reinterpret_cast<Work*>(req->data);
|
Work* work = reinterpret_cast<Work*>(req->data);
|
||||||
if (!work->pool->stopped()) {
|
if (!work->pool->stopped()) {
|
||||||
work->hasher->set_old_seed(work->seed);
|
work->hasher->set_old_seed(work->seed);
|
||||||
|
@ -175,7 +175,7 @@ void RandomX_Hasher::set_old_seed_async(const hash& seed)
|
||||||
[](uv_work_t* req, int)
|
[](uv_work_t* req, int)
|
||||||
{
|
{
|
||||||
delete reinterpret_cast<Work*>(req->data);
|
delete reinterpret_cast<Work*>(req->data);
|
||||||
num_running_jobs.fetch_sub(1);
|
bkg_jobs_tracker.stop("RandomX_Hasher::set_old_seed_async");
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -621,8 +621,6 @@ void SideChain::print_status()
|
||||||
"\nYour shares = " << our_blocks_in_window << " blocks (+" << our_uncles_in_window << " uncles, " << our_orphans << " orphans)"
|
"\nYour shares = " << our_blocks_in_window << " blocks (+" << our_uncles_in_window << " uncles, " << our_orphans << " orphans)"
|
||||||
"\nNext payout = " << log::XMRAmount(m_pool->block_template().next_payout())
|
"\nNext payout = " << log::XMRAmount(m_pool->block_template().next_payout())
|
||||||
);
|
);
|
||||||
|
|
||||||
LOGINFO(0, "background jobs running: " << num_running_jobs.load());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SideChain::split_reward(uint64_t reward, const std::vector<MinerShare>& shares, std::vector<uint64_t>& rewards)
|
bool SideChain::split_reward(uint64_t reward, const std::vector<MinerShare>& shares, std::vector<uint64_t>& rewards)
|
||||||
|
@ -1323,6 +1321,11 @@ void SideChain::update_depths(PoolBlock* block)
|
||||||
block = blocks_to_update.back();
|
block = blocks_to_update.back();
|
||||||
blocks_to_update.pop_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);
|
auto it = m_blocksById.find(block->m_parent);
|
||||||
if (it != m_blocksById.end()) {
|
if (it != m_blocksById.end()) {
|
||||||
if (it->second->m_sidechainHeight + 1 != block->m_sidechainHeight) {
|
if (it->second->m_sidechainHeight + 1 != block->m_sidechainHeight) {
|
||||||
|
|
|
@ -357,7 +357,7 @@ void StratumServer::on_blobs_ready()
|
||||||
|
|
||||||
void StratumServer::on_share_found(uv_work_t* req)
|
void StratumServer::on_share_found(uv_work_t* req)
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_add(1);
|
bkg_jobs_tracker.start("StratumServer::on_share_found");
|
||||||
|
|
||||||
SubmittedShare* share = reinterpret_cast<SubmittedShare*>(req->data);
|
SubmittedShare* share = reinterpret_cast<SubmittedShare*>(req->data);
|
||||||
StratumClient* client = share->m_client;
|
StratumClient* client = share->m_client;
|
||||||
|
@ -427,7 +427,7 @@ void StratumServer::on_after_share_found(uv_work_t* req, int /*status*/)
|
||||||
ON_SCOPE_LEAVE(
|
ON_SCOPE_LEAVE(
|
||||||
[share]()
|
[share]()
|
||||||
{
|
{
|
||||||
num_running_jobs.fetch_sub(1);
|
bkg_jobs_tracker.stop("StratumServer::on_share_found");
|
||||||
|
|
||||||
MutexLock lock(share->m_server->m_submittedSharesPoolLock);
|
MutexLock lock(share->m_server->m_submittedSharesPoolLock);
|
||||||
share->m_server->m_submittedSharesPool.push_back(share);
|
share->m_server->m_submittedSharesPool.push_back(share);
|
||||||
|
|
86
src/util.cpp
86
src/util.cpp
|
@ -18,6 +18,9 @@
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
#include "uv_util.h"
|
#include "uv_util.h"
|
||||||
|
#include <map>
|
||||||
|
#include <thread>
|
||||||
|
#include <chrono>
|
||||||
|
|
||||||
#ifndef _WIN32
|
#ifndef _WIN32
|
||||||
#include <sched.h>
|
#include <sched.h>
|
||||||
|
@ -27,8 +30,6 @@ static constexpr char log_category_prefix[] = "Util ";
|
||||||
|
|
||||||
namespace p2pool {
|
namespace p2pool {
|
||||||
|
|
||||||
std::atomic<int32_t> num_running_jobs{ 0 };
|
|
||||||
|
|
||||||
MinerCallbackHandler::~MinerCallbackHandler() {}
|
MinerCallbackHandler::~MinerCallbackHandler() {}
|
||||||
|
|
||||||
void panic()
|
void panic()
|
||||||
|
@ -131,4 +132,85 @@ void uv_rwlock_init_checked(uv_rwlock_t* lock)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct BackgroundJobTracker::Impl
|
||||||
|
{
|
||||||
|
Impl() { uv_mutex_init_checked(&m_lock); }
|
||||||
|
~Impl() { uv_mutex_destroy(&m_lock); }
|
||||||
|
|
||||||
|
void start(const char* name)
|
||||||
|
{
|
||||||
|
MutexLock lock(m_lock);
|
||||||
|
|
||||||
|
auto it = m_jobs.insert({ name, 1 });
|
||||||
|
if (!it.second) {
|
||||||
|
++it.first->second;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void stop(const char* name)
|
||||||
|
{
|
||||||
|
MutexLock lock(m_lock);
|
||||||
|
|
||||||
|
auto it = m_jobs.find(name);
|
||||||
|
if (it == m_jobs.end()) {
|
||||||
|
LOGWARN(1, "background job " << name << " is not running, but stop() was called");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
--it->second;
|
||||||
|
if (it->second <= 0) {
|
||||||
|
m_jobs.erase(it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void wait()
|
||||||
|
{
|
||||||
|
do {
|
||||||
|
bool is_empty = true;
|
||||||
|
{
|
||||||
|
MutexLock lock(m_lock);
|
||||||
|
is_empty = m_jobs.empty();
|
||||||
|
for (const auto& job : m_jobs) {
|
||||||
|
LOGINFO(1, "waiting for " << job.second << " \"" << job.first << "\" jobs to finish");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_empty) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::this_thread::sleep_for(std::chrono::milliseconds(1000));
|
||||||
|
} while (1);
|
||||||
|
}
|
||||||
|
|
||||||
|
uv_mutex_t m_lock;
|
||||||
|
std::map<const char*, int32_t> m_jobs;
|
||||||
|
};
|
||||||
|
|
||||||
|
BackgroundJobTracker::BackgroundJobTracker() : m_impl(new Impl())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
BackgroundJobTracker::~BackgroundJobTracker()
|
||||||
|
{
|
||||||
|
delete m_impl;
|
||||||
|
}
|
||||||
|
|
||||||
|
void BackgroundJobTracker::start(const char* name)
|
||||||
|
{
|
||||||
|
m_impl->start(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void BackgroundJobTracker::stop(const char* name)
|
||||||
|
{
|
||||||
|
m_impl->stop(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
void BackgroundJobTracker::wait()
|
||||||
|
{
|
||||||
|
m_impl->wait();
|
||||||
|
}
|
||||||
|
|
||||||
|
BackgroundJobTracker bkg_jobs_tracker;
|
||||||
|
|
||||||
} // namespace p2pool
|
} // namespace p2pool
|
||||||
|
|
17
src/util.h
17
src/util.h
|
@ -98,7 +98,22 @@ template<typename T, size_t N> FORCEINLINE constexpr size_t array_size(T(&)[N])
|
||||||
|
|
||||||
void make_thread_background();
|
void make_thread_background();
|
||||||
|
|
||||||
extern std::atomic<int32_t> num_running_jobs;
|
class BackgroundJobTracker
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
BackgroundJobTracker();
|
||||||
|
~BackgroundJobTracker();
|
||||||
|
|
||||||
|
void start(const char* name);
|
||||||
|
void stop(const char* name);
|
||||||
|
void wait();
|
||||||
|
|
||||||
|
private:
|
||||||
|
struct Impl;
|
||||||
|
Impl* m_impl;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern BackgroundJobTracker bkg_jobs_tracker;
|
||||||
|
|
||||||
} // namespace p2pool
|
} // namespace p2pool
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue