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.15 by root, Fri Nov 30 03:06:37 2001 UTC vs.
Revision 1.24 by root, Tue Dec 4 02:46:29 2001 UTC

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

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines