--- cvsroot/EV/README 2008/04/16 17:08:29 1.24 +++ cvsroot/EV/README 2008/07/08 18:56:48 1.28 @@ -3,63 +3,70 @@ loop SYNOPSIS - use EV; - - # TIMERS - - my $w = EV::timer 2, 0, sub { - warn "is called after 2s"; - }; - - my $w = EV::timer 2, 2, sub { - warn "is called roughly every 2s (repeat = 2)"; - }; - - undef $w; # destroy event watcher again - - my $w = EV::periodic 0, 60, 0, sub { - warn "is called every minute, on the minute, exactly"; - }; - - # IO - - my $w = EV::io *STDIN, EV::READ, sub { - my ($w, $revents) = @_; # all callbacks receive the watcher and event mask - warn "stdin is readable, you entered: ", ; - }; - - # SIGNALS - - my $w = EV::signal 'QUIT', sub { - warn "sigquit received\n"; - }; - - # CHILD/PID STATUS CHANGES - - my $w = EV::child 666, 0, sub { - my ($w, $revents) = @_; - my $status = $w->rstatus; - }; - - # STAT CHANGES - my $w = EV::stat "/etc/passwd", 10, sub { - my ($w, $revents) = @_; - warn $w->path, " has changed somehow.\n"; - }; - - # MAINLOOP - EV::loop; # loop until EV::unloop is called or all watchers stop - EV::loop EV::LOOP_ONESHOT; # block until at least one event could be handled - EV::loop EV::LOOP_NONBLOCK; # try to handle same events, but do not block + use EV; + + # TIMERS + + my $w = EV::timer 2, 0, sub { + warn "is called after 2s"; + }; + + my $w = EV::timer 2, 2, sub { + warn "is called roughly every 2s (repeat = 2)"; + }; + + undef $w; # destroy event watcher again + + my $w = EV::periodic 0, 60, 0, sub { + warn "is called every minute, on the minute, exactly"; + }; + + # IO + + my $w = EV::io *STDIN, EV::READ, sub { + my ($w, $revents) = @_; # all callbacks receive the watcher and event mask + warn "stdin is readable, you entered: ", ; + }; + + # SIGNALS + + my $w = EV::signal 'QUIT', sub { + warn "sigquit received\n"; + }; + + # CHILD/PID STATUS CHANGES + + my $w = EV::child 666, 0, sub { + my ($w, $revents) = @_; + my $status = $w->rstatus; + }; + + # STAT CHANGES + my $w = EV::stat "/etc/passwd", 10, sub { + my ($w, $revents) = @_; + warn $w->path, " has changed somehow.\n"; + }; + + # MAINLOOP + EV::loop; # loop until EV::unloop is called or all watchers stop + EV::loop EV::LOOP_ONESHOT; # block until at least one event could be handled + EV::loop EV::LOOP_NONBLOCK; # try to handle same events, but do not block DESCRIPTION This module provides an interface to libev (). While the documentation below is comprehensive, one might also consult the documentation of - libev itself () for more subtle - details on watcher semantics or some discussion on the available - backends, or how to force a specific backend with "LIBEV_FLAGS", or just - about in any case because it has much more detailed information. + libev itself () + for more subtle details on watcher semantics or some discussion on the + available backends, or how to force a specific backend with + "LIBEV_FLAGS", or just about in any case because it has much more + detailed information. + + This module is very fast and scalable. It is actually so fast that you + can use it through the AnyEvent module, stay portable to other event + loops (if you don't rely on any watcher types not available through it) + and still be faster than with any other event loop currently supported + in Perl. EVENT LOOPS EV supports multiple event loops: There is a single "default event loop" @@ -76,6 +83,13 @@ For specific programs you can create additional event loops dynamically. + If you want to take avdantage of kqueue (which often works properly for + sockets only) even though the default loop doesn't enable it, you can + *embed* a kqueue loop into the default loop: running the default loop + will then also service the kqueue loop to some extent. See the example + in the section about embed watchers for an example on how to achieve + that. + $loop = new EV::loop [$flags] Create a new event loop as per the specified flags. Please refer to the "ev_loop_new ()" function description in the libev documentation @@ -91,11 +105,20 @@ $loop->loop_fork Must be called after a fork in the child, before entering or continuing the event loop. An alternative is to use - "EV::FLAG_FORKCHECK" which calls this fucntion automatically, at + "EV::FLAG_FORKCHECK" which calls this function automatically, at some performance loss (refer to the libev documentation). + $loop->loop_verify + Calls "ev_verify" to make internal consistency checks (for debugging + libev) and abort the program if any data structures were found to be + corrupted. + $loop = EV::default_loop [$flags] - Return the default loop (which is a singleton object). + Return the default loop (which is a singleton object). Since this + module already creates the default loop with default flags, + specifying flags here will not have any effect unless you destroy + the default loop first, which isn't supported. So in short: don't do + it, and if you break it, you get to keep the pieces. BASIC INTERFACE $EV::DIED @@ -203,10 +226,10 @@ event. For instance, if you want to wait for STDIN to become readable, you would create an EV::io watcher for that: - my $watcher = EV::io *STDIN, EV::READ, sub { - my ($watcher, $revents) = @_; - warn "yeah, STDIN should now be readable without blocking!\n" - }; + my $watcher = EV::io *STDIN, EV::READ, sub { + my ($watcher, $revents) = @_; + warn "yeah, STDIN should now be readable without blocking!\n" + }; All watchers can be active (waiting for events) or inactive (paused). Only active watchers will have their callbacks invoked. All callbacks @@ -318,7 +341,7 @@ my $udp_socket = ... my $udp_watcher = EV::io $udp_socket, EV::READ, sub { ... }; - $1000udp_watcher->keepalive (0); + $udp_watcher->keepalive (0); $loop = $w->loop Return the loop that this watcher is attached to. @@ -451,14 +474,16 @@ first, and the current time as second argument. *This callback MUST NOT stop or destroy this or any other - periodic watcher, ever*. If you need to stop it, return 1e30 and - stop it afterwards. + periodic watcher, ever, and MUST NOT call any event loop + functions or methods*. If you need to stop it, return 1e30 and + stop it afterwards. You may create and start a "EV::prepare" + watcher for this task. It must return the next time to trigger, based on the passed - time value (that is, the lowest time value larger than to the - second argument). It will usually be called just before the - callback will be triggered, but might be called at other times, - too. + time value (that is, the lowest time value larger than or equal + to to the second argument). It will usually be called just + before the callback will be triggered, but might be called at + other times, too. This can be used to create very complex timers, such as a timer that triggers on each midnight, local time (actually 24 hours @@ -752,19 +777,19 @@ In short, this watcher is most useful on BSD systems without working kqueue to still be able to handle a large number of sockets: - my $socket_loop; - - # check wether we use SELECT or POLL _and_ KQUEUE is supported - if ( - (EV::backend & (EV::BACKEND_POLL | EV::BACKEND_SELECT)) - && (EV::supported_backends & EV::embeddable_backends & EV::BACKEND_KQUEUE) - ) { - # use kqueue for sockets - $socket_loop = new EV::Loop EV::BACKEND_KQUEUE | EV::FLAG_NOENV; - } - - # use the default loop otherwise - $socket_loop ||= EV::default_loop; + my $socket_loop; + + # check wether we use SELECT or POLL _and_ KQUEUE is supported + if ( + (EV::backend & (EV::BACKEND_POLL | EV::BACKEND_SELECT)) + && (EV::supported_backends & EV::embeddable_backends & EV::BACKEND_KQUEUE) + ) { + # use kqueue for sockets + $socket_loop = new EV::Loop EV::BACKEND_KQUEUE | EV::FLAG_NOENV; + } + + # use the default loop otherwise + $socket_loop ||= EV::default_loop; $w = EV::embed $otherloop, $callback $w = EV::embed_ns $otherloop, $callback @@ -835,9 +860,10 @@ SEE ALSO EV::ADNS (asynchronous DNS), Glib::EV (makes Glib/Gtk2 use EV as event loop), EV::Glib (embed Glib into EV), Coro::EV (efficient coroutines - with EV), Net::SNMP::EV (asynchronous SNMP). + with EV), Net::SNMP::EV (asynchronous SNMP), AnyEvent for event-loop + agnostic and portable event driven programming. AUTHOR - Marc Lehmann - http://home.schmorp.de/ + Marc Lehmann + http://home.schmorp.de/