X-Git-Url: http://git.meshlink.io/?a=blobdiff_plain;f=timeeventq.c;fp=timeeventq.c;h=b3dd897f38d2db9200b8a46892773cf8b23d1794;hb=0781d5363fb6fd723a2316fc7558aef6439b2f71;hp=0000000000000000000000000000000000000000;hpb=d6e2dbabccb08970da991e6d2b0fda7a56d83e6f;p=catta diff --git a/timeeventq.c b/timeeventq.c new file mode 100644 index 0000000..b3dd897 --- /dev/null +++ b/timeeventq.c @@ -0,0 +1,144 @@ +#include "timeeventq.h" +#include "util.h" + +static gint compare(gconstpointer _a, gconstpointer _b) { + const flxTimeEvent *a = _a, *b = _b; + + return flx_timeval_compare(&a->expiry, &b->expiry); +} + +static gboolean prepare_func(GSource *source, gint *timeout) { + flxTimeEventQueue *q = (flxTimeEventQueue*) source; + flxTimeEvent *e; + GTimeVal now; + + g_assert(source); + g_assert(timeout); + + if (!q->prioq->root) { + *timeout = -1; + return FALSE; + } + + e = q->prioq->root->data; + g_assert(e); + + g_source_get_current_time(source, &now); + + if (flx_timeval_compare(&now, &e->expiry) >= 0) { + *timeout = -1; + return TRUE; + } + + *timeout = (gint) (flx_timeval_diff(&e->expiry, &now)/1000); + + return FALSE; +} + +static gboolean check_func(GSource *source) { + flxTimeEventQueue *q = (flxTimeEventQueue*) source; + flxTimeEvent *e; + GTimeVal now; + + g_assert(source); + + if (!q->prioq->root) + return FALSE; + + e = q->prioq->root->data; + g_assert(e); + + g_source_get_current_time(source, &now); + + return flx_timeval_compare(&now, &e->expiry) >= 0; +} + +static gboolean dispatch_func(GSource *source, GSourceFunc callback, gpointer user_data) { + flxTimeEventQueue *q = (flxTimeEventQueue*) source; + GTimeVal now; + + g_assert(source); + + g_source_get_current_time(source, &now); + + while (q->prioq->root) { + flxTimeEvent *e = q->prioq->root->data; + + if (flx_timeval_compare(&now, &e->expiry) < 0) + break; + + g_assert(e->callback); + e->callback(e, e->userdata); + } + + return TRUE; +} + +flxTimeEventQueue* flx_time_event_queue_new(GMainContext *context) { + flxTimeEventQueue *q; + + static GSourceFuncs source_funcs = { + prepare_func, + check_func, + dispatch_func, + NULL, + NULL, + NULL + }; + + q = (flxTimeEventQueue*) g_source_new(&source_funcs, sizeof(flxTimeEventQueue)); + q->prioq = flx_prio_queue_new(compare); + + g_source_attach(&q->source, context); + + return q; +} + +void flx_time_event_queue_free(flxTimeEventQueue *q) { + g_assert(q); + + while (q->prioq->root) + flx_time_event_queue_remove(q, q->prioq->root->data); + flx_prio_queue_free(q->prioq); + + g_source_destroy(&q->source); + g_source_unref(&q->source); +} + +flxTimeEvent* flx_time_event_queue_add(flxTimeEventQueue *q, const GTimeVal *timeval, void (*callback)(flxTimeEvent *e, void *userdata), void *userdata) { + flxTimeEvent *e; + + g_assert(q); + g_assert(timeval); + g_assert(callback); + g_assert(userdata); + + e = g_new(flxTimeEvent, 1); + e->queue = q; + e->expiry = *timeval; + e->callback = callback; + e->userdata = userdata; + + e->node = flx_prio_queue_put(q->prioq, e); + + return e; +} + +void flx_time_event_queue_remove(flxTimeEventQueue *q, flxTimeEvent *e) { + g_assert(q); + g_assert(e); + g_assert(e->queue == q); + + flx_prio_queue_remove(q->prioq, e->node); + g_free(e); +} + +void flx_time_event_update(flxTimeEventQueue *q, flxTimeEvent *e, const GTimeVal *timeval) { + g_assert(q); + g_assert(e); + g_assert(e->queue == q); + + e->expiry = *timeval; + + flx_prio_queue_shuffle(q->prioq, e->node); +}