/* * This file is part of the Monero P2Pool * Copyright (c) 2021-2024 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 #include #include static_assert(sizeof(in6_addr) == 16, "struct in6_addr has invalid size"); static_assert(sizeof(in_addr) == 4, "struct in_addr has invalid size"); namespace p2pool { struct MutexLock : public nocopy_nomove { explicit FORCEINLINE MutexLock(uv_mutex_t& handle) : m_handle(&handle) { uv_mutex_lock(&handle); } FORCEINLINE ~MutexLock() { uv_mutex_unlock(m_handle); } private: uv_mutex_t* m_handle; }; template struct RWLock; template<> struct RWLock : public nocopy_nomove { explicit FORCEINLINE RWLock(uv_rwlock_t& handle) : m_handle(&handle) { uv_rwlock_rdlock(&handle); } FORCEINLINE ~RWLock() { uv_rwlock_rdunlock(m_handle); } private: uv_rwlock_t* m_handle; }; typedef RWLock ReadLock; template<> struct RWLock : public nocopy_nomove { explicit FORCEINLINE RWLock(uv_rwlock_t& handle) : m_handle(&handle) { uv_rwlock_wrlock(&handle); } FORCEINLINE ~RWLock() { uv_rwlock_wrunlock(m_handle); } private: uv_rwlock_t* m_handle; }; typedef RWLock WriteLock; void uv_cond_init_checked(uv_cond_t* cond); void uv_mutex_init_checked(uv_mutex_t* mutex); void uv_rwlock_init_checked(uv_rwlock_t* lock); void uv_async_init_checked(uv_loop_t* loop, uv_async_t* async, uv_async_cb async_cb); uv_loop_t* uv_default_loop_checked(); typedef Callback::Base UV_LoopCallbackBase; struct UV_LoopUserData { uv_loop_t* m_loop; uv_async_t* m_async; uv_mutex_t m_callbacksLock; std::vector m_callbacks; std::vector m_callbacksToRun; explicit UV_LoopUserData(uv_loop_t* loop) : m_loop(loop) , m_async(new uv_async_t{}) , m_callbacksLock{} , m_callbacks{} , m_callbacksToRun{} { uv_async_init_checked(m_loop, m_async, async_cb); m_async->data = this; uv_mutex_init_checked(&m_callbacksLock); m_callbacks.reserve(2); m_callbacksToRun.reserve(2); } ~UV_LoopUserData() { uv_mutex_destroy(&m_callbacksLock); uv_close(reinterpret_cast(m_async), [](uv_handle_t* h) { delete reinterpret_cast(h); }); for (const UV_LoopCallbackBase* cb : m_callbacks) { delete cb; } } static void async_cb(uv_async_t* h) { UV_LoopUserData* data = reinterpret_cast(h->data); data->m_callbacksToRun.clear(); { MutexLock lock(data->m_callbacksLock); std::swap(data->m_callbacks, data->m_callbacksToRun); } for (UV_LoopCallbackBase* cb : data->m_callbacksToRun) { (*cb)(); delete cb; } } UV_LoopUserData(const UV_LoopUserData&) = delete; UV_LoopUserData& operator=(const UV_LoopUserData&) = delete; }; UV_LoopUserData* GetLoopUserData(uv_loop_t* loop, bool create = true); void DeleteLoopUserData(uv_loop_t* loop); template bool CallOnLoop(uv_loop_t* loop, T&& callback) { UV_LoopUserData* data = GetLoopUserData(loop, false); if (!data) { return false; } UV_LoopCallbackBase* cb = new Callback::Derived(std::move(callback)); { MutexLock lock(data->m_callbacksLock); data->m_callbacks.push_back(cb); } if (uv_async_send(data->m_async) == 0) { return true; } // Clean up after uv_async_send error bool found = false; { MutexLock lock(data->m_callbacksLock); auto it = std::find(data->m_callbacks.begin(), data->m_callbacks.end(), cb); if (it != data->m_callbacks.end()) { found = true; data->m_callbacks.erase(it); } } if (found) { delete cb; } return false; } template void parallel_run(uv_loop_t* loop, T&& callback, bool wait = false) { const uint32_t THREAD_COUNT = std::thread::hardware_concurrency(); // Don't start other threads on single CPU systems if (THREAD_COUNT <= 1) { callback(); return; } // "THREAD_COUNT - 1" because current thread is already running // No more than 8 threads because our UV worker thread pool has 8 threads const uint32_t THREADS_TO_START = std::min(THREAD_COUNT - 1, 8); struct Callback { explicit FORCEINLINE Callback(T&& f) : m_func(std::move(f)) {} Callback& operator=(Callback&&) = delete; T m_func; }; std::shared_ptr cb = std::make_shared(std::move(callback)); struct Work { uv_work_t req; std::shared_ptr cb; }; for (size_t i = 0; i < THREADS_TO_START; ++i) { Work* w = new Work{ {}, cb }; w->req.data = w; const int err = uv_queue_work(loop, &w->req, [](uv_work_t* req) { std::shared_ptr& cb = reinterpret_cast(req->data)->cb; cb->m_func(); cb.reset(); }, [](uv_work_t* req, int) { delete reinterpret_cast(req->data); }); if (err) { delete w; } } if (wait) { cb->m_func(); while (cb.use_count() > 1) { std::this_thread::yield(); } } } } // namespace p2pool