Blame src/verto-libev.c

Packit 086201
/*
Packit 086201
 * Copyright 2011 Red Hat, Inc.
Packit 086201
 *
Packit 086201
 * Permission is hereby granted, free of charge, to any person
Packit 086201
 * obtaining a copy of this software and associated documentation files
Packit 086201
 * (the "Software"), to deal in the Software without restriction,
Packit 086201
 * including without limitation the rights to use, copy, modify, merge,
Packit 086201
 * publish, distribute, sublicense, and/or sell copies of the Software,
Packit 086201
 * and to permit persons to whom the Software is furnished to do so,
Packit 086201
 * subject to the following conditions:
Packit 086201
 *
Packit 086201
 * The above copyright notice and this permission notice shall be
Packit 086201
 * included in all copies or substantial portions of the Software.
Packit 086201
 *
Packit 086201
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
Packit 086201
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
Packit 086201
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
Packit 086201
 * NONINFRINGEMENT.  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
Packit 086201
 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
Packit 086201
 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
Packit 086201
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
Packit 086201
 * SOFTWARE.
Packit 086201
 */
Packit 086201
Packit 086201
#include <stdlib.h>
Packit 086201
#include <string.h>
Packit 086201
#include <errno.h>
Packit 086201
Packit 086201
#include <verto-libev.h>
Packit 086201
#define VERTO_MODULE_TYPES
Packit 086201
typedef struct ev_loop verto_mod_ctx;
Packit 086201
typedef ev_watcher verto_mod_ev;
Packit 086201
#include <verto-module.h>
Packit 086201
Packit 086201
static verto_mod_ctx *
Packit 086201
libev_ctx_new(void)
Packit 086201
{
Packit 086201
    return ev_loop_new(EVFLAG_AUTO);
Packit 086201
}
Packit 086201
Packit 086201
static verto_mod_ctx *
Packit 086201
libev_ctx_default(void)
Packit 086201
{
Packit 086201
    return ev_default_loop(EVFLAG_AUTO);
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_ctx_free(verto_mod_ctx *ctx)
Packit 086201
{
Packit 086201
    if (ctx != EV_DEFAULT)
Packit 086201
        ev_loop_destroy(ctx);
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_ctx_run(verto_mod_ctx *ctx)
Packit 086201
{
Packit 086201
    ev_run(ctx, 0);
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_ctx_run_once(verto_mod_ctx *ctx)
Packit 086201
{
Packit 086201
    ev_run(ctx, EVRUN_ONCE);
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_ctx_break(verto_mod_ctx *ctx)
Packit 086201
{
Packit 086201
    ev_break(ctx, EVBREAK_ONE);
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_ctx_reinitialize(verto_mod_ctx *ctx)
Packit 086201
{
Packit 086201
    ev_loop_fork(ctx);
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_callback(EV_P_ ev_watcher *w, int revents)
Packit 086201
{
Packit 086201
    verto_ev_flag state = VERTO_EV_FLAG_NONE;
Packit 086201
Packit 086201
#if EV_MULTIPLICITY
Packit 086201
    /* Match the check in ev.h, which doesn't mark this unused */
Packit 086201
    (void) EV_A;
Packit 086201
#endif
Packit 086201
Packit 086201
    if (verto_get_type(w->data)== VERTO_EV_TYPE_CHILD)
Packit 086201
        verto_set_proc_status(w->data, ((ev_child*) w)->rstatus);
Packit 086201
Packit 086201
    if (revents & EV_READ)
Packit 086201
        state |= VERTO_EV_FLAG_IO_READ;
Packit 086201
    if (revents & EV_WRITE)
Packit 086201
        state |= VERTO_EV_FLAG_IO_WRITE;
Packit 086201
    if (revents & EV_ERROR)
Packit 086201
        state |= VERTO_EV_FLAG_IO_ERROR;
Packit 086201
Packit 086201
    verto_set_fd_state(w->data, state);
Packit 086201
    verto_fire(w->data);
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_ctx_set_flags(verto_mod_ctx *ctx, const verto_ev *ev,
Packit 086201
                    verto_mod_ev *evpriv)
Packit 086201
{
Packit 086201
    if (verto_get_type(ev) == VERTO_EV_TYPE_IO) {
Packit 086201
        int events = EV_NONE;
Packit 086201
Packit 086201
        if (verto_get_flags(ev) & VERTO_EV_FLAG_IO_READ)
Packit 086201
            events |= EV_READ;
Packit 086201
        if (verto_get_flags(ev) & VERTO_EV_FLAG_IO_WRITE)
Packit 086201
            events |= EV_WRITE;
Packit 086201
Packit 086201
        ev_io_stop(ctx, (ev_io*) evpriv);
Packit 086201
        ev_io_set(((ev_io*) evpriv), verto_get_fd(ev), events);
Packit 086201
        ev_io_start(ctx, (ev_io*) evpriv);
Packit 086201
    }
Packit 086201
}
Packit 086201
Packit 086201
#define setuptype(type, ...) \
Packit 086201
    w.type = malloc(sizeof(ev_ ## type)); \
Packit 086201
    if (w.type) { \
Packit 086201
    	ev_ ## type ## _init(w.type, (EV_CB(type, (*))) __VA_ARGS__); \
Packit 086201
    	ev_ ## type ## _start(ctx, w.type); \
Packit 086201
    } \
Packit 086201
    break
Packit 086201
Packit 086201
static verto_mod_ev *
Packit 086201
libev_ctx_add(verto_mod_ctx *ctx, const verto_ev *ev, verto_ev_flag *flags)
Packit 086201
{
Packit 086201
    union {
Packit 086201
       ev_watcher *watcher;
Packit 086201
       ev_io *io;
Packit 086201
       ev_timer *timer;
Packit 086201
       ev_idle *idle;
Packit 086201
       ev_signal *signal;
Packit 086201
       ev_child *child;
Packit 086201
    } w;
Packit 086201
    ev_tstamp interval;
Packit 086201
Packit 086201
    w.watcher = NULL;
Packit 086201
    *flags |= VERTO_EV_FLAG_PERSIST;
Packit 086201
    switch (verto_get_type(ev)) {
Packit 086201
        case VERTO_EV_TYPE_IO:
Packit 086201
            setuptype(io, libev_callback, verto_get_fd(ev), EV_NONE);
Packit 086201
        case VERTO_EV_TYPE_TIMEOUT:
Packit 086201
            interval = ((ev_tstamp) verto_get_interval(ev)) / 1000.0;
Packit 086201
            setuptype(timer, libev_callback, interval, interval);
Packit 086201
        case VERTO_EV_TYPE_IDLE:
Packit 086201
            setuptype(idle, libev_callback);
Packit 086201
        case VERTO_EV_TYPE_SIGNAL:
Packit 086201
            setuptype(signal, libev_callback, verto_get_signal(ev));
Packit 086201
        case VERTO_EV_TYPE_CHILD:
Packit 086201
            *flags &= ~VERTO_EV_FLAG_PERSIST; /* Child events don't persist */
Packit 086201
            setuptype(child, libev_callback, verto_get_proc(ev), 0);
Packit 086201
        default:
Packit 086201
            break; /* Not supported */
Packit 086201
    }
Packit 086201
Packit 086201
    if (w.watcher) {
Packit 086201
        w.watcher->data = (void*) ev;
Packit 086201
        libev_ctx_set_flags(ctx, ev, w.watcher);
Packit 086201
    }
Packit 086201
    return w.watcher;
Packit 086201
}
Packit 086201
Packit 086201
static void
Packit 086201
libev_ctx_del(verto_mod_ctx *ctx, const verto_ev *ev, verto_mod_ev *evpriv)
Packit 086201
{
Packit 086201
    switch (verto_get_type(ev)) {
Packit 086201
        case VERTO_EV_TYPE_IO:
Packit 086201
            ev_io_stop(ctx, (ev_io*) evpriv);
Packit 086201
            break;
Packit 086201
        case VERTO_EV_TYPE_TIMEOUT:
Packit 086201
            ev_timer_stop(ctx, (ev_timer*) evpriv);
Packit 086201
            break;
Packit 086201
        case VERTO_EV_TYPE_IDLE:
Packit 086201
            ev_idle_stop(ctx, (ev_idle*) evpriv);
Packit 086201
            break;
Packit 086201
        case VERTO_EV_TYPE_SIGNAL:
Packit 086201
            ev_signal_stop(ctx, (ev_signal*) evpriv);
Packit 086201
            break;
Packit 086201
        case VERTO_EV_TYPE_CHILD:
Packit 086201
            ev_child_stop(ctx, (ev_child*) evpriv);
Packit 086201
            break;
Packit 086201
        default:
Packit 086201
            break;
Packit 086201
    }
Packit 086201
Packit 086201
    free(evpriv);
Packit 086201
}
Packit 086201
Packit 086201
VERTO_MODULE(libev, ev_loop_new,
Packit 086201
             VERTO_EV_TYPE_IO |
Packit 086201
             VERTO_EV_TYPE_TIMEOUT |
Packit 086201
             VERTO_EV_TYPE_IDLE |
Packit 086201
             VERTO_EV_TYPE_SIGNAL |
Packit 086201
             VERTO_EV_TYPE_CHILD);
Packit 086201
Packit 086201
verto_ctx *
Packit 086201
verto_convert_libev(struct ev_loop* loop)
Packit 086201
{
Packit 086201
    return verto_convert(libev, 0, loop);
Packit 086201
}