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

Comparing Coro/Coro.pm (file contents):
Revision 1.131 by root, Thu Sep 20 12:24:42 2007 UTC vs.
Revision 1.137 by root, Wed Sep 26 19:26:48 2007 UTC

155 # call all destruction callbacks 155 # call all destruction callbacks
156 $_->(@{$self->{status}}) 156 $_->(@{$self->{status}})
157 for @{(delete $self->{destroy_cb}) || []}; 157 for @{(delete $self->{destroy_cb}) || []};
158} 158}
159 159
160sub _do_trace {
161 $current->{_trace_cb}->();
162}
163
160# this coroutine is necessary because a coroutine 164# this coroutine is necessary because a coroutine
161# cannot destroy itself. 165# cannot destroy itself.
162my @destroy; 166my @destroy;
163my $manager; 167my $manager;
164 168
165$manager = new Coro sub { 169$manager = new Coro sub {
166 $current->desc ("[coro manager]");
167
168 while () { 170 while () {
169 (shift @destroy)->_cancel 171 (shift @destroy)->_cancel
170 while @destroy; 172 while @destroy;
171 173
172 &schedule; 174 &schedule;
173 } 175 }
174}; 176};
175 177$manager->desc ("[coro manager]");
176$manager->prio (PRIO_MAX); 178$manager->prio (PRIO_MAX);
177 179
178# static methods. not really. 180# static methods. not really.
179 181
180=back 182=back
226The pool size is limited to 8 idle coroutines (this can be adjusted by 228The pool size is limited to 8 idle coroutines (this can be adjusted by
227changing $Coro::POOL_SIZE), and there can be as many non-idle coros as 229changing $Coro::POOL_SIZE), and there can be as many non-idle coros as
228required. 230required.
229 231
230If you are concerned about pooled coroutines growing a lot because a 232If you are concerned about pooled coroutines growing a lot because a
231single C<async_pool> used a lot of stackspace you can e.g. C<async_pool { 233single C<async_pool> used a lot of stackspace you can e.g. C<async_pool
232terminate }> once per second or so to slowly replenish the pool. 234{ terminate }> once per second or so to slowly replenish the pool. In
235addition to that, when the stacks used by a handler grows larger than 16kb
236(adjustable with $Coro::POOL_RSS) it will also exit.
233 237
234=cut 238=cut
235 239
236our $POOL_SIZE = 8; 240our $POOL_SIZE = 8;
237our $MAX_POOL_RSS = 64 * 1024; 241our $POOL_RSS = 16 * 1024;
238our @pool; 242our @async_pool;
239 243
240sub pool_handler { 244sub pool_handler {
245 my $cb;
246
241 while () { 247 while () {
242 $current->{desc} = "[async_pool]";
243
244 eval { 248 eval {
245 my ($cb, @arg) = @{ delete $current->{_invoke} or return }; 249 while () {
246 $cb->(@arg); 250 _pool_1 $cb;
251 &$cb;
252 _pool_2 $cb;
253 &schedule;
254 }
247 }; 255 };
256
257 last if $@ eq "\3terminate\2\n";
248 warn $@ if $@; 258 warn $@ if $@;
249
250 last if @pool >= $POOL_SIZE || $current->rss >= $MAX_POOL_RSS;
251
252 push @pool, $current;
253 $current->{desc} = "[async_pool idle]";
254 $current->save (Coro::State::SAVE_DEF);
255 $current->prio (0);
256 schedule;
257 } 259 }
258} 260}
259 261
260sub async_pool(&@) { 262sub async_pool(&@) {
261 # this is also inlined into the unlock_scheduler 263 # this is also inlined into the unlock_scheduler
262 my $coro = (pop @pool) || new Coro \&pool_handler;; 264 my $coro = (pop @async_pool) || new Coro \&pool_handler;
263 265
264 $coro->{_invoke} = [@_]; 266 $coro->{_invoke} = [@_];
265 $coro->ready; 267 $coro->ready;
266 268
267 $coro 269 $coro
533 535
534# we create a special coro because we want to cede, 536# we create a special coro because we want to cede,
535# to reduce pressure on the coro pool (because most callbacks 537# to reduce pressure on the coro pool (because most callbacks
536# return immediately and can be reused) and because we cannot cede 538# return immediately and can be reused) and because we cannot cede
537# inside an event callback. 539# inside an event callback.
538our $unblock_scheduler = async { 540our $unblock_scheduler = new Coro sub {
539 $current->desc ("[unblock_sub scheduler]");
540 while () { 541 while () {
541 while (my $cb = pop @unblock_queue) { 542 while (my $cb = pop @unblock_queue) {
542 # this is an inlined copy of async_pool 543 # this is an inlined copy of async_pool
543 my $coro = (pop @pool or new Coro \&pool_handler); 544 my $coro = (pop @async_pool) || new Coro \&pool_handler;
544 545
545 $coro->{_invoke} = $cb; 546 $coro->{_invoke} = $cb;
546 $coro->ready; 547 $coro->ready;
547 cede; # for short-lived callbacks, this reduces pressure on the coro pool 548 cede; # for short-lived callbacks, this reduces pressure on the coro pool
548 } 549 }
549 schedule; # sleep well 550 schedule; # sleep well
550 } 551 }
551}; 552};
553$unblock_scheduler->desc ("[unblock_sub scheduler]");
552 554
553sub unblock_sub(&) { 555sub unblock_sub(&) {
554 my $cb = shift; 556 my $cb = shift;
555 557
556 sub { 558 sub {

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines