mirror of
https://github.com/ruby/ruby.git
synced 2025-08-15 13:39:04 +02:00
merge revision(s) 43148,43149,43152: [Backport #8433]
* thread.c (terminate_atfork_i): fix locking mutexes not unlocked in forks when not tracked in thread. [ruby-core:55102] [Bug #8433] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_1_9_3@45026 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
4c58aa8ec7
commit
efe5d8db6e
4 changed files with 61 additions and 9 deletions
|
@ -1,3 +1,8 @@
|
|||
Mon Feb 17 18:04:40 2014 Aaron Pfeifer <aaron.pfeifer@gmail.com>
|
||||
|
||||
* thread.c (terminate_atfork_i): fix locking mutexes not unlocked in
|
||||
forks when not tracked in thread. [ruby-core:55102] [Bug #8433]
|
||||
|
||||
Fri Feb 14 21:01:12 2014 NAKAMURA Usaku <usa@ruby-lang.org>
|
||||
|
||||
* ext/socket: revert r44943 because it causes errors on some linux
|
||||
|
|
|
@ -710,4 +710,31 @@ class TestThreadGroup < Test::Unit::TestCase
|
|||
end
|
||||
assert_in_delta(t1 - t0, 1, 1)
|
||||
end
|
||||
|
||||
def test_blocking_mutex_unlocked_on_fork
|
||||
bug8433 = '[ruby-core:55102] [Bug #8433]'
|
||||
|
||||
mutex = Mutex.new
|
||||
flag = false
|
||||
mutex.lock
|
||||
|
||||
th = Thread.new do
|
||||
mutex.synchronize do
|
||||
flag = true
|
||||
sleep
|
||||
end
|
||||
end
|
||||
|
||||
Thread.pass until th.stop?
|
||||
mutex.unlock
|
||||
|
||||
pid = Process.fork do
|
||||
exit(mutex.locked?)
|
||||
end
|
||||
|
||||
th.kill
|
||||
|
||||
pid, status = Process.waitpid2(pid)
|
||||
assert_equal(false, status.success?, bug8433)
|
||||
end if Process.respond_to?(:fork)
|
||||
end
|
||||
|
|
32
thread.c
32
thread.c
|
@ -345,6 +345,8 @@ typedef struct rb_mutex_struct
|
|||
} rb_mutex_t;
|
||||
|
||||
static void rb_mutex_abandon_all(rb_mutex_t *mutexes);
|
||||
static void rb_mutex_abandon_keeping_mutexes(rb_thread_t *th);
|
||||
static void rb_mutex_abandon_locking_mutex(rb_thread_t *th);
|
||||
static const char* rb_mutex_unlock_th(rb_mutex_t *mutex, rb_thread_t volatile *th);
|
||||
|
||||
void
|
||||
|
@ -3109,10 +3111,8 @@ terminate_atfork_i(st_data_t key, st_data_t val, st_data_t current_th)
|
|||
GetThreadPtr(thval, th);
|
||||
|
||||
if (th != (rb_thread_t *)current_th) {
|
||||
if (th->keeping_mutexes) {
|
||||
rb_mutex_abandon_all(th->keeping_mutexes);
|
||||
}
|
||||
th->keeping_mutexes = NULL;
|
||||
rb_mutex_abandon_keeping_mutexes(th);
|
||||
rb_mutex_abandon_locking_mutex(th);
|
||||
thread_cleanup_func(th, TRUE);
|
||||
}
|
||||
return ST_CONTINUE;
|
||||
|
@ -3370,8 +3370,6 @@ thgroup_add(VALUE group, VALUE thread)
|
|||
#define GetMutexPtr(obj, tobj) \
|
||||
TypedData_Get_Struct((obj), rb_mutex_t, &mutex_data_type, (tobj))
|
||||
|
||||
static const char *rb_mutex_unlock_th(rb_mutex_t *mutex, rb_thread_t volatile *th);
|
||||
|
||||
#define mutex_mark NULL
|
||||
|
||||
static void
|
||||
|
@ -3687,6 +3685,28 @@ rb_mutex_unlock(VALUE self)
|
|||
return self;
|
||||
}
|
||||
|
||||
static void
|
||||
rb_mutex_abandon_keeping_mutexes(rb_thread_t *th)
|
||||
{
|
||||
if (th->keeping_mutexes) {
|
||||
rb_mutex_abandon_all(th->keeping_mutexes);
|
||||
}
|
||||
th->keeping_mutexes = NULL;
|
||||
}
|
||||
|
||||
static void
|
||||
rb_mutex_abandon_locking_mutex(rb_thread_t *th)
|
||||
{
|
||||
rb_mutex_t *mutex;
|
||||
|
||||
if (!th->locking_mutex) return;
|
||||
|
||||
GetMutexPtr(th->locking_mutex, mutex);
|
||||
if (mutex->th == th)
|
||||
rb_mutex_abandon_all(mutex);
|
||||
th->locking_mutex = Qfalse;
|
||||
}
|
||||
|
||||
static void
|
||||
rb_mutex_abandon_all(rb_mutex_t *mutexes)
|
||||
{
|
||||
|
|
|
@ -1,10 +1,10 @@
|
|||
#define RUBY_VERSION "1.9.3"
|
||||
#define RUBY_PATCHLEVEL 534
|
||||
#define RUBY_PATCHLEVEL 535
|
||||
|
||||
#define RUBY_RELEASE_DATE "2014-02-14"
|
||||
#define RUBY_RELEASE_DATE "2014-02-17"
|
||||
#define RUBY_RELEASE_YEAR 2014
|
||||
#define RUBY_RELEASE_MONTH 2
|
||||
#define RUBY_RELEASE_DAY 14
|
||||
#define RUBY_RELEASE_DAY 17
|
||||
|
||||
#include "ruby/version.h"
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue