| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Parallel::ForkManager::Scaled; |
|
2
|
8
|
|
|
8
|
|
253055
|
use Moo; |
|
|
8
|
|
|
|
|
46126
|
|
|
|
8
|
|
|
|
|
54
|
|
|
3
|
8
|
|
|
8
|
|
9770
|
use namespace::clean; |
|
|
8
|
|
|
|
|
82965
|
|
|
|
8
|
|
|
|
|
33
|
|
|
4
|
8
|
|
|
8
|
|
3105
|
use Unix::Statgrab; |
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
use List::Util qw( min max ); |
|
6
|
|
|
|
|
|
|
use Storable qw( freeze thaw ); |
|
7
|
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
use v5.10; |
|
9
|
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
our $VERSION = '0.16'; |
|
11
|
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
extends 'Parallel::ForkManager'; |
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
has hard_min_procs => ( is => 'rw', lazy => 1, builder => 1 ); |
|
15
|
|
|
|
|
|
|
has hard_max_procs => ( is => 'rw', lazy => 1, builder => 1 ); |
|
16
|
|
|
|
|
|
|
has soft_min_procs => ( is => 'rw', lazy => 1, builder => 1, trigger => 1 ); |
|
17
|
|
|
|
|
|
|
has soft_max_procs => ( is => 'rw', lazy => 1, builder => 1, trigger => 1 ); |
|
18
|
|
|
|
|
|
|
has initial_procs => ( is => 'lazy' ); |
|
19
|
|
|
|
|
|
|
has update_frequency => ( is => 'rw', default => 1 ); |
|
20
|
|
|
|
|
|
|
has idle_target => ( is => 'rw', default => 0 ); |
|
21
|
|
|
|
|
|
|
has idle_threshold => ( is => 'rw', default => 1 ); |
|
22
|
|
|
|
|
|
|
has run_on_update => ( is => 'rw', clearer => 1, predicate => 1 ); |
|
23
|
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
has last_update => ( is => 'rwp', default => sub{ time } ); |
|
25
|
|
|
|
|
|
|
has _stats_pct => ( is => 'rw', clearer => 1, predicate => 1, handles => [ qw( idle ) ] ); |
|
26
|
|
|
|
|
|
|
has _host_info => ( is => 'rw', clearer => 1, predicate => 1, lazy => 1, builder => 1, handles => [ qw( ncpus ) ] ); |
|
27
|
|
|
|
|
|
|
has _last_stats => ( is => 'rw', clearer => 1, predicate => 1, default => sub{ get_cpu_stats } ); |
|
28
|
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
has __unstorable => ( is => 'ro', init_arg => undef, default => sub{[qw( _stats_pct _host_info _last_stats )]} ); |
|
30
|
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
# |
|
32
|
|
|
|
|
|
|
# Once Parallel::ForkManager has converted to Moo (in development) |
|
33
|
|
|
|
|
|
|
# this will no longer be necessary. Probably. :) |
|
34
|
|
|
|
|
|
|
# |
|
35
|
|
|
|
|
|
|
sub FOREIGNBUILDARGS { |
|
36
|
|
|
|
|
|
|
my ($class, @args) = @_; |
|
37
|
|
|
|
|
|
|
my @ret; |
|
38
|
|
|
|
|
|
|
|
|
39
|
|
|
|
|
|
|
my $args = @args > 1 ? {@args} : $args[0]; |
|
40
|
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
push @ret, 1; # will get changed later in BUILD() |
|
42
|
|
|
|
|
|
|
push @ret, $args->{tempdir} if defined $args->{tempdir}; |
|
43
|
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
@ret; |
|
45
|
|
|
|
|
|
|
} |
|
46
|
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
sub BUILD { |
|
48
|
|
|
|
|
|
|
my $self = shift; |
|
49
|
|
|
|
|
|
|
$self->set_max_procs(min($self->soft_max_procs, max($self->soft_min_procs, $self->initial_procs))); |
|
50
|
|
|
|
|
|
|
$self->update_stats_pct; |
|
51
|
|
|
|
|
|
|
}; |
|
52
|
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
sub _build_hard_min_procs { 1 } |
|
54
|
|
|
|
|
|
|
sub _build_hard_max_procs { (shift->ncpus // 1) * 2 } |
|
55
|
|
|
|
|
|
|
sub _build_soft_min_procs { shift->hard_min_procs }; |
|
56
|
|
|
|
|
|
|
sub _build_soft_max_procs { shift->hard_max_procs }; |
|
57
|
|
|
|
|
|
|
sub _build__host_info { get_host_info } |
|
58
|
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
# pick a value half way between our soft min and max |
|
60
|
|
|
|
|
|
|
sub _build_initial_procs { |
|
61
|
|
|
|
|
|
|
my $self = shift; |
|
62
|
|
|
|
|
|
|
$self->hard_min_procs+int(($self->soft_max_procs-$self->soft_min_procs)/2); |
|
63
|
|
|
|
|
|
|
} |
|
64
|
|
|
|
|
|
|
|
|
65
|
|
|
|
|
|
|
# soft min cannot be less than hard min |
|
66
|
|
|
|
|
|
|
sub _trigger_soft_min_procs { |
|
67
|
|
|
|
|
|
|
my ($self, $newval) = @_; |
|
68
|
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
$self->soft_min_procs($self->hard_min_procs) |
|
70
|
|
|
|
|
|
|
if $newval < $self->hard_min_procs; |
|
71
|
|
|
|
|
|
|
} |
|
72
|
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
# soft max cannot exceed hard_max |
|
74
|
|
|
|
|
|
|
sub _trigger_soft_max_procs { |
|
75
|
|
|
|
|
|
|
my ($self, $newval) = @_; |
|
76
|
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
$self->soft_max_procs($self->hard_max_procs) |
|
78
|
|
|
|
|
|
|
if $newval > $self->hard_max_procs; |
|
79
|
|
|
|
|
|
|
} |
|
80
|
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
sub update_stats_pct { |
|
82
|
|
|
|
|
|
|
my $self = shift; |
|
83
|
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
my $stats = get_cpu_stats; |
|
85
|
|
|
|
|
|
|
my $pcts = $stats->get_cpu_stats_diff($self->_last_stats)->get_cpu_percents; |
|
86
|
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
# Not enough time has elapsed to get a difference, libstatgrab returned NaN |
|
88
|
|
|
|
|
|
|
# Allow it initially to get _stats_pct set but not after |
|
89
|
|
|
|
|
|
|
return if $self->_stats_pct && $pcts->idle eq 'NaN'; |
|
90
|
|
|
|
|
|
|
|
|
91
|
|
|
|
|
|
|
$self->_stats_pct($pcts); |
|
92
|
|
|
|
|
|
|
$self->_last_stats($stats); |
|
93
|
|
|
|
|
|
|
$self->_set_last_update(time); |
|
94
|
|
|
|
|
|
|
} |
|
95
|
|
|
|
|
|
|
|
|
96
|
|
|
|
|
|
|
# |
|
97
|
|
|
|
|
|
|
# (Possibly) adjust our max_procs before the call to start(). |
|
98
|
|
|
|
|
|
|
# |
|
99
|
|
|
|
|
|
|
before start => sub { |
|
100
|
|
|
|
|
|
|
my $self = shift; |
|
101
|
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
return if time - $self->last_update < $self->update_frequency; |
|
103
|
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
$self->update_stats_pct; |
|
105
|
|
|
|
|
|
|
|
|
106
|
|
|
|
|
|
|
my $new_procs; |
|
107
|
|
|
|
|
|
|
my $min_ok = max( 0, $self->idle_target - $self->idle_threshold); |
|
108
|
|
|
|
|
|
|
my $max_ok = min(100, $self->idle_target + $self->idle_threshold); |
|
109
|
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
# |
|
111
|
|
|
|
|
|
|
# It's possible for idle to be NaN if not enough time has elapsed between |
|
112
|
|
|
|
|
|
|
# the initial call to update_stats_pct and the latest call. In this case |
|
113
|
|
|
|
|
|
|
# neither check against $self->idle will be true and no update will occur |
|
114
|
|
|
|
|
|
|
# |
|
115
|
|
|
|
|
|
|
if ($self->idle >= $max_ok && $self->running_procs >= $self->max_procs) { |
|
116
|
|
|
|
|
|
|
$new_procs = $self->adjust_up; |
|
117
|
|
|
|
|
|
|
|
|
118
|
|
|
|
|
|
|
} elsif ($self->idle <= $min_ok) { |
|
119
|
|
|
|
|
|
|
$new_procs = $self->adjust_down; |
|
120
|
|
|
|
|
|
|
} |
|
121
|
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
my $prev_procs = $self->max_procs; |
|
123
|
|
|
|
|
|
|
|
|
124
|
|
|
|
|
|
|
$self->set_max_procs($new_procs) |
|
125
|
|
|
|
|
|
|
if $new_procs; |
|
126
|
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
$self->run_on_update->($self, $prev_procs) |
|
128
|
|
|
|
|
|
|
if ($self->run_on_update && ref($self->run_on_update) eq 'CODE'); |
|
129
|
|
|
|
|
|
|
}; |
|
130
|
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
# |
|
132
|
|
|
|
|
|
|
# constrain max_procs to be within our soft min and max |
|
133
|
|
|
|
|
|
|
# |
|
134
|
|
|
|
|
|
|
around set_max_procs => sub { |
|
135
|
|
|
|
|
|
|
my ($orig, $self, $new_val) = @_; |
|
136
|
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
$orig->($self, |
|
138
|
|
|
|
|
|
|
min( $self->soft_max_procs, max($self->soft_min_procs, $new_val) |
|
139
|
|
|
|
|
|
|
) |
|
140
|
|
|
|
|
|
|
); |
|
141
|
|
|
|
|
|
|
}; |
|
142
|
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
sub stats { |
|
144
|
|
|
|
|
|
|
my $self = shift; |
|
145
|
|
|
|
|
|
|
my $prev_procs = shift // $self->max_procs; |
|
146
|
|
|
|
|
|
|
|
|
147
|
|
|
|
|
|
|
sprintf( |
|
148
|
|
|
|
|
|
|
"%5.1f id %3d run %3d omax %3d nmax %3d smin %3d smax %3d hmin %3d hmax", |
|
149
|
|
|
|
|
|
|
$self->idle, |
|
150
|
|
|
|
|
|
|
scalar($self->running_procs), |
|
151
|
|
|
|
|
|
|
$prev_procs, |
|
152
|
|
|
|
|
|
|
$self->max_procs, |
|
153
|
|
|
|
|
|
|
$self->soft_min_procs, |
|
154
|
|
|
|
|
|
|
$self->soft_max_procs, |
|
155
|
|
|
|
|
|
|
$self->hard_min_procs, |
|
156
|
|
|
|
|
|
|
$self->hard_max_procs |
|
157
|
|
|
|
|
|
|
); |
|
158
|
|
|
|
|
|
|
} |
|
159
|
|
|
|
|
|
|
|
|
160
|
|
|
|
|
|
|
sub dump_stats { |
|
161
|
|
|
|
|
|
|
my $self = shift; |
|
162
|
|
|
|
|
|
|
print STDERR $self->stats(@_)."\n"; |
|
163
|
|
|
|
|
|
|
} |
|
164
|
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
# |
|
166
|
|
|
|
|
|
|
# Increase soft_max_procs to a maximum of hard_max_procs |
|
167
|
|
|
|
|
|
|
# |
|
168
|
|
|
|
|
|
|
# We'll use the system's idle percentage to tell us how much |
|
169
|
|
|
|
|
|
|
# to increase by, the more idle the system is, the more we'll |
|
170
|
|
|
|
|
|
|
# allow soft_max_procs to grow. Hopefully this will allow us |
|
171
|
|
|
|
|
|
|
# to quickly adjust to the system without over-loading it if |
|
172
|
|
|
|
|
|
|
# it's already close to our target idle state |
|
173
|
|
|
|
|
|
|
# |
|
174
|
|
|
|
|
|
|
sub adjust_soft_max { |
|
175
|
|
|
|
|
|
|
my $self = shift; |
|
176
|
|
|
|
|
|
|
$self->soft_max_procs( |
|
177
|
|
|
|
|
|
|
min($self->hard_max_procs, |
|
178
|
|
|
|
|
|
|
$self->soft_max_procs |
|
179
|
|
|
|
|
|
|
+ max(1, int( |
|
180
|
|
|
|
|
|
|
($self->hard_max_procs - $self->max_procs) |
|
181
|
|
|
|
|
|
|
* ($self->idle - $self->idle_target) |
|
182
|
|
|
|
|
|
|
/ 100 |
|
183
|
|
|
|
|
|
|
)) |
|
184
|
|
|
|
|
|
|
) |
|
185
|
|
|
|
|
|
|
); |
|
186
|
|
|
|
|
|
|
} |
|
187
|
|
|
|
|
|
|
|
|
188
|
|
|
|
|
|
|
# |
|
189
|
|
|
|
|
|
|
# Decrease soft_min_procs, the system is too busy |
|
190
|
|
|
|
|
|
|
# |
|
191
|
|
|
|
|
|
|
sub adjust_soft_min { |
|
192
|
|
|
|
|
|
|
my $self = shift; |
|
193
|
|
|
|
|
|
|
$self->soft_min_procs( |
|
194
|
|
|
|
|
|
|
max($self->hard_min_procs, |
|
195
|
|
|
|
|
|
|
$self->hard_min_procs |
|
196
|
|
|
|
|
|
|
+ max(0, int( |
|
197
|
|
|
|
|
|
|
($self->max_procs - $self->hard_min_procs) |
|
198
|
|
|
|
|
|
|
* ($self->idle_target - $self->idle) |
|
199
|
|
|
|
|
|
|
/ 100 |
|
200
|
|
|
|
|
|
|
)) |
|
201
|
|
|
|
|
|
|
) |
|
202
|
|
|
|
|
|
|
); |
|
203
|
|
|
|
|
|
|
} |
|
204
|
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
# |
|
206
|
|
|
|
|
|
|
# Adjust our number of running processes (max_procs) to half way between |
|
207
|
|
|
|
|
|
|
# the current number and our soft max. If we're already at |
|
208
|
|
|
|
|
|
|
# soft max, try to adjust the soft max up first. |
|
209
|
|
|
|
|
|
|
# |
|
210
|
|
|
|
|
|
|
# Set the soft min to the current number of running procs |
|
211
|
|
|
|
|
|
|
# as it wasn't enough to hit our idle target so we shouldn't |
|
212
|
|
|
|
|
|
|
# go below it again (although we can if we actually need to). |
|
213
|
|
|
|
|
|
|
# |
|
214
|
|
|
|
|
|
|
sub adjust_up { |
|
215
|
|
|
|
|
|
|
my $self = shift; |
|
216
|
|
|
|
|
|
|
my $cur = $self->max_procs; |
|
217
|
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
my $max = $cur >= $self->soft_max_procs |
|
219
|
|
|
|
|
|
|
? $self->adjust_soft_max |
|
220
|
|
|
|
|
|
|
: $self->soft_max_procs; |
|
221
|
|
|
|
|
|
|
|
|
222
|
|
|
|
|
|
|
$self->soft_min_procs($cur); |
|
223
|
|
|
|
|
|
|
$cur + max(1,int(($max - $cur)/2)); |
|
224
|
|
|
|
|
|
|
} |
|
225
|
|
|
|
|
|
|
|
|
226
|
|
|
|
|
|
|
sub adjust_down { |
|
227
|
|
|
|
|
|
|
my $self = shift; |
|
228
|
|
|
|
|
|
|
my $cur = $self->max_procs; |
|
229
|
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
my $min = $cur <= $self->soft_min_procs |
|
231
|
|
|
|
|
|
|
? $self->adjust_soft_min |
|
232
|
|
|
|
|
|
|
: $self->soft_min_procs; |
|
233
|
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
# Shouldn't happen, but test for it anyway |
|
235
|
|
|
|
|
|
|
return undef unless $cur > $min; |
|
236
|
|
|
|
|
|
|
|
|
237
|
|
|
|
|
|
|
$self->soft_max_procs($cur); |
|
238
|
|
|
|
|
|
|
$min + int(($cur - $min)/2); |
|
239
|
|
|
|
|
|
|
} |
|
240
|
|
|
|
|
|
|
|
|
241
|
|
|
|
|
|
|
|
|
242
|
|
|
|
|
|
|
# |
|
243
|
|
|
|
|
|
|
# libstatgrab doesn't like freeze/thaw (saw assertion errors from vector.c) |
|
244
|
|
|
|
|
|
|
# so we need to set those # attributes that house Unix::Statgrab objects to |
|
245
|
|
|
|
|
|
|
# undef before # being frozen. Restore them after freezing. |
|
246
|
|
|
|
|
|
|
# |
|
247
|
|
|
|
|
|
|
# Also, freeze/thaw can't handle CODE references so we'll clear |
|
248
|
|
|
|
|
|
|
# our run_on_update hook. There will still be problems with the |
|
249
|
|
|
|
|
|
|
# underlying Parallel::ForkManager hooks but I'm not going to |
|
250
|
|
|
|
|
|
|
# try to fix those here. That should be handled by Parallel::ForkManager |
|
251
|
|
|
|
|
|
|
# I believe. |
|
252
|
|
|
|
|
|
|
# |
|
253
|
|
|
|
|
|
|
sub STORABLE_freeze { |
|
254
|
|
|
|
|
|
|
my ($self, $cloning) = @_; |
|
255
|
|
|
|
|
|
|
state $storing = 0; |
|
256
|
|
|
|
|
|
|
return if $cloning || $storing; |
|
257
|
|
|
|
|
|
|
|
|
258
|
|
|
|
|
|
|
# libstatgrab isn't happy when it's frozen / thawed |
|
259
|
|
|
|
|
|
|
my %save; |
|
260
|
|
|
|
|
|
|
for (@{$self->__unstorable}) { |
|
261
|
|
|
|
|
|
|
$save{$_} = $self->$_; |
|
262
|
|
|
|
|
|
|
$self->$_(undef); |
|
263
|
|
|
|
|
|
|
} |
|
264
|
|
|
|
|
|
|
$save{run_on_update} = $self->run_on_update; |
|
265
|
|
|
|
|
|
|
$self->clear_run_on_update; |
|
266
|
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
$storing = 1; |
|
268
|
|
|
|
|
|
|
my $ret = freeze($self); |
|
269
|
|
|
|
|
|
|
$storing = 0; |
|
270
|
|
|
|
|
|
|
|
|
271
|
|
|
|
|
|
|
$self->$_($save{$_}) for @{$self->__unstorable}; |
|
272
|
|
|
|
|
|
|
$self->run_on_update($save{run_on_update}); |
|
273
|
|
|
|
|
|
|
|
|
274
|
|
|
|
|
|
|
$ret; |
|
275
|
|
|
|
|
|
|
}; |
|
276
|
|
|
|
|
|
|
|
|
277
|
|
|
|
|
|
|
# |
|
278
|
|
|
|
|
|
|
# Since our Unix::Statgrab objects are all lazily built, they were |
|
279
|
|
|
|
|
|
|
# set to undef before freeze(). We need to clear them in thaw() so |
|
280
|
|
|
|
|
|
|
# they can be re-built. Not perfect but should keep things working |
|
281
|
|
|
|
|
|
|
# |
|
282
|
|
|
|
|
|
|
# We will have lost the run_on_update hook if it was set, but nothing |
|
283
|
|
|
|
|
|
|
# to be done about that. |
|
284
|
|
|
|
|
|
|
# |
|
285
|
|
|
|
|
|
|
sub STORABLE_thaw { |
|
286
|
|
|
|
|
|
|
my ($self, $cloning, $data) = @_; |
|
287
|
|
|
|
|
|
|
state $thawing = 0; |
|
288
|
|
|
|
|
|
|
|
|
289
|
|
|
|
|
|
|
return if $cloning || $thawing; |
|
290
|
|
|
|
|
|
|
|
|
291
|
|
|
|
|
|
|
$thawing = 1; |
|
292
|
|
|
|
|
|
|
%$self = %{thaw($data)}; |
|
293
|
|
|
|
|
|
|
|
|
294
|
|
|
|
|
|
|
eval "\$self->_clear$_" for @{$self->__unstorable}; |
|
295
|
|
|
|
|
|
|
|
|
296
|
|
|
|
|
|
|
# And this non-hidden code ref |
|
297
|
|
|
|
|
|
|
$self->clear_run_on_update; |
|
298
|
|
|
|
|
|
|
|
|
299
|
|
|
|
|
|
|
$thawing = 0; |
|
300
|
|
|
|
|
|
|
} |
|
301
|
|
|
|
|
|
|
|
|
302
|
|
|
|
|
|
|
1; |
|
303
|
|
|
|
|
|
|
|
|
304
|
|
|
|
|
|
|
__END__ |