Thread-safe FileLogWriter

This commit is contained in:
SChernykh 2024-03-04 08:45:22 +01:00
parent 48fa095e3e
commit 688d4f5ee1
2 changed files with 76 additions and 16 deletions

View file

@ -22,7 +22,6 @@
#include <cassert> #include <cassert>
#include <uv.h>
namespace xmrig { namespace xmrig {
@ -40,6 +39,32 @@ static void fsWriteCallback(uv_fs_t *req)
} // namespace xmrig } // namespace xmrig
xmrig::FileLogWriter::FileLogWriter()
{
init();
}
xmrig::FileLogWriter::FileLogWriter(const char* fileName)
{
init();
open(fileName);
}
xmrig::FileLogWriter::~FileLogWriter()
{
uv_close(reinterpret_cast<uv_handle_t*>(&m_flushAsync), nullptr);
uv_mutex_destroy(&m_buffersLock);
}
void xmrig::FileLogWriter::init()
{
uv_mutex_init(&m_buffersLock);
uv_async_init(uv_default_loop(), &m_flushAsync, on_flush);
m_flushAsync.data = this;
}
bool xmrig::FileLogWriter::open(const char *fileName) bool xmrig::FileLogWriter::open(const char *fileName)
{ {
assert(fileName != nullptr); assert(fileName != nullptr);
@ -77,11 +102,12 @@ bool xmrig::FileLogWriter::write(const char *data, size_t size)
uv_buf_t buf = uv_buf_init(new char[size], size); uv_buf_t buf = uv_buf_init(new char[size], size);
memcpy(buf.base, data, size); memcpy(buf.base, data, size);
auto req = new uv_fs_t; uv_mutex_lock(&m_buffersLock);
req->data = buf.base;
uv_fs_write(uv_default_loop(), req, m_file, &buf, 1, m_pos, fsWriteCallback); m_buffers.emplace_back(buf);
m_pos += size; uv_async_send(&m_flushAsync);
uv_mutex_unlock(&m_buffersLock);
return true; return true;
} }
@ -89,18 +115,38 @@ bool xmrig::FileLogWriter::write(const char *data, size_t size)
bool xmrig::FileLogWriter::writeLine(const char *data, size_t size) bool xmrig::FileLogWriter::writeLine(const char *data, size_t size)
{ {
const uv_buf_t buf[2] = { if (!isOpen()) {
uv_buf_init(new char[size], size), return false;
uv_buf_init(const_cast<char *>(m_endl), sizeof(m_endl) - 1) }
};
memcpy(buf[0].base, data, size); constexpr size_t N = sizeof(m_endl) - 1;
auto req = new uv_fs_t; uv_buf_t buf = uv_buf_init(new char[size + N], size + N);
req->data = buf[0].base; memcpy(buf.base, data, size);
memcpy(buf.base + size, m_endl, N);
uv_fs_write(uv_default_loop(), req, m_file, buf, 2, m_pos, fsWriteCallback); uv_mutex_lock(&m_buffersLock);
m_pos += (buf[0].len + buf[1].len);
m_buffers.emplace_back(buf);
uv_async_send(&m_flushAsync);
uv_mutex_unlock(&m_buffersLock);
return true; return true;
} }
void xmrig::FileLogWriter::flush()
{
uv_mutex_lock(&m_buffersLock);
for (uv_buf_t buf : m_buffers) {
uv_fs_t* req = new uv_fs_t;
req->data = buf.base;
uv_fs_write(uv_default_loop(), req, m_file, &buf, 1, m_pos, fsWriteCallback);
m_pos += buf.len;
}
m_buffers.clear();
uv_mutex_unlock(&m_buffersLock);
}

View file

@ -22,6 +22,8 @@
#include <cstddef> #include <cstddef>
#include <cstdint> #include <cstdint>
#include <vector>
#include <uv.h>
namespace xmrig { namespace xmrig {
@ -30,8 +32,10 @@ namespace xmrig {
class FileLogWriter class FileLogWriter
{ {
public: public:
FileLogWriter() = default; FileLogWriter();
FileLogWriter(const char *fileName) { open(fileName); } FileLogWriter(const char* fileName);
~FileLogWriter();
inline bool isOpen() const { return m_file >= 0; } inline bool isOpen() const { return m_file >= 0; }
inline int64_t pos() const { return m_pos; } inline int64_t pos() const { return m_pos; }
@ -49,6 +53,16 @@ private:
int m_file = -1; int m_file = -1;
int64_t m_pos = 0; int64_t m_pos = 0;
uv_mutex_t m_buffersLock;
std::vector<uv_buf_t> m_buffers;
uv_async_t m_flushAsync;
void init();
static void on_flush(uv_async_t* async) { reinterpret_cast<FileLogWriter*>(async->data)->flush(); }
void flush();
}; };