--- EV/EV.xs 2007/12/20 08:18:54 1.92 +++ EV/EV.xs 2007/12/20 09:26:40 1.93 @@ -25,28 +25,30 @@ # include #endif +#define e_loop(w) INT2PTR (struct ev_loop *, SvIVX ((w)->loop)) + #define WFLAG_KEEPALIVE 1 #define UNREF(w) \ if (!((w)->flags & WFLAG_KEEPALIVE) \ && !ev_is_active (w)) \ - ev_unref (w->loop); + ev_unref (e_loop (w)); #define REF(w) \ if (!((w)->flags & WFLAG_KEEPALIVE) \ && ev_is_active (w)) \ - ev_ref (w->loop); + ev_ref (e_loop (w)); #define START(type,w) \ do { \ UNREF (w); \ - ev_ ## type ## _start (w->loop, w); \ + ev_ ## type ## _start (e_loop (w), w); \ } while (0) #define STOP(type,w) \ do { \ REF (w); \ - ev_ ## type ## _stop (w->loop, w); \ + ev_ ## type ## _stop (e_loop (w), w); \ } while (0) #define RESET(type,w,seta) \ @@ -59,9 +61,12 @@ typedef int Signal; +static SV *default_loop_sv; + static struct EVAPI evapi; static HV + *stash_loop, *stash_watcher, *stash_io, *stash_timer, @@ -123,7 +128,7 @@ } static void * -e_new (int size, SV *cb_sv) +e_new (int size, SV *cb_sv, SV *loop) { ev_watcher *w; SV *self = NEWSV (0, size); @@ -134,11 +139,11 @@ ev_init (w, e_cb); - w->loop = EV_DEFAULT; + w->loop = SvREFCNT_inc (SvRV (loop)); w->flags = WFLAG_KEEPALIVE; w->data = 0; w->fh = 0; - w->cb_sv = newSVsv (cb_sv); + w->cb_sv = SvTEMP (cb_sv) && SvREFCNT (cb_sv) == 1 ? SvREFCNT_inc (cb_sv) : newSVsv (cb_sv); w->self = self; return (void *)w; @@ -149,6 +154,7 @@ { ev_watcher *w = (ev_watcher *)w_; + SvREFCNT_dec (w->loop ); w->loop = 0; SvREFCNT_dec (w->fh ); w->fh = 0; SvREFCNT_dec (w->cb_sv); w->cb_sv = 0; SvREFCNT_dec (w->data ); w->data = 0; @@ -357,6 +363,7 @@ for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; ) newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv)); + stash_loop = gv_stashpv ("EV::Loop" , 1); stash_watcher = gv_stashpv ("EV::Watcher" , 1); stash_io = gv_stashpv ("EV::IO" , 1); stash_timer = gv_stashpv ("EV::Timer" , 1); @@ -433,12 +440,17 @@ SV *ev_default_loop (unsigned int flags = ev_supported_backends ()) CODE: { - evapi.default_loop = ev_default_loop (flags); - if (!evapi.default_loop) - XSRETURN_UNDEF; + if (!default_loop_sv) + { + evapi.default_loop = ev_default_loop (flags); + + if (!evapi.default_loop) + XSRETURN_UNDEF; - RETVAL = sv_bless (newRV_noinc (newSViv (PTR2IV (ev_default_loop (flags)))), - gv_stashpv ("EV::Loop::Default", 1)); + default_loop_sv = sv_bless (newRV_noinc (newSViv (PTR2IV (evapi.default_loop))), stash_loop); + } + + RETVAL = newSVsv (default_loop_sv); } OUTPUT: RETVAL @@ -480,7 +492,7 @@ int fd = sv_fileno (fh); CHECK_FD (fh, fd); - RETVAL = e_new (sizeof (ev_io), cb); + RETVAL = e_new (sizeof (ev_io), cb, default_loop_sv); RETVAL->fh = newSVsv (fh); ev_io_set (RETVAL, fd, events); if (!ix) START (io, RETVAL); @@ -494,7 +506,7 @@ INIT: CHECK_REPEAT (repeat); CODE: - RETVAL = e_new (sizeof (ev_timer), cb); + RETVAL = e_new (sizeof (ev_timer), cb, default_loop_sv); ev_timer_set (RETVAL, after, repeat); if (!ix) START (timer, RETVAL); OUTPUT: @@ -508,7 +520,7 @@ CODE: { ev_periodic *w; - w = e_new (sizeof (ev_periodic), cb); + w = e_new (sizeof (ev_periodic), cb, default_loop_sv); w->fh = SvTRUE (reschedule_cb) ? newSVsv (reschedule_cb) : 0; ev_periodic_set (w, at, interval, w->fh ? e_periodic_cb : 0); RETVAL = e_bless ((ev_watcher *)w, stash_periodic); @@ -525,7 +537,7 @@ Signal signum = sv_signum (signal); CHECK_SIG (signal, signum); - RETVAL = e_new (sizeof (ev_signal), cb); + RETVAL = e_new (sizeof (ev_signal), cb, default_loop_sv); ev_signal_set (RETVAL, signum); if (!ix) START (signal, RETVAL); } @@ -536,7 +548,7 @@ ALIAS: idle_ns = 1 CODE: - RETVAL = e_new (sizeof (ev_idle), cb); + RETVAL = e_new (sizeof (ev_idle), cb, default_loop_sv); ev_idle_set (RETVAL); if (!ix) START (idle, RETVAL); OUTPUT: @@ -546,7 +558,7 @@ ALIAS: prepare_ns = 1 CODE: - RETVAL = e_new (sizeof (ev_prepare), cb); + RETVAL = e_new (sizeof (ev_prepare), cb, default_loop_sv); ev_prepare_set (RETVAL); if (!ix) START (prepare, RETVAL); OUTPUT: @@ -556,7 +568,7 @@ ALIAS: check_ns = 1 CODE: - RETVAL = e_new (sizeof (ev_check), cb); + RETVAL = e_new (sizeof (ev_check), cb, default_loop_sv); ev_check_set (RETVAL); if (!ix) START (check, RETVAL); OUTPUT: @@ -566,7 +578,7 @@ ALIAS: fork_ns = 1 CODE: - RETVAL = e_new (sizeof (ev_fork), cb); + RETVAL = e_new (sizeof (ev_fork), cb, default_loop_sv); ev_fork_set (RETVAL); if (!ix) START (fork, RETVAL); OUTPUT: @@ -576,7 +588,7 @@ ALIAS: child_ns = 1 CODE: - RETVAL = e_new (sizeof (ev_child), cb); + RETVAL = e_new (sizeof (ev_child), cb, default_loop_sv); ev_child_set (RETVAL, pid); if (!ix) START (child, RETVAL); OUTPUT: @@ -586,13 +598,24 @@ ALIAS: stat_ns = 1 CODE: - RETVAL = e_new (sizeof (ev_stat), cb); + RETVAL = e_new (sizeof (ev_stat), cb, default_loop_sv); RETVAL->fh = newSVsv (path); ev_stat_set (RETVAL, SvPVbyte_nolen (RETVAL->fh), interval); if (!ix) START (stat, RETVAL); OUTPUT: RETVAL +ev_embed *embed (struct ev_loop *loop, SV *cb) + ALIAS: + embed_ns = 1 + CODE: + RETVAL = e_new (sizeof (ev_embed), cb, default_loop_sv); + RETVAL->fh = newSVsv (ST (0)); + ev_embed_set (RETVAL, loop); + if (!ix) START (embed, RETVAL); + OUTPUT: + RETVAL + void once (SV *fh, int events, SV *timeout, SV *cb) CODE: ev_once ( @@ -612,13 +635,13 @@ int ev_is_pending (ev_watcher *w) void ev_invoke (ev_watcher *w, int revents = EV_NONE) - C_ARGS: w->loop, w, revents + C_ARGS: e_loop (w), w, revents int ev_clear_pending (ev_watcher *w) - C_ARGS: w->loop, w + C_ARGS: e_loop (w), w void ev_feed_event (ev_watcher *w, int revents = EV_NONE) - C_ARGS: w->loop, w, revents + C_ARGS: e_loop (w), w, revents int keepalive (ev_watcher *w, int new_value = 0) CODE: @@ -805,7 +828,7 @@ CHECK_REPEAT (w->repeat); CODE: REF (w); - ev_timer_again (w->loop, w); + ev_timer_again (e_loop (w), w); UNREF (w); void DESTROY (ev_timer *w) @@ -840,7 +863,7 @@ void ev_periodic_again (ev_periodic *w) CODE: REF (w); - ev_periodic_again (w->loop, w); + ev_periodic_again (e_loop (w), w); UNREF (w); void DESTROY (ev_periodic *w) @@ -1021,7 +1044,7 @@ ev_statdata *s = ix ? &w->attr : &w->prev; if (ix == 1) - ev_stat_stat (w->loop, w); + ev_stat_stat (e_loop (w), w); else if (!s->st_nlink) errno = ENOENT; @@ -1058,52 +1081,45 @@ } } -#if 0 +MODULE = EV PACKAGE = EV::Embed PREFIX = ev_embed_ -MODULE = EV PACKAGE = EV::HTTP PREFIX = evhttp_ +void ev_embed_start (ev_embed *w) + CODE: + START (embed, w); -BOOT: -{ - HV *stash = gv_stashpv ("EV::HTTP", 1); +void ev_embed_stop (ev_embed *w) + CODE: + STOP (embed, w); - static const struct { - const char *name; - IV iv; - } *civ, const_iv[] = { -# define const_iv(pfx, name) { # name, (IV) pfx ## name }, - const_iv (HTTP_, OK) - const_iv (HTTP_, NOCONTENT) - const_iv (HTTP_, MOVEPERM) - const_iv (HTTP_, MOVETEMP) - const_iv (HTTP_, NOTMODIFIED) - const_iv (HTTP_, BADREQUEST) - const_iv (HTTP_, NOTFOUND) - const_iv (HTTP_, SERVUNAVAIL) - const_iv (EVHTTP_, REQ_OWN_CONNECTION) - const_iv (EVHTTP_, PROXY_REQUEST) - const_iv (EVHTTP_, REQ_GET) - const_iv (EVHTTP_, REQ_POST) - const_iv (EVHTTP_, REQ_HEAD) - const_iv (EVHTTP_, REQUEST) - const_iv (EVHTTP_, RESPONSE) - }; +void DESTROY (ev_embed *w) + CODE: + STOP (embed, w); + e_destroy (w); - for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; ) - newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv)); +void set (ev_embed *w, struct ev_loop *loop) + CODE: +{ + sv_setsv (w->fh, ST (1)); + RESET (embed, w, (w, loop)); } -MODULE = EV PACKAGE = EV::HTTP::Request PREFIX = evhttp_request_ - -#HttpRequest new (SV *klass, SV *cb) - -#void DESTROY (struct evhttp_request *req); - -#endif - - - +MODULE = EV PACKAGE = EV::Loop PREFIX = ev_loop_ +SV *new (SV *klass, unsigned int flags = ev_supported_backends ()) + CODE: +{ + struct ev_loop *loop = ev_loop_new (flags); + if (!loop) + XSRETURN_UNDEF; + RETVAL = sv_bless (newRV_noinc (newSViv (PTR2IV (evapi.default_loop))), stash_loop); +} + OUTPUT: + RETVAL +void DESTROY (struct ev_loop *loop) + CODE: + if (loop != evapi.default_loop) /* global destruction sucks */ + ev_loop_destroy (loop);