summaryrefslogtreecommitdiffstats
path: root/src/buffer.c
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@rambler-co.ru>2009-02-20 18:49:04 +0300
committerVsevolod Stakhov <vsevolod@rambler-co.ru>2009-02-20 18:49:04 +0300
commit606128de4cb33a2727d6609df46ecf0c72006a73 (patch)
tree8a12200c73b1ca0446b6f855beb46546ffbd8a81 /src/buffer.c
parentbcece60fa1bfd4bbb09a64c058835fe3245e1d18 (diff)
downloadrspamd-606128de4cb33a2727d6609df46ecf0c72006a73.tar.gz
rspamd-606128de4cb33a2727d6609df46ecf0c72006a73.zip
* Fix dispatcher timeouts handling
* Add wanna_die flag that can be used in dispatcher's callbacks
Diffstat (limited to 'src/buffer.c')
-rw-r--r--src/buffer.c44
1 files changed, 34 insertions, 10 deletions
diff --git a/src/buffer.c b/src/buffer.c
index 46ef98850..268b6cbb1 100644
--- a/src/buffer.c
+++ b/src/buffer.c
@@ -24,6 +24,7 @@
#include "config.h"
#include "buffer.h"
+#include "main.h"
#define G_DISPATCHER_ERROR dispatcher_error_quark()
@@ -57,7 +58,6 @@ write_buffers (int fd, rspamd_io_dispatcher_t *d)
cur = g_list_next (cur);
continue;
}
-
r = write (fd, buf->pos, BUFREMAIN (buf));
if (r == -1 && errno != EAGAIN) {
if (d->err_callback) {
@@ -70,6 +70,7 @@ write_buffers (int fd, rspamd_io_dispatcher_t *d)
buf->pos += r;
if (BUFREMAIN (buf) != 0) {
/* Continue with this buffer */
+ msg_debug ("write_buffers: wrote %ld bytes of %ld", (long int)r, (long int)buf->data->len);
continue;
}
}
@@ -83,6 +84,9 @@ write_buffers (int fd, rspamd_io_dispatcher_t *d)
}
else if (errno == EAGAIN) {
/* Wait for other event */
+ event_del (d->ev);
+ event_set (d->ev, fd, EV_WRITE, dispatcher_cb, (void *)d);
+ event_add (d->ev, d->tv);
return;
}
cur = g_list_next (cur);
@@ -93,23 +97,37 @@ write_buffers (int fd, rspamd_io_dispatcher_t *d)
g_list_free (d->out_buffers);
d->out_buffers = NULL;
+ msg_debug ("write_buffers: all buffers were written successfully");
+
if (d->write_callback) {
d->write_callback (d->user_data);
+ if (d->wanna_die) {
+ msg_debug ("write_buffers: callback set wanna_die flag, terminating");
+ rspamd_remove_dispatcher (d);
+ return;
+ }
}
-
+
event_del (d->ev);
event_set (d->ev, fd, EV_READ | EV_PERSIST, dispatcher_cb, (void *)d);
event_add (d->ev, d->tv);
}
+ else {
+ /* Plan other write event */
+ event_del (d->ev);
+ event_set (d->ev, fd, EV_WRITE, dispatcher_cb, (void *)d);
+ event_add (d->ev, d->tv);
+ }
}
static void
read_buffers (int fd, rspamd_io_dispatcher_t *d)
{
- ssize_t r, len;
+ ssize_t r;
GError *err;
f_str_t res;
char *c;
+ unsigned int len;
if (d->in_buf == NULL) {
d->in_buf = memory_pool_alloc (d->pool, sizeof (rspamd_buffer_t));
@@ -170,12 +188,17 @@ read_buffers (int fd, rspamd_io_dispatcher_t *d)
res.len = r;
if (d->read_callback) {
d->read_callback (&res, d->user_data);
+ if (d->wanna_die) {
+ msg_debug ("read_buffers: callback set wanna_die flag, terminating");
+ rspamd_remove_dispatcher (d);
+ return;
+ }
if (r < len - 1 && *(c + 1) == '\n') {
r ++;
c ++;
}
/* Move remaining string to begin of buffer (draining) */
- memmove (d->in_buf->data->begin, c, len - r);
+ memmove (d->in_buf->data->begin, c + 1, len - r - 1);
c = d->in_buf->data->begin;
d->in_buf->data->len -= r + 1;
d->in_buf->pos -= r + 1;
@@ -274,7 +297,7 @@ rspamd_create_dispatcher (int fd, enum io_policy policy,
new->ev = memory_pool_alloc0 (new->pool, sizeof (struct event));
new->fd = fd;
- event_set (new->ev, fd, EV_READ | EV_WRITE | EV_PERSIST, dispatcher_cb, (void *)new);
+ event_set (new->ev, fd, EV_WRITE, dispatcher_cb, (void *)new);
event_add (new->ev, new->tv);
return new;
@@ -328,18 +351,19 @@ rspamd_dispatcher_write (rspamd_io_dispatcher_t *d,
rspamd_buffer_t *newbuf;
newbuf = memory_pool_alloc (d->pool, sizeof (rspamd_buffer_t));
- newbuf->data = memory_pool_alloc (d->pool, sizeof (f_str_t));
-
- newbuf->data->begin = memory_pool_alloc (d->pool, len);
+ newbuf->data = fstralloc (d->pool, len);
+
+ /* We need to copy data to temporary internal buffer to avoid using of stack variables */
memcpy (newbuf->data->begin, data, len);
- newbuf->data->size = len;
newbuf->pos = newbuf->data->begin;
+ newbuf->data->len = len;
d->out_buffers = g_list_prepend (d->out_buffers, newbuf);
if (!delayed) {
+ msg_debug ("rspamd_dispatcher_write: plan write event");
event_del (d->ev);
- event_set (d->ev, d->fd, EV_READ | EV_WRITE | EV_PERSIST, dispatcher_cb, (void *)d);
+ event_set (d->ev, d->fd, EV_WRITE, dispatcher_cb, (void *)d);
event_add (d->ev, d->tv);
}
}