mirror of
https://github.com/MariaDB/server.git
synced 2025-10-22 19:52:58 +03:00
perl The problem here was the method how MTR gets its unique thread ids. Prior to this patch, the method to do it was to maintain a global table of pid,mtr_unique_id) pairs. The table was backed by a text file. The table was cleaned up one in a while and dead processes leaking unique_ids were determined with with kill(0) or with scripting tasklist on Windows. This method is flawed specifically on native Windows Perl. fork() is implemented with starting a new thread, give it a syntetic negative PID (threadID*(-1)), until this thread creates a new process with exec() However, neither tasklist nor any other native Windows tool can cope with negative perl PIDs. This lead to incorrect determination of dead process and reusing already used mtr_unique_id. The patch introduces alternative portable method of solving unique-id problem. When a process needs a unique id in range [min...max], it just starts to open files named min, min+1,...max in a loop . After file is opened, we do non-blocking flock(). When flock() succeeds, process has allocated the ID. When process dies, file is unlocked . Checks for zombies are not necessary. Since the change would create a co-existence problems with older version of MTR, because of different way to calculate IDs, the default ID range is changed from 250-299 to 300-349. Another fix that was necessary enable --parallel option was to serialize spawn() calls on Windows. specifically, IO redirects needed to be protected. This patch also fixes hanging CRTL-C (as described in Bug #38629) for the "new" MTR. The fix was already in 6.0 and is now downported.
222 lines
5.0 KiB
Perl
222 lines
5.0 KiB
Perl
# -*- cperl -*-
|
|
# Copyright (C) 2004-2006 MySQL AB
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; version 2 of the License.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
# This is a library file used by the Perl version of mysql-test-run,
|
|
# and is part of the translation of the Bourne shell script with the
|
|
# same name.
|
|
|
|
use strict;
|
|
|
|
package My::SafeProcess::Base;
|
|
|
|
#
|
|
# Utility functions for Process management
|
|
#
|
|
|
|
use Carp;
|
|
use IO::Pipe;
|
|
|
|
use base qw(Exporter);
|
|
our @EXPORT= qw(create_process);
|
|
|
|
|
|
|
|
#
|
|
# safe_fork
|
|
# Retry a couple of times if fork returns EAGAIN
|
|
#
|
|
sub _safe_fork {
|
|
my $retries= 5;
|
|
my $pid;
|
|
|
|
FORK:
|
|
{
|
|
$pid= fork;
|
|
if ( not defined($pid)) {
|
|
|
|
croak("fork failed after: $!") if (!$retries--);
|
|
|
|
warn("fork failed sleep 1 second and redo: $!");
|
|
sleep(1);
|
|
redo FORK;
|
|
}
|
|
}
|
|
|
|
return $pid;
|
|
};
|
|
|
|
|
|
#
|
|
# Decode exit status
|
|
#
|
|
sub exit_status {
|
|
my $self= shift;
|
|
my $raw= $self->{EXIT_STATUS};
|
|
|
|
croak("Can't call exit_status before process has died")
|
|
unless defined $raw;
|
|
|
|
if ($raw & 127)
|
|
{
|
|
# Killed by signal
|
|
my $signal_num= $raw & 127;
|
|
my $dumped_core= $raw & 128;
|
|
return 1; # Return error code
|
|
}
|
|
else
|
|
{
|
|
# Normal process exit
|
|
return $raw >> 8;
|
|
};
|
|
}
|
|
|
|
# threads.pm may not exist everywhere, so use only on Windows.
|
|
|
|
use if $^O eq "MSWin32", "threads";
|
|
use if $^O eq "MSWin32", "threads::shared";
|
|
|
|
my $win32_spawn_lock :shared;
|
|
|
|
|
|
#
|
|
# Create a new process
|
|
# Return pid of the new process
|
|
#
|
|
sub create_process {
|
|
my %opts=
|
|
(
|
|
@_
|
|
);
|
|
|
|
my $path = delete($opts{'path'}) or die "path required";
|
|
my $args = delete($opts{'args'}) or die "args required";
|
|
my $input = delete($opts{'input'});
|
|
my $output = delete($opts{'output'});
|
|
my $error = delete($opts{'error'});
|
|
|
|
my $open_mode= $opts{append} ? ">>" : ">";
|
|
|
|
if ($^O eq "MSWin32"){
|
|
|
|
lock($win32_spawn_lock);
|
|
|
|
#printf STDERR "stdin %d, stdout %d, stderr %d\n",
|
|
# fileno STDIN, fileno STDOUT, fileno STDERR;
|
|
|
|
# input output redirect
|
|
my ($oldin, $oldout, $olderr);
|
|
open $oldin, '<&', \*STDIN or die "Failed to save old stdin: $!";
|
|
open $oldout, '>&', \*STDOUT or die "Failed to save old stdout: $!";
|
|
open $olderr, '>&', \*STDERR or die "Failed to save old stderr: $!";
|
|
|
|
if ( $input ) {
|
|
if ( ! open(STDIN, "<", $input) ) {
|
|
croak("can't redirect STDIN to '$input': $!");
|
|
}
|
|
}
|
|
|
|
if ( $output ) {
|
|
if ( ! open(STDOUT, $open_mode, $output) ) {
|
|
croak("can't redirect STDOUT to '$output': $!");
|
|
}
|
|
}
|
|
|
|
if ( $error ) {
|
|
if ( $output eq $error ) {
|
|
if ( ! open(STDERR, ">&STDOUT") ) {
|
|
croak("can't dup STDOUT: $!");
|
|
}
|
|
}
|
|
elsif ( ! open(STDERR, $open_mode, $error) ) {
|
|
croak("can't redirect STDERR to '$error': $!");
|
|
}
|
|
}
|
|
|
|
|
|
# Magic use of 'system(1, @args)' to spawn a process
|
|
# and get a proper Win32 pid
|
|
unshift (@$args, $path);
|
|
my $pid= system(1, @$args);
|
|
if ( $pid == 0 ){
|
|
print $olderr "create_process failed: $^E\n";
|
|
die "create_process failed: $^E";
|
|
}
|
|
|
|
# Retore IO redirects
|
|
open STDERR, '>&', $olderr
|
|
or croak("unable to reestablish STDERR");
|
|
open STDOUT, '>&', $oldout
|
|
or croak("unable to reestablish STDOUT");
|
|
open STDIN, '<&', $oldin
|
|
or croak("unable to reestablish STDIN");
|
|
#printf STDERR "stdin %d, stdout %d, stderr %d\n",
|
|
# fileno STDIN, fileno STDOUT, fileno STDERR;
|
|
return $pid;
|
|
|
|
}
|
|
|
|
local $SIG{PIPE}= sub { print STDERR "Got signal $@\n"; };
|
|
my $pipe= IO::Pipe->new();
|
|
my $pid= _safe_fork();
|
|
if ($pid){
|
|
# Parent
|
|
$pipe->reader();
|
|
my $line= <$pipe>; # Wait for child to say it's ready
|
|
return $pid;
|
|
}
|
|
|
|
$SIG{INT}= 'DEFAULT';
|
|
|
|
# Make this process it's own process group to be able to kill
|
|
# it and any childs(that hasn't changed group themself)
|
|
setpgrp(0,0) if $opts{setpgrp};
|
|
|
|
if ( $output and !open(STDOUT, $open_mode, $output) ) {
|
|
croak("can't redirect STDOUT to '$output': $!");
|
|
}
|
|
|
|
if ( $error ) {
|
|
if ( defined $output and $output eq $error ) {
|
|
if ( ! open(STDERR, ">&STDOUT") ) {
|
|
croak("can't dup STDOUT: $!");
|
|
}
|
|
}
|
|
elsif ( ! open(STDERR, $open_mode, $error) ) {
|
|
croak("can't redirect STDERR to '$error': $!");
|
|
}
|
|
}
|
|
|
|
if ( $input ) {
|
|
if ( ! open(STDIN, "<", $input) ) {
|
|
croak("can't redirect STDIN to '$input': $!");
|
|
}
|
|
}
|
|
|
|
# Tell parent to continue
|
|
$pipe->writer();
|
|
print $pipe "ready\n";
|
|
|
|
if ( !exec($path, @$args) ){
|
|
croak("Failed to exec '$path': $!");
|
|
}
|
|
|
|
croak("Should never come here");
|
|
|
|
}
|
|
|
|
1;
|
|
|