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.17 by root, Sat Dec 1 01:09:56 2001 UTC vs.
Revision 1.26 by root, Tue Dec 4 15:57:59 2001 UTC

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

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines