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

Comparing Coro/myhttpd/access.pl (file contents):
Revision 1.13 by root, Thu Nov 29 02:30:04 2001 UTC vs.
Revision 1.29 by root, Wed Jan 23 03:07:05 2002 UTC

1package transferqueue; 1package transferqueue;
2
3my @reserve = (
4 [ 1_200_000, 2],
5 [ 8_000_000, 1],
6 [ 75_000_000, 1],
7);
2 8
3sub new { 9sub new {
4 my $class = shift; 10 my $class = shift;
5 bless { 11 my $self = bless {
6 conns => $_[0], 12 slots => $_[0],
13 lastspb => 0,
14 avgspb => 0,
7 }, $class; 15 }, $class;
16 $self->{reschedule} = Event->timer(
17 after => 10,
18 interval => 3,
19 cb => sub { $self->wake_next },
20 );
21 $self;
8} 22}
9 23
10sub start_transfer { 24sub start_transfer {
11 my $self = shift; 25 my $self = shift;
26 my $size = $_[0];
12 27
13 my $trans = bless [ $self, $Coro::current ], transfer::; 28 my $transfer = bless {
14 Scalar::Util::weaken($trans->[0]); 29 queue => $self,
30 time => $::NOW,
31 size => $size,
32 coro => $Coro::current,
33 started => 0,
34 }, transfer::;
15 35
16 push @{$self->{wait}}, $trans; 36 push @{$self->{wait}}, $transfer;
17 Scalar::Util::weaken($self->{wait}[-1]);
18 37
19 if (--$self->{conns} >= 0) {
20 $self->wake_next; 38 $self->wake_next;
21 }
22 39
23 $trans; 40 $transfer;
41}
42
43sub sort {
44 my @queue = grep $_, @{$_[0]{wait}};
45
46 $_->{spb} = ($::NOW-$_->{time}) / ($_->{size} || 1) for @queue;
47
48 $_[0]{wait} = [sort { $b->{spb} <=> $a->{spb} } @queue];
49
50 Scalar::Util::weaken $_ for @{$_[0]{wait}};
24} 51}
25 52
26sub wake_next { 53sub wake_next {
27 my $self = shift; 54 my $self = shift;
28 55
29 return unless $self->{conns} >= 0; 56 $self->sort;
30 57
31 if (@{$self->{wait}}) { 58 while (@{$self->{wait}}) {
32 while() { 59 my $size = $self->{wait}[0]{size};
60 my $min = 0;
61 for (@reserve) {
62 last if $size <= $_->[0];
63 $min += $_->[1];
64 }
65 last unless $self->{slots} > $min;
33 my $transfer = shift @{$self->{wait}}; 66 my $transfer = shift @{$self->{wait}};
34 if ($transfer) { 67 $self->{lastspb} = $transfer->{spb};
68 $self->{avgspb} = $self->{avgspb} * 0.99 + $transfer->{spb} * 0.01;
69 $self->{started}++;
35 $transfer->wake; 70 $transfer->wake;
36 last; 71 last;
37 }
38 }
39 } 72 }
40} 73}
41 74
42sub waiters { 75sub waiters {
43 map $_->[1], @{$_[0]{wait}}; 76 $_[0]->sort;
77 @{$_[0]{wait}};
78}
79
80sub DESTROY {
81 my $self = shift;
82
83 $self->{reschedule}->cancel;
44} 84}
45 85
46package transfer; 86package transfer;
47 87
48use Coro::Timer (); 88use Coro::Timer ();
49 89
90sub wake {
91 my $self = shift;
92
93 $self->{alloc} = 1;
94 $self->{queue}{slots}--;
95 $self->{wake} and $self->{wake}->ready;
96}
97
50sub try { 98sub try {
51 my $self = shift; 99 my $self = shift;
100
101 $self->{alloc} || do {
52 my $timeout = Coro::Timer::timeout $_[0]; 102 my $timeout = Coro::Timer::timeout $_[0];
103 local $self->{wake} = $self->{coro};
53 104
54 Coro::schedule; 105 Coro::schedule;
55 106
56 return $self->[2]; 107 $self->{alloc};
57} 108 }
58
59sub wake {
60 my $self = shift;
61 $self->[2] = 1;
62 $self->[1]->ready;
63} 109}
64 110
65sub DESTROY { 111sub DESTROY {
66 my $self = shift; 112 my $self = shift;
67 $self->[0]{conns}++; 113
114 if ($self->{alloc}) {
115 $self->{queue}{slots}++;
68 $self->[0]->wake_next; 116 $self->{queue}->wake_next;
117 }
69} 118}
70 119
71package conn; 120package conn;
72 121
73our %blockuri; 122our %blockuri;

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines