1 | package transferqueue; |
|
|
2 | |
|
|
3 | sub new { |
|
|
4 | my $class = shift; |
|
|
5 | bless { |
|
|
6 | conns => $_[0], |
|
|
7 | }, $class; |
|
|
8 | } |
|
|
9 | |
|
|
10 | sub start_transfer { |
|
|
11 | my $self = shift; |
|
|
12 | |
|
|
13 | my $trans = bless [ $self ], transfer::; |
|
|
14 | |
|
|
15 | push @{$self->{wait}}, $trans; |
|
|
16 | Scalar::Util::weaken($self->{wait}[-1]); |
|
|
17 | |
|
|
18 | if (--$self->{conns} >= 0) { |
|
|
19 | $self->wake_next; |
|
|
20 | } |
|
|
21 | |
|
|
22 | $trans; |
|
|
23 | } |
|
|
24 | |
|
|
25 | sub wake_next { |
|
|
26 | my $self = shift; |
|
|
27 | |
|
|
28 | while(@{$self->{wait}}) { |
|
|
29 | my $transfer = shift @{$self->{wait}}; |
|
|
30 | if ($transfer) { |
|
|
31 | $transfer->wake; |
|
|
32 | last; |
|
|
33 | } |
|
|
34 | } |
|
|
35 | } |
|
|
36 | |
|
|
37 | sub waiters { |
|
|
38 | map $_->[1], @{$_[0]{wait}}; |
|
|
39 | } |
|
|
40 | |
|
|
41 | package transfer; |
|
|
42 | |
|
|
43 | use Coro::Timer (); |
|
|
44 | |
|
|
45 | sub wake { |
|
|
46 | my $self = shift; |
|
|
47 | $self->[2] = 1; |
|
|
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]; |
|
|
62 | } |
|
|
63 | |
|
|
64 | sub DESTROY { |
|
|
65 | my $self = shift; |
|
|
66 | $self->[0]{conns}++; |
|
|
67 | $self->[0]->wake_next if $self->[2]; |
|
|
68 | } |
|
|
69 | |
|
|
70 | package conn; |
1 | package conn; |
71 | |
2 | |
72 | our %blockuri; |
3 | our %blockuri; |
73 | our $blockref; |
4 | our $blockref; |
74 | |
5 | |