0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2025-01-03 15:41:54 +00:00

Logging: uses writev instead of in-buffer magic

This commit is contained in:
Maria Matejka 2024-01-28 21:42:39 +01:00
parent 0e1bfdd9e0
commit 86ba37d8f1
5 changed files with 150 additions and 61 deletions

View File

@ -163,10 +163,18 @@ typedef struct buffer {
#define LOG_BUFFER_SIZE 2560 #define LOG_BUFFER_SIZE 2560
enum log_buffer_pos {
LBP_TIMESTAMP = 0,
LBP_THREAD_ID,
LBP_CLASS,
LBP_MSG,
LBP__MAX,
LBPP_TERMINAL,
};
typedef struct log_buffer { typedef struct log_buffer {
struct buffer buf; struct buffer buf;
byte *tm_pos; byte *pos[LBP__MAX+1];
byte *msg_pos;
int class; int class;
char block[LOG_BUFFER_SIZE]; char block[LOG_BUFFER_SIZE];
} log_buffer; } log_buffer;

View File

@ -66,7 +66,7 @@ log_file:
this_log->filename = $2; this_log->filename = $2;
} }
| SYSLOG syslog_name { this_log->rf = NULL; new_config->syslog_name = $2; } | SYSLOG syslog_name { this_log->rf = NULL; new_config->syslog_name = $2; }
| STDERR { this_log->rf = &rf_stderr; } | STDERR { this_log->rf = &rf_stderr; this_log->terminal_flag = 1; }
; ;
log_mask: log_mask:

View File

@ -192,36 +192,88 @@ rf_same(struct rfile *a, struct rfile *b)
(a->stat.st_ino == b->stat.st_ino); (a->stat.st_ino == b->stat.st_ino);
} }
int void
rf_write(struct rfile *r, const void *buf, size_t _count) rf_write_crude(struct rfile *r, const char *buf, int sz)
{ {
off_t count = _count; if (r->mapping)
memcpy(r->mapping, buf, sz);
else
write(r->fd, buf, sz);
}
int
rf_writev(struct rfile *r, struct iovec *iov, int iov_count)
{
off_t size = 0;
for (int i = 0; i < iov_count; i++)
size += iov[i].iov_len;
if (r->mapping) if (r->mapping)
{ {
/* Update the pointer */ /* Update the pointer */
off_t target = atomic_fetch_add_explicit(&r->pos, count, memory_order_relaxed) % r->limit; off_t target = atomic_fetch_add_explicit(&r->pos, size, memory_order_relaxed) % r->limit;
/* Take care of wrapping */
if (target + count > r->limit)
{
memcpy(r->mapping, buf + (r->limit - target), target + count - r->limit);
count = r->limit - target;
}
/* Write the line */ /* Write the line */
memcpy(r->mapping + target, buf, count); for (int i = 0; i < iov_count; i++)
{
/* Take care of wrapping; this should really happen only once */
off_t rsz;
while ((rsz = r->limit - target) < (off_t) iov[i].iov_len)
{
memcpy(r->mapping + target, iov[i].iov_base, rsz);
iov[i].iov_base += rsz;
iov[i].iov_len -= rsz;
target = 0;
}
memcpy(r->mapping + target, iov[i].iov_base, iov[i].iov_len);
target += iov[i].iov_len;
}
return 1; return 1;
} }
else if (r->limit && (atomic_fetch_add_explicit(&r->pos, count, memory_order_relaxed) + count > r->limit)) else if (r->limit && (atomic_fetch_add_explicit(&r->pos, size, memory_order_relaxed) + size > r->limit))
{ {
atomic_fetch_sub_explicit(&r->pos, count, memory_order_relaxed); atomic_fetch_sub_explicit(&r->pos, size, memory_order_relaxed);
return 0; return 0;
} }
else else
{ {
while ((write(r->fd, buf, count) < 0) && (errno == EINTR)) while (size > 0)
; {
/* Try to write */
ssize_t e = writev(r->fd, iov, iov_count);
if (e < 0)
if (errno == EINTR)
continue;
else
return 1; /* FIXME: What should we do when we suddenly can't write? */
/* It is expected that we always write the whole bunch at once */
if (e == size)
return 1;
/* Block split should not happen (we write small enough messages)
* but if it happens, let's try to write the rest of the log */
size -= e;
while (e > 0)
{
if ((ssize_t) iov[0].iov_len > e)
{
/* Some bytes are remaining in the first chunk */
iov[0].iov_len -= e;
iov[0].iov_base += e;
break;
}
/* First chunk written completely, get rid of it */
e -= iov[0].iov_len;
iov++;
iov_count--;
ASSERT_DIE(iov_count > 0);
}
}
return 1; return 1;
} }
} }

View File

@ -48,8 +48,6 @@ static DOMAIN(logging) log_domain;
static struct log_channel * _Atomic global_logs; static struct log_channel * _Atomic global_logs;
/* Logging flags to validly prepare logging messages */ /* Logging flags to validly prepare logging messages */
#define LOGGING_TO_TERMINAL 0x1
#define LOGGING_TO_FILE 0x2
static _Atomic uint logging_flags; static _Atomic uint logging_flags;
static _Atomic uint logging_mask; static _Atomic uint logging_mask;
@ -92,7 +90,7 @@ struct log_channel {
off_t limit; /* Log size limit */ off_t limit; /* Log size limit */
_Atomic uint mask; /* Classes to log */ _Atomic uint mask; /* Classes to log */
uint new_mask; /* Pending new mask */ uint new_mask; /* Pending new mask */
uint terminal:1; /* Is terminal */ uint prepare; /* Which message parts to prepare */
}; };
struct log_thread_syncer { struct log_thread_syncer {
@ -174,6 +172,10 @@ log_rotate(struct log_channel *lc)
void void
log_commit(log_buffer *buf) log_commit(log_buffer *buf)
{ {
/* Store the last pointer */
buf->pos[LBP__MAX] = buf->buf.pos;
/* Append the too-long message if too long */
if (buf->buf.pos == buf->buf.end) if (buf->buf.pos == buf->buf.end)
#define TOO_LONG " ... <too long>" #define TOO_LONG " ... <too long>"
memcpy(buf->buf.end - sizeof TOO_LONG, TOO_LONG, sizeof TOO_LONG); memcpy(buf->buf.end - sizeof TOO_LONG, TOO_LONG, sizeof TOO_LONG);
@ -190,18 +192,44 @@ log_commit(log_buffer *buf)
continue; continue;
struct rfile *rf = atomic_load_explicit(&l->rf, memory_order_acquire); struct rfile *rf = atomic_load_explicit(&l->rf, memory_order_acquire);
if (rf && buf->tm_pos) if (rf)
{ {
*buf->buf.pos = '\n'; /* Construct the iovec */
byte *begin = l->terminal ? buf->buf.start : buf->tm_pos; static char terminal_prefix[] = "bird: ",
off_t msg_len = buf->buf.pos - begin + 1; newline[] = "\n";
STATIC_ASSERT(sizeof newline == 2);
struct iovec iov[LBP__MAX+2];
uint iov_count = 0;
if (BIT32_TEST(&l->prepare, LBPP_TERMINAL))
iov[iov_count++] = (struct iovec) {
.iov_base = terminal_prefix,
.iov_len = sizeof terminal_prefix - 1,
};
for (uint p = 0; p < LBP__MAX; p++)
if (BIT32_TEST(&l->prepare, p))
{
off_t sz = buf->pos[p+1] - buf->pos[p];
if (sz > 0)
iov[iov_count++] = (struct iovec) {
.iov_base = buf->pos[p],
.iov_len = sz,
};
}
iov[iov_count++] = (struct iovec) {
.iov_base = newline,
.iov_len = sizeof newline - 1,
};
do { do {
if (rf_write(rf, begin, msg_len)) if (rf_writev(rf, iov, iov_count))
break; break;
log_lock(); log_lock();
rf = atomic_load_explicit(&l->rf, memory_order_acquire); rf = atomic_load_explicit(&l->rf, memory_order_acquire);
if (rf_write(rf, begin, msg_len)) if (rf_writev(rf, iov, iov_count))
{ {
log_unlock(); log_unlock();
break; break;
@ -211,18 +239,15 @@ log_commit(log_buffer *buf)
log_unlock(); log_unlock();
rf = atomic_load_explicit(&l->rf, memory_order_relaxed); rf = atomic_load_explicit(&l->rf, memory_order_relaxed);
} while (!rf_write(rf, begin, msg_len)); } while (!rf_writev(rf, iov, iov_count));
} }
#ifdef HAVE_SYSLOG_H #ifdef HAVE_SYSLOG_H
else else
{ {
*buf->buf.pos = '\0'; syslog(syslog_priorities[buf->class], "%s", buf->pos[LBP_MSG]);
syslog(syslog_priorities[buf->class], "%s", buf->msg_pos);
} }
#endif #endif
} }
buf->msg_pos = buf->tm_pos = NULL;
} }
int buffer_vprint(buffer *buf, const char *fmt, va_list args); int buffer_vprint(buffer *buf, const char *fmt, va_list args);
@ -230,31 +255,35 @@ int buffer_vprint(buffer *buf, const char *fmt, va_list args);
void void
log_prepare(log_buffer *buf, int class) log_prepare(log_buffer *buf, int class)
{ {
buf->class = class;
buf->buf.start = buf->buf.pos = buf->block; buf->buf.start = buf->buf.pos = buf->block;
buf->buf.end = buf->block + sizeof buf->block; buf->buf.end = buf->block + sizeof buf->block;
int lf = atomic_load_explicit(&logging_flags, memory_order_acquire); int lf = atomic_load_explicit(&logging_flags, memory_order_acquire);
if (lf & LOGGING_TO_TERMINAL)
buffer_puts(&buf->buf, "bird: ");
if (lf & LOGGING_TO_FILE) buf->pos[LBP_TIMESTAMP] = buf->buf.pos;
if (BIT32_TEST(&lf, LBP_TIMESTAMP))
{ {
const char *fmt = config ? config->tf_log.fmt1 : "%F %T.%3f"; const char *fmt = config ? config->tf_log.fmt1 : "%F %T.%3f";
buf->tm_pos = buf->buf.pos;
int t = tm_format_real_time(buf->buf.pos, buf->buf.end - buf->buf.pos, fmt, current_real_time()); int t = tm_format_real_time(buf->buf.pos, buf->buf.end - buf->buf.pos, fmt, current_real_time());
if (t) if (t)
buf->buf.pos += t; buf->buf.pos += t;
else else
buffer_puts(&buf->buf, "<time format error>"); buffer_puts(&buf->buf, "<time format error>");
buffer_print(&buf->buf, " [%04x] <%s> ", THIS_THREAD_ID, class_names[class]); *(buf->buf.pos++) = ' ';
} }
else
buf->tm_pos = NULL;
buf->msg_pos = buf->buf.pos; buf->pos[LBP_THREAD_ID] = buf->buf.pos;
buf->class = class; if (BIT32_TEST(&lf, LBP_THREAD_ID))
buffer_print(&buf->buf, "[%04x] ", THIS_THREAD_ID);
buf->pos[LBP_CLASS] = buf->buf.pos;
if (BIT32_TEST(&lf, LBP_CLASS))
buffer_print(&buf->buf, "<%s> ", class_names[class]);
buf->pos[LBP_MSG] = buf->buf.pos;
} }
static void static void
@ -374,7 +403,11 @@ debug(const char *msg, ...)
if (s < 0) if (s < 0)
bug("Extremely long debug output, split it."); bug("Extremely long debug output, split it.");
rf_write(dbg_rf, buf, s); struct iovec i = {
.iov_base = buf,
.iov_len = s,
};
rf_writev(dbg_rf, &i, 1);
} }
va_end(args); va_end(args);
} }
@ -391,7 +424,7 @@ void
debug_safe(const char *msg) debug_safe(const char *msg)
{ {
if (dbg_rf) if (dbg_rf)
rf_write(dbg_rf, msg, strlen(msg)); rf_write_crude(dbg_rf, msg, strlen(msg));
} }
static list * static list *
@ -406,7 +439,7 @@ default_log_list(int initial, const char **syslog_name)
{ {
static struct log_config lc_syslog; static struct log_config lc_syslog;
lc_syslog = (struct log_config){ lc_syslog = (struct log_config){
.mask = ~0 .mask = ~0,
}; };
add_tail(&log_list, &lc_syslog.n); add_tail(&log_list, &lc_syslog.n);
@ -453,7 +486,9 @@ log_switch(int initial, list *logs, const char *new_syslog_name)
/* Create syslog channel */ /* Create syslog channel */
struct log_channel *lc = mb_alloc(log_pool, sizeof *lc); struct log_channel *lc = mb_alloc(log_pool, sizeof *lc);
*lc = (struct log_channel) {}; *lc = (struct log_channel) {
.prepare = BIT32_ALL(LBP_MSG),
};
ASSERT_DIE(NULL == atomic_exchange_explicit(&global_logs, lc, memory_order_release)); ASSERT_DIE(NULL == atomic_exchange_explicit(&global_logs, lc, memory_order_release));
#endif #endif
@ -503,12 +538,7 @@ log_switch(int initial, list *logs, const char *new_syslog_name)
total_mask |= l->mask; total_mask |= l->mask;
/* Merge flags */ /* Merge flags */
flags |= LOGGING_TO_FILE; flags |= ol->prepare;
if (l->terminal_flag)
{
flags |= LOGGING_TO_TERMINAL;
ol->terminal = 1;
}
/* The filehandle is no longer needed */ /* The filehandle is no longer needed */
if ((l->rf != &rf_stderr ) && (l->rf != dbg_rf)) if ((l->rf != &rf_stderr ) && (l->rf != dbg_rf))
@ -552,18 +582,15 @@ log_switch(int initial, list *logs, const char *new_syslog_name)
.rf = l->rf, .rf = l->rf,
.limit = l->limit, .limit = l->limit,
.new_mask = l->mask, .new_mask = l->mask,
.terminal = l->terminal_flag, .prepare = BIT32_ALL(LBP_TIMESTAMP, LBP_THREAD_ID, LBP_CLASS, LBP_MSG) |
(l->terminal_flag ? BIT32_VAL(LBPP_TERMINAL) : 0),
}; };
/* Mask union */
total_mask |= l->mask; total_mask |= l->mask;
/* Message preparation flags */ /* Message preparation flags */
flags |= LOGGING_TO_FILE; flags |= lc->prepare;
if (l->terminal_flag)
{
flags |= LOGGING_TO_TERMINAL;
lc->terminal = 1;
}
/* Now the file handle ownership is transferred to the log channel */ /* Now the file handle ownership is transferred to the log channel */
l->rf = NULL; l->rf = NULL;

View File

@ -13,6 +13,7 @@
#include "lib/io-loop.h" #include "lib/io-loop.h"
#include <sys/socket.h> #include <sys/socket.h>
#include <sys/uio.h>
#include <signal.h> #include <signal.h>
struct pool; struct pool;
@ -123,7 +124,8 @@ enum rf_mode {
struct rfile *rf_open(struct pool *, const char *name, enum rf_mode mode, off_t limit); struct rfile *rf_open(struct pool *, const char *name, enum rf_mode mode, off_t limit);
off_t rf_size(struct rfile *); off_t rf_size(struct rfile *);
int rf_same(struct rfile *, struct rfile *); int rf_same(struct rfile *, struct rfile *);
int rf_write(struct rfile *, const void *, size_t); int rf_writev(struct rfile *, struct iovec *, int);
void rf_write_crude(struct rfile *, const char *, int);
extern struct rfile rf_stderr; extern struct rfile rf_stderr;