Skip to content

Commit

Permalink
merge revision(s) 43148,43149,43152: [Backport #8433]
Browse files Browse the repository at this point in the history
	* 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
  • Loading branch information
unak committed Feb 17, 2014
1 parent 4c58aa8 commit efe5d8d
Show file tree
Hide file tree
Showing 4 changed files with 61 additions and 9 deletions.
5 changes: 5 additions & 0 deletions ChangeLog
@@ -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
Expand Down
27 changes: 27 additions & 0 deletions test/ruby/test_thread.rb
Expand Up @@ -710,4 +710,31 @@ def test_thread_timer_and_interrupt
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 changes: 26 additions & 6 deletions thread.c
Expand Up @@ -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
Expand Down Expand Up @@ -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;
Expand Down Expand Up @@ -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
Expand Down Expand Up @@ -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)
{
Expand Down
6 changes: 3 additions & 3 deletions version.h
@@ -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"

Expand Down

0 comments on commit efe5d8d

Please sign in to comment.