aboutsummaryrefslogtreecommitdiffstats
path: root/src/events.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/events.c')
-rw-r--r--src/events.c173
1 files changed, 173 insertions, 0 deletions
diff --git a/src/events.c b/src/events.c
new file mode 100644
index 000000000..9092e85c9
--- /dev/null
+++ b/src/events.c
@@ -0,0 +1,173 @@
+/*
+ * Copyright (c) 2009, Rambler media
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY Rambler media ''AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL Rambler BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include "config.h"
+#include "main.h"
+#include "events.h"
+
+
+struct rspamd_async_session*
+new_async_session (memory_pool_t *pool, event_finalizer_t fin, void *user_data)
+{
+ struct rspamd_async_session *new;
+
+ new = memory_pool_alloc (pool, sizeof (struct rspamd_async_session));
+ new->pool = pool;
+ new->fin = fin;
+ new->user_data = user_data;
+ new->wanna_die = FALSE;
+ new->events = g_queue_new ();
+
+ memory_pool_add_destructor (pool, (pool_destruct_func)g_queue_free, new->events);
+
+ return new;
+}
+
+void
+register_async_event (struct rspamd_async_session *session, event_finalizer_t fin, void *user_data, gboolean forced)
+{
+ struct rspamd_async_event *new, *ev;
+ GList *cur;
+
+ if (session == NULL) {
+ msg_info ("register_async_event: session is NULL");
+ return;
+ }
+
+ if (forced) {
+ /* For forced events try first to increase its reference */
+ cur = session->events->head;
+ while (cur) {
+ ev = cur->data;
+ if (ev->forced && ev->fin == fin) {
+ ev->ref ++;
+ return;
+ }
+ cur = g_list_next (cur);
+ }
+ }
+
+ new = memory_pool_alloc (session->pool, sizeof (struct rspamd_async_event));
+ new->fin = fin;
+ new->user_data = user_data;
+ new->forced = forced;
+ new->ref = 1;
+ g_queue_push_head (session->events, new);
+}
+
+void
+remove_forced_event (struct rspamd_async_session *session, event_finalizer_t fin)
+{
+ struct rspamd_async_event *ev;
+ GList *cur;
+
+ if (session == NULL) {
+ msg_info ("remove_forced_event: session is NULL");
+ return;
+ }
+
+ cur = session->events->head;
+ while (cur) {
+ ev = cur->data;
+ if (ev->forced && ev->fin == fin) {
+ ev->ref --;
+ if (ev->ref == 0) {
+ g_queue_delete_link (session->events, cur);
+ }
+ break;
+ }
+ cur = g_list_next (cur);
+ }
+
+ if (session->wanna_die && session->fin != NULL && g_queue_get_length (session->events) == 0) {
+ /* Call session destroy after all forced events are ready */
+ session->fin (session->user_data);
+ }
+}
+
+void
+remove_normal_event (struct rspamd_async_session *session, event_finalizer_t fin, void *ud)
+{
+ struct rspamd_async_event *ev;
+ GList *cur;
+
+ if (session == NULL) {
+ msg_info ("remove_forced_event: session is NULL");
+ return;
+ }
+
+ cur = session->events->head;
+ while (cur) {
+ ev = cur->data;
+ if (ev->fin == fin && ev->user_data == ud && !ev->forced) {
+ g_queue_delete_link (session->events, cur);
+ if (ev->fin) {
+ ev->fin (ev->user_data);
+ }
+ break;
+ }
+ cur = g_list_next (cur);
+ }
+}
+
+gboolean
+destroy_session (struct rspamd_async_session *session)
+{
+ struct rspamd_async_event *ev;
+ GList *cur, *tmp;
+
+ if (session == NULL) {
+ msg_info ("destroy_session: session is NULL");
+ return FALSE;
+ }
+
+ session->wanna_die = TRUE;
+
+ cur = session->events->head;
+
+ while (cur) {
+ ev = cur->data;
+ if (!ev->forced) {
+ if (ev->fin != NULL) {
+ ev->fin (ev->user_data);
+ }
+ tmp = cur;
+ cur = g_list_next (cur);
+ g_queue_delete_link (session->events, tmp);
+ }
+ else {
+ /* Do nothing with forced callbacks */
+ cur = g_list_next (cur);
+ }
+ }
+
+ if (g_queue_get_length (session->events) == 0) {
+ if (session->fin != NULL) {
+ session->fin (session->user_data);
+ }
+ return TRUE;
+ }
+
+ return FALSE;
+}