summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authornaruse <naruse@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2019-01-29 05:31:00 +0000
committernaruse <naruse@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2019-01-29 05:31:00 +0000
commitfbad5b97e8a42304b73b6141b57fb8ac45565ec2 (patch)
treeb9adfb254da0a4b31ef2ef5280088d9cd95c54cc
parentbf8cc37b467e1d372e1b01b4d25e9ef6f8cc927f (diff)
merge revision(s) 66708: [Backport #15499]
thread.c (call_without_gvl): spawn thread for UBF iff single-threaded We need another native thread to call some unblocking functions which aren't RUBY_UBF_IO or RUBY_UBF_PROCESS. Instead of a permanent thread in <= 2.5, we can now rely on the thread cache feature to perform interrupts. [ruby-core:90865] [Bug #15499] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/branches/ruby_2_6@66940 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rw-r--r--thread.c8
-rw-r--r--thread_pthread.c19
-rw-r--r--thread_win32.c6
-rw-r--r--version.h2
4 files changed, 34 insertions, 1 deletions
diff --git a/thread.c b/thread.c
index 58e7651446..eca14b4b4c 100644
--- a/thread.c
+++ b/thread.c
@@ -1429,11 +1429,15 @@ call_without_gvl(void *(*func)(void *), void *data1,
rb_execution_context_t *ec = GET_EC();
rb_thread_t *th = rb_ec_thread_ptr(ec);
int saved_errno = 0;
+ VALUE ubf_th = Qfalse;
if (ubf == RUBY_UBF_IO || ubf == RUBY_UBF_PROCESS) {
ubf = ubf_select;
data2 = th;
}
+ else if (ubf && vm_living_thread_num(th->vm) == 1) {
+ ubf_th = rb_thread_start_unblock_thread();
+ }
BLOCKING_REGION(th, {
val = func(data1);
@@ -1444,6 +1448,10 @@ call_without_gvl(void *(*func)(void *), void *data1,
RUBY_VM_CHECK_INTS_BLOCKING(ec);
}
+ if (ubf_th != Qfalse) {
+ thread_value(rb_thread_kill(ubf_th));
+ }
+
errno = saved_errno;
return val;
diff --git a/thread_pthread.c b/thread_pthread.c
index e074a854f5..d8d3184c62 100644
--- a/thread_pthread.c
+++ b/thread_pthread.c
@@ -2206,4 +2206,23 @@ timer_pthread_fn(void *p)
return 0;
}
#endif /* UBF_TIMER_PTHREAD */
+
+static VALUE
+ubf_caller(const void *ignore)
+{
+ rb_thread_sleep_forever();
+
+ return Qfalse;
+}
+
+/*
+ * Called if and only if one thread is running, and
+ * the unblock function is NOT async-signal-safe
+ * This assumes USE_THREAD_CACHE is true for performance reasons
+ */
+static VALUE
+rb_thread_start_unblock_thread(void)
+{
+ return rb_thread_create(ubf_caller, 0);
+}
#endif /* THREAD_SYSTEM_DEPENDENT_IMPLEMENTATION */
diff --git a/thread_win32.c b/thread_win32.c
index d4db9e3824..545d0dd35b 100644
--- a/thread_win32.c
+++ b/thread_win32.c
@@ -708,6 +708,12 @@ rb_thread_wakeup_timer_thread(int sig)
/* do nothing */
}
+static VALUE
+rb_thread_start_unblock_thread(void)
+{
+ return Qfalse; /* no-op */
+}
+
static void
rb_thread_create_timer_thread(void)
{
diff --git a/version.h b/version.h
index e16812c628..da5327b96c 100644
--- a/version.h
+++ b/version.h
@@ -1,6 +1,6 @@
#define RUBY_VERSION "2.6.1"
#define RUBY_RELEASE_DATE RUBY_RELEASE_YEAR_STR"-"RUBY_RELEASE_MONTH_STR"-"RUBY_RELEASE_DAY_STR
-#define RUBY_PATCHLEVEL 30
+#define RUBY_PATCHLEVEL 31
#define RUBY_RELEASE_YEAR 2019
#define RUBY_RELEASE_MONTH 1