/* * This file is part of the Monero P2Pool * Copyright (c) 2021-2023 SChernykh * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, version 3. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #pragma once #ifdef _MSC_VER #pragma warning(push) #pragma warning(disable : 4623 5026 5027) #endif #define ROBIN_HOOD_MALLOC(size) p2pool::malloc_hook(size) #define ROBIN_HOOD_CALLOC(count, size) p2pool::calloc_hook((count), (size)) #define ROBIN_HOOD_FREE(ptr) p2pool::free_hook(ptr) #include "robin_hood.h" #ifdef _MSC_VER #pragma warning(pop) #endif namespace p2pool { #define P2POOL_VERSION_MAJOR 2 #define P2POOL_VERSION_MINOR 7 extern const char* VERSION; template struct not_implemented { enum { value = 0 }; }; struct nocopy_nomove { nocopy_nomove() = default; nocopy_nomove(const nocopy_nomove&) = delete; nocopy_nomove(nocopy_nomove&&) = delete; nocopy_nomove& operator=(const nocopy_nomove&) = delete; nocopy_nomove& operator=(nocopy_nomove&&) = delete; }; template struct ScopeGuard { explicit FORCEINLINE ScopeGuard(T&& handler) : m_handler(std::move(handler)) {} FORCEINLINE ~ScopeGuard() { m_handler(); } T m_handler; // Disable copying/moving of ScopeGuard objects // We can't declare copy constructor as explicitly deleted because of copy elision semantics // Just leave it without definition and it'll fail when linking if someone tries to copy a ScopeGuard object ScopeGuard(const ScopeGuard&); private: ScopeGuard& operator=(const ScopeGuard&) = delete; ScopeGuard& operator=(ScopeGuard&&) = delete; }; template FORCEINLINE ScopeGuard on_scope_leave(T&& handler) { return ScopeGuard(std::move(handler)); } #define ON_SCOPE_LEAVE(...) auto CONCAT(scope_guard_, __LINE__) = on_scope_leave(__VA_ARGS__); struct MinerCallbackHandler { virtual ~MinerCallbackHandler() = 0; virtual void handle_tx(TxMempoolData& tx) = 0; virtual void handle_miner_data(MinerData& data) = 0; virtual void handle_chain_main(ChainMain& data, const char* extra) = 0; }; template static FORCEINLINE bool from_hex(char c, T& out_value) { if ('0' <= c && c <= '9') { out_value = static_cast(c - '0'); return true; } if ('a' <= c && c <= 'f') { out_value = static_cast((c - 'a') + 10); return true; } if ('A' <= c && c <= 'F') { out_value = static_cast((c - 'A') + 10); return true; } return false; } template struct abs_helper {}; template struct abs_helper { static FORCEINLINE T value(T x) { return x; } }; template struct abs_helper { static FORCEINLINE T value(T x) { return (x >= 0) ? x : -x; } }; template FORCEINLINE T abs(T x) { return abs_helper::value>::value(x); } template FORCEINLINE void writeVarint(T value, U&& callback) { while (value >= 0x80) { callback(static_cast((value & 0x7F) | 0x80)); value >>= 7; } callback(static_cast(value)); } template FORCEINLINE void writeVarint(T value, std::vector& out) { writeVarint(value, [&out](uint8_t b) { out.emplace_back(b); }); } template const uint8_t* readVarint(const uint8_t* data, const uint8_t* data_end, T& b) { uint64_t result = 0; int k = 0; while (data < data_end) { if (k >= static_cast(sizeof(T)) * 8) { return nullptr; } const uint64_t cur_byte = *(data++); result |= (cur_byte & 0x7F) << k; k += 7; if ((cur_byte & 0x80) == 0) { b = result; return data; } } return nullptr; } template FORCEINLINE T read_unaligned(const T* p) { static_assert(std::is_trivially_copyable::value, "T must be a trivially copyable type"); T result; memcpy(&result, p, sizeof(T)); return result; } template FORCEINLINE constexpr size_t array_size(T(&)[N]) { return N; } template FORCEINLINE constexpr size_t array_size(T(U::*)[N]) { return N; } [[noreturn]] void panic_stop(const char* message); #define STR(X) #X #define STR2(X) STR(X) #define PANIC_STOP(...) panic_stop(__FILE__ ":" STR2(__LINE__)) void make_thread_background(); class BackgroundJobTracker : public nocopy_nomove { public: BackgroundJobTracker(); ~BackgroundJobTracker(); template FORCEINLINE void start(const char (&name)[N]) { start_internal(name); } template FORCEINLINE void stop (const char (&name)[N]) { stop_internal (name); } void wait(); void print_status(); private: void start_internal(const char* name); void stop_internal(const char* name); struct Impl; Impl* m_impl; }; extern BackgroundJobTracker bkg_jobs_tracker; #define BACKGROUND_JOB_START(x) do { bkg_jobs_tracker.start(#x); } while (0) #define BACKGROUND_JOB_STOP(x) do { bkg_jobs_tracker.stop(#x); } while (0) void set_main_thread(); bool is_main_thread(); extern bool disable_resolve_host; bool resolve_host(std::string& host, bool& is_v6); template using unordered_map = robin_hood::detail::Table, std::equal_to>; template using unordered_set = robin_hood::detail::Table, std::equal_to>; // Fills the whole initial MT19937-64 state with non-deterministic random numbers struct RandomDeviceSeed { using result_type = std::random_device::result_type; static_assert(sizeof(result_type) >= 4, "result_type must have at least 32 bits"); template static void generate(T begin, T end) { std::random_device rd; for (T i = begin; i != end; ++i) { *i = rd(); } } static RandomDeviceSeed instance; }; FORCEINLINE uint64_t xorshift64star(uint64_t x) { x ^= x >> 12; x ^= x << 25; x ^= x >> 27; return x * 0x2545F4914F6CDD1DULL; } FORCEINLINE uint64_t seconds_since_epoch() { using namespace std::chrono; return duration_cast(steady_clock::now().time_since_epoch()).count(); } uint64_t bsr_reference(uint64_t x); #ifdef HAVE_BUILTIN_CLZLL #define bsr(x) (63 - __builtin_clzll(x)) #elif defined HAVE_BITSCANREVERSE64 #pragma intrinsic(_BitScanReverse64) FORCEINLINE uint64_t bsr(uint64_t x) { unsigned long index; _BitScanReverse64(&index, x); return index; } #else #define bsr bsr_reference #endif bool str_to_ip(bool is_v6, const char* ip, raw_ip& result); bool is_localhost(const std::string& host); } // namespace p2pool void memory_tracking_start(); void memory_tracking_stop(); void p2pool_usage(); namespace robin_hood { template<> struct hash { FORCEINLINE size_t operator()(const p2pool::hash& value) const noexcept { return hash_bytes(value.h, p2pool::HASH_SIZE); } }; template struct hash> { FORCEINLINE size_t operator()(const std::array& value) const noexcept { return hash_bytes(value.data(), N); } }; template<> struct hash { FORCEINLINE size_t operator()(const p2pool::raw_ip& value) const noexcept { return hash_bytes(value.data, sizeof(value.data)); } }; template<> struct hash> { FORCEINLINE size_t operator()(const std::pair& value) const noexcept { static_assert(sizeof(value) == sizeof(uint64_t) * 2, "Invalid std::pair size"); return hash_bytes(&value, sizeof(value)); } }; } // namespace robin_hood