ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/Coro/Event/Event.pm
(Generate patch)

Comparing Coro/Event/Event.pm (file contents):
Revision 1.11 by root, Thu Nov 21 13:08:06 2002 UTC vs.
Revision 1.45 by root, Fri Dec 29 11:37:49 2006 UTC

6 6
7 use Coro; 7 use Coro;
8 use Coro::Event; 8 use Coro::Event;
9 9
10 sub keyboard : Coro { 10 sub keyboard : Coro {
11 my $w = Coro::Event->io(fd => *STDIN, poll => 'r'); 11 my $w = Coro::Event->io(fd => \*STDIN, poll => 'r');
12 while() { 12 while() {
13 print "cmd> "; 13 print "cmd> ";
14 my $ev = $w->next; my $cmd = <STDIN>; 14 my $ev = $w->next; my $cmd = <STDIN>;
15 unloop unless $cmd ne ""; 15 unloop unless $cmd ne "";
16 print "data> "; 16 print "data> ";
30(I<flavour>) (see L<Event>). The only difference between these and the 30(I<flavour>) (see L<Event>). The only difference between these and the
31watcher constructors from Event is that you do not specify a callback 31watcher constructors from Event is that you do not specify a callback
32function - it will be managed by this module. 32function - it will be managed by this module.
33 33
34Your application should just create all necessary coroutines and then call 34Your application should just create all necessary coroutines and then call
35Coro::Event->main. 35Coro::Event::loop.
36
37Please note that even programs or modules (such as
38L<Coro::Handle|Coro::Handle>) that use "traditional"
39event-based/continuation style will run more efficient with this module
40then when using only Event.
41
42=head1 WARNING
43
44Please note that Event does not support coroutines or threads. That
45means that you B<MUST NOT> block in an event callback. Again: In Event
46callbacks, you I<must never ever> call a Coroutine fucntion that blocks
47the current coroutine.
48
49While this seems to work superficially, it will eventually cause memory
50corruption.
51
52=head1 SEMANTICS
53
54Whenever Event blocks (e.g. in a call to C<one_event>, C<loop< etc.),
55this module cede's to all other coroutines with the same or higher
56priority. When any coroutines of lower priority are ready, it will not
57block but run one of them and then check for events.
58
59The effect is that coroutines with the same or higher priority than
60the blocking coroutine will keep Event from checking for events, while
61coroutines with lower priority are being run, but Event checks for new
62events after every cede.
63
64=head1 FUNCTIONS
36 65
37=over 4 66=over 4
38 67
39=cut 68=cut
40 69
41package Coro::Event; 70package Coro::Event;
42 71
43no warnings qw(uninitialized); 72no warnings;
44 73
45use Carp; 74use Carp;
75no warnings;
46 76
47use Coro; 77use Coro;
78use Coro::Timer;
48use Event qw(loop unloop); # we are re-exporting this, cooool! 79use Event qw(loop unloop); # we are re-exporting this, cooool!
49 80
81use XSLoader;
82
50use base 'Exporter'; 83use base Exporter::;
51 84
52@EXPORT = qw(loop unloop sweep reschedule); 85our @EXPORT = qw(loop unloop sweep);
53 86
54BEGIN { 87BEGIN {
55 $VERSION = 0.6; 88 our $VERSION = 1.9;
56 89
57 local $^W = 0; # avoid redefine warning for Coro::ready 90 local $^W = 0; # avoid redefine warning for Coro::ready;
58 require XSLoader;
59 XSLoader::load Coro::Event, $VERSION; 91 XSLoader::load __PACKAGE__, $VERSION;
60} 92}
61 93
62=item $w = Coro::Event->flavour(args...) 94=item $w = Coro::Event->flavour (args...)
63 95
64Create and return a watcher of the given type. 96Create and return a watcher of the given type.
65 97
66Examples: 98Examples:
67 99
74 106
75Return the next event of the event queue of the watcher. 107Return the next event of the event queue of the watcher.
76 108
77=cut 109=cut
78 110
79=item do_flavour(args...) 111=item do_flavour args...
80 112
81Create a watcher of the given type and immediately call it's next 113Create a watcher of the given type and immediately call it's next
82method. This is less efficient then calling the constructor once and the 114method. This is less efficient then calling the constructor once and the
83next method often, but it does save typing sometimes. 115next method often, but it does save typing sometimes.
84 116
96 128
97 shift eq Coro::Event:: 129 shift eq Coro::Event::
98 or croak "event constructor \"Coro::Event->$flavour\" must be called as a static method"; 130 or croak "event constructor \"Coro::Event->$flavour\" must be called as a static method";
99 131
100 my $w = $new->($class, 132 my $w = $new->($class,
101 desc => $flavour, 133 desc => $flavour,
102 @_, 134 @_,
103 parked => 1, 135 parked => 1,
104 ); 136 );
137
105 _install_std_cb($w, $type); 138 _install_std_cb $w, $type;
106 bless $w, $class; # reblessing due to broken Event 139
140 # reblessing due to Event being broken
141 bless $w, $class
107 }; 142 };
108 *{ $flavour } = $coronew; 143 *{ $flavour } = $coronew;
109 *{"do_$flavour"} = sub { 144 *{"do_$flavour"} = sub {
110 unshift @_, Coro::Event::; 145 unshift @_, Coro::Event::;
111 my $e = (&$coronew)->next; 146 @_ = &$coronew;
112 $e->cancel; # $e = $e->w->cancel ($e == $e->w!) 147 &Coro::schedule while &_next;
113 $e; 148 $_[0]->cancel;
149 &_event
114 }; 150 };
115} 151}
116 152
117# double calls to avoid stack-cloning ;() 153# do schedule in perl to avoid forcing a stack allocation.
118# is about 10% slower, though. 154# this is about 10% slower, though.
119sub next($) { 155sub next($) {
120 &Coro::schedule if &_next; $_[0]; 156 &Coro::schedule while &_next;
157
158 &_event
121} 159}
122 160
123sub Coro::Event::w { $_[0] } 161sub Coro::Event::w { $_[0] }
124sub Coro::Event::prio { $_[0]{Coro::Event}[3] } 162sub Coro::Event::prio { $_[0]{Coro::Event}[3] }
125sub Coro::Event::hits { $_[0]{Coro::Event}[4] } 163sub Coro::Event::hits { $_[0]{Coro::Event}[4] }
137into the Event dispatcher. 175into the Event dispatcher.
138 176
139=cut 177=cut
140 178
141sub sweep { 179sub sweep {
142 Event::one_event(0); # for now 180 Event::one_event 0; # for now
143} 181}
144 182
145=item $result = loop([$timeout]) 183=item $result = loop([$timeout])
146 184
147This is the version of C<loop> you should use instead of C<Event::loop> 185This is the version of C<loop> you should use instead of C<Event::loop>
148when using this module - it will ensure correct scheduling in the presence 186when using this module - it will ensure correct scheduling in the presence
149of events. 187of events.
150 188
151=begin comment
152
153Unlike loop's counterpart it is not an error when no watchers are active -
154loop silently returns in this case, as if unloop(undef) were called.
155
156=end comment
157
158=cut
159
160# no longer do something special - it's done internally now
161
162#sub loop(;$) {
163# #local $Coro::idle = $Coro::current;
164# #Coro::schedule; # become idle task, which is implicitly ready
165# &Event::loop;
166#}
167
168=item unloop([$result]) 189=item unloop([$result])
169 190
170Same as Event::unloop (provided here for your convinience only). 191Same as Event::unloop (provided here for your convinience only).
171 192
172=cut 193=cut
173 194
174$Coro::idle = new Coro sub { 195$Coro::idle = \&Event::one_event; # inefficient
175 while () {
176 Event::one_event; # inefficient
177 Coro::schedule;
178 }
179};
180
181# provide hooks for Coro::Timer
182
183package Coro::Timer;
184
185unless ($override) {
186 $override = 1;
187 *_new_timer = sub {
188 Event->timer(at => $_[0], cb => $_[1]);
189 };
190}
191 196
1921; 1971;
193 198
199=back
200
194=head1 AUTHOR 201=head1 AUTHOR
195 202
196 Marc Lehmann <pcg@goof.com> 203 Marc Lehmann <schmorp@schmorp.de>
197 http://www.goof.com/pcg/marc/ 204 http://home.schmorp.de/
198 205
199=cut 206=cut
200 207

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines