uv: fix performance issue

unix implementation of uv_async_t has been wasting cpu cycles for nothing since 1.29.0 release
implement efficient callback scheduling for linux
This commit is contained in:
cohcho 2020-10-12 04:09:09 +00:00
parent f85efd163c
commit 65fa1d9bf3
5 changed files with 110 additions and 0 deletions

View file

@ -130,6 +130,9 @@ else()
src/base/io/json/Json_unix.cpp src/base/io/json/Json_unix.cpp
src/base/kernel/Platform_unix.cpp src/base/kernel/Platform_unix.cpp
) )
list(APPEND SOURCES_BASE
src/base/io/Async.cpp
)
endif() endif()

78
src/base/io/Async.cpp Normal file
View file

@ -0,0 +1,78 @@
#include "base/io/Async.h"
#if defined(XMRIG_UV_PERFORMANCE_BUG)
#include <sys/eventfd.h>
#include <sys/poll.h>
#include <unistd.h>
#include <cstdlib>
namespace xmrig {
uv_async_t::~uv_async_t()
{
close(m_fd);
}
static void on_schedule(uv_poll_t *handle, int status, int events)
{
static uint64_t val;
uv_async_t *async = reinterpret_cast<uv_async_t *>(handle);
for (;;) {
int r = read(async->m_fd, &val, sizeof(val));
if (r == sizeof(val))
continue;
if (r != -1)
break;
if (errno == EAGAIN || errno == EWOULDBLOCK)
break;
if (errno == EINTR)
continue;
abort();
}
if (async->m_cb) {
(*async->m_cb)(async);
}
}
int uv_async_init(uv_loop_t *loop, uv_async_t *async, uv_async_cb cb)
{
int fd = eventfd(0, EFD_CLOEXEC | EFD_NONBLOCK);
if (fd < 0) {
return uv_translate_sys_error(errno);
}
uv_poll_init(loop, (uv_poll_t *)async, fd);
uv_poll_start((uv_poll_t *)async, POLLIN, on_schedule);
async->m_cb = cb;
async->m_fd = fd;
return 0;
}
int uv_async_send(uv_async_t *async)
{
static const uint64_t val = 1;
int r;
do {
r = write(async->m_fd, &val, sizeof(val));
}
while (r == -1 && errno == EINTR);
if (r == sizeof(val) || (r == 1 && (errno == EAGAIN || errno == EWOULDBLOCK))) {
return 0;
}
abort();
}
} // namespace xmrig
#endif

27
src/base/io/Async.h Normal file
View file

@ -0,0 +1,27 @@
#pragma once
#include <uv.h>
// since 2019.05.16, Version 1.29.0 (Stable)
#if (UV_VERSION_MAJOR >= 1) && (UV_VERSION_MINOR >= 29) && defined(__linux__)
#define XMRIG_UV_PERFORMANCE_BUG
namespace xmrig {
struct uv_async_t: uv_poll_t
{
typedef void (*uv_async_cb)(uv_async_t* handle);
~uv_async_t();
int m_fd = -1;
uv_async_cb m_cb = nullptr;
};
using uv_async_cb = uv_async_t::uv_async_cb;
extern int uv_async_init(uv_loop_t *loop, uv_async_t *async, uv_async_cb cb);
extern int uv_async_send(uv_async_t *async);
} // namespace xmrig
#endif

View file

@ -28,6 +28,7 @@
#define XMRIG_RX_QUEUE_H #define XMRIG_RX_QUEUE_H
#include "base/io/Async.h"
#include "base/tools/Object.h" #include "base/tools/Object.h"
#include "crypto/common/HugePagesInfo.h" #include "crypto/common/HugePagesInfo.h"
#include "crypto/rx/RxConfig.h" #include "crypto/rx/RxConfig.h"

View file

@ -24,6 +24,7 @@
#include "net/JobResults.h" #include "net/JobResults.h"
#include "base/io/Async.h"
#include "base/io/log/Log.h" #include "base/io/log/Log.h"
#include "base/tools/Handle.h" #include "base/tools/Handle.h"
#include "base/tools/Object.h" #include "base/tools/Object.h"