… | |
… | |
8 | } |
8 | } |
9 | |
9 | |
10 | sub start_transfer { |
10 | sub start_transfer { |
11 | my $self = shift; |
11 | my $self = shift; |
12 | |
12 | |
13 | my $trans = bless [ $self, $Coro::current ], transfer::; |
13 | my $trans = bless [ $self ], transfer::; |
14 | print "$self $trans $trans->[0] <<\n";#d# |
|
|
15 | Scalar::Util::weaken($trans->[0]); |
|
|
16 | |
14 | |
17 | push @{$self->{wait}}, $trans; |
15 | push @{$self->{wait}}, $trans; |
|
|
16 | Scalar::Util::weaken($self->{wait}[-1]); |
18 | |
17 | |
19 | if (--$self->{conns} >= 0) { |
18 | if (--$self->{conns} >= 0) { |
20 | $self->wake_next; |
19 | $self->wake_next; |
21 | } |
20 | } |
22 | |
21 | |
… | |
… | |
24 | } |
23 | } |
25 | |
24 | |
26 | sub wake_next { |
25 | sub wake_next { |
27 | my $self = shift; |
26 | my $self = shift; |
28 | |
27 | |
29 | return unless $self->{conns} >= 0; |
28 | while(@{$self->{wait}}) { |
30 | |
29 | my $transfer = shift @{$self->{wait}}; |
31 | (pop @{$self->{wait}})->wake if @{$self->{wait}}; |
30 | if ($transfer) { |
|
|
31 | $transfer->wake; |
|
|
32 | last; |
|
|
33 | } |
|
|
34 | } |
32 | } |
35 | } |
33 | |
36 | |
34 | sub waiters { |
37 | sub waiters { |
35 | map $_->[1], @{$_[0]{wait}}; |
38 | map $_->[1], @{$_[0]{wait}}; |
36 | } |
39 | } |
37 | |
40 | |
38 | package transfer; |
41 | package transfer; |
39 | |
42 | |
40 | use Coro::Timer (); |
43 | use Coro::Timer (); |
41 | |
44 | |
42 | sub try { |
|
|
43 | my $self = shift; |
|
|
44 | my $timeout = Coro::Timer::timeout $_[0]; |
|
|
45 | |
|
|
46 | Coro::schedule; |
|
|
47 | |
|
|
48 | return $self->[2]; |
|
|
49 | } |
|
|
50 | |
|
|
51 | sub wake { |
45 | sub wake { |
52 | my $self = shift; |
46 | my $self = shift; |
53 | $self->[2] = 1; |
47 | $self->[2] = 1; |
54 | $self->[1]->ready; |
48 | ref $self->[1] and $self->[1]->ready; |
|
|
49 | } |
|
|
50 | |
|
|
51 | sub try { |
|
|
52 | my $self = shift; |
|
|
53 | |
|
|
54 | unless ($self->[2]) { |
|
|
55 | my $timeout = Coro::Timer::timeout $_[0]; |
|
|
56 | local $self->[1] = $Coro::current; |
|
|
57 | |
|
|
58 | Coro::schedule; |
|
|
59 | } |
|
|
60 | |
|
|
61 | return $self->[2]; |
55 | } |
62 | } |
56 | |
63 | |
57 | sub DESTROY { |
64 | sub DESTROY { |
58 | my $self = shift; |
65 | my $self = shift; |
59 | $self->[0]{conns}++; |
66 | $self->[0]{conns}++; |
60 | $self->[0]->wake_next; |
67 | $self->[0]->wake_next if $self->[2]; |
61 | } |
68 | } |
62 | |
69 | |
63 | package conn; |
70 | package conn; |
64 | |
71 | |
65 | our %blockuri; |
72 | our %blockuri; |