process.c (waitpid_wait): do not set ECHILD prematurely

It is possible to have both MJIT and normal child processes
alive, so we cannot set ECHILD based on such a guess.  We can
still elide waitpid(PID <= 0) calls if we have callers in
vm->waiting_pids, however.

For specs, ensure Process.waitall does not leak MJIT
PIDs to Rubyspace.

git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@63764 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
normal 2018-06-27 10:09:33 +00:00
Родитель 6b534134a7
Коммит 92f5653f45
4 изменённых файлов: 20 добавлений и 8 удалений

Просмотреть файл

@ -1074,12 +1074,6 @@ waitpid_wait(struct waitpid_state *w)
}
else if (w->options & WNOHANG) {
w->cond = 0;
/* MJIT must be waiting, but don't tell Ruby callers about it */
if (w->pid < 0 && !list_empty(&vm->waiting_pids)) {
w->ret = -1;
w->errnum = ECHILD;
}
}
else {
w->cond = rb_sleep_cond_get(w->ec);

Просмотреть файл

@ -39,3 +39,9 @@ end
def with_feature(*features, &block)
FeatureGuard.new(*features).run_if(:with_feature, &block)
end
MSpecEnv.class_eval do
def without_feature(*features, &block)
FeatureGuard.new(*features).run_unless(:without_feature, &block)
end
end

Просмотреть файл

@ -4,11 +4,19 @@ describe "Process.wait2" do
before :all do
# HACK: this kludge is temporarily necessary because some
# misbehaving spec somewhere else does not clear processes
# Note: background processes are unavoidable with MJIT,
# but we shouldn't reap them from Ruby-space
begin
Process.wait(-1, Process::WNOHANG)
$stderr.puts "Leaked process before wait2 specs! Waiting for it"
without_feature :mjit do
$stderr.puts "Leaked process before wait2 specs! Waiting for it"
end
leaked = Process.waitall
$stderr.puts "leaked before wait2 specs: #{leaked}"
$stderr.puts "leaked before wait2 specs: #{leaked}" unless leaked.empty?
with_feature :mjit do
# Ruby-space should not see PIDs used by mjit
leaked.should be_empty
end
rescue Errno::ECHILD # No child processes
rescue NotImplementedError
end

Просмотреть файл

@ -8,6 +8,10 @@ describe "Process.wait" do
begin
leaked = Process.waitall
puts "leaked before wait specs: #{leaked}" unless leaked.empty?
with_feature :mjit do
# Ruby-space should not see PIDs used by mjit
leaked.should be_empty
end
rescue NotImplementedError
end
end