1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00

* process.c (before_exec, after_exec): change SIGPIPE handler to SIG_DFL

before calling execve(). Because r31760 reintroduced an issue that
  system() may hang up (i.e. [ruby-dev:12261]).
* process.c (save_sigpipe, restore_sigpipe): new.



git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@31761 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
kosaki 2011-05-28 13:52:03 +00:00
parent e39b50d4b5
commit ed02c4122a
3 changed files with 54 additions and 7 deletions

View file

@ -1,3 +1,10 @@
Sat May 28 19:30:17 2011 KOSAKI Motohiro <kosaki.motohiro@gmail.com>
* process.c (before_exec, after_exec): change SIGPIPE handler to SIG_DFL
before calling execve(). Because r31760 reintroduced an issue that
system() may hang up (i.e. [ruby-dev:12261]).
* process.c (save_sigpipe, restore_sigpipe): new.
Sat May 28 16:08:16 2011 KOSAKI Motohiro <kosaki.motohiro@gmail.com> Sat May 28 16:08:16 2011 KOSAKI Motohiro <kosaki.motohiro@gmail.com>
* signal.c (Init_signal, default_handler): change default SIGPIPE handler * signal.c (Init_signal, default_handler): change default SIGPIPE handler

View file

@ -984,14 +984,41 @@ void rb_thread_reset_timer_thread(void);
static int forked_child = 0; static int forked_child = 0;
#ifdef SIGPIPE
static RETSIGTYPE (*saved_sigpipe_handler)(int) = 0;
#endif
#if defined(POSIX_SIGNAL)
# define signal(a,b) posix_signal((a),(b))
#endif
static void save_sigpipe(void)
{
#ifdef SIGPIPE
/*
* Some OS commands don't initialize signal handler properly. Thus we have to
* reset signal handler before exec(). Otherwise, system() and similar child process
* interaction might fail. (e.g. ruby -e "system 'yes | ls'") [ruby-dev:12261]
*/
saved_sigpipe_handler = signal(SIGPIPE, SIG_DFL);
#endif
}
static void restore_sigpipe(void)
{
#ifdef SIGPIPE
signal(SIGPIPE, saved_sigpipe_handler);
#endif
}
/* /*
* On old MacOS X, exec() may return ENOTSUPP if the process have multiple threads. * On old MacOS X, exec() may return ENOTSUPP if the process have multiple threads.
* Therefore we have to kill internal threads at once. [ruby-core: 10583] * Therefore we have to kill internal threads at once. [ruby-core: 10583]
*/ */
#define before_exec() \ #define before_exec() \
(rb_enable_interrupt(), (void)(forked_child ? 0 : (rb_thread_stop_timer_thread(), 1))) (rb_enable_interrupt(), save_sigpipe(), (void)(forked_child ? 0 : (rb_thread_stop_timer_thread(), 1)))
#define after_exec() \ #define after_exec() \
(rb_thread_reset_timer_thread(), rb_thread_start_timer_thread(), forked_child = 0, rb_disable_interrupt()) (rb_thread_reset_timer_thread(), rb_thread_start_timer_thread(), forked_child = 0, restore_sigpipe(), rb_disable_interrupt())
#define before_fork() before_exec() #define before_fork() before_exec()
#define after_fork() (GET_THREAD()->thrown_errinfo = 0, after_exec()) #define after_fork() (GET_THREAD()->thrown_errinfo = 0, after_exec())
@ -2907,11 +2934,6 @@ rb_f_abort(int argc, VALUE *argv)
return Qnil; /* not reached */ return Qnil; /* not reached */
} }
#if defined(POSIX_SIGNAL)
# define signal(a,b) posix_signal((a),(b))
#endif
void void
rb_syswait(rb_pid_t pid) rb_syswait(rb_pid_t pid)
{ {

View file

@ -1,6 +1,7 @@
require 'test/unit' require 'test/unit'
require 'tmpdir' require 'tmpdir'
require 'pathname' require 'pathname'
require 'timeout'
require_relative 'envutil' require_relative 'envutil'
require 'rbconfig' require 'rbconfig'
@ -1250,4 +1251,21 @@ class TestProcess < Test::Unit::TestCase
exs << Errno::E2BIG if defined?(Errno::E2BIG) exs << Errno::E2BIG if defined?(Errno::E2BIG)
assert_raise(*exs, bug4315) {Process.spawn('"a"|'*10_000_000)} assert_raise(*exs, bug4315) {Process.spawn('"a"|'*10_000_000)}
end end
def test_system_sigpipe
return if /mswin|mingw/ =~ RUBY_PLATFORM
pid = 0
with_tmpchdir do
assert_nothing_raised('[ruby-dev:12261]') do
timeout(3) do
pid = spawn('yes | ls')
Process.waitpid pid
end
end
end
ensure
Process.kill(:KILL, pid) if (pid != 0) rescue false
end
end end