[前][次][番号順一覧][スレッド一覧]

ruby-changes:46760

From: normal <ko1@a...>
Date: Thu, 25 May 2017 03:59:31 +0900 (JST)
Subject: [ruby-changes:46760] normal:r58874 (trunk): thread_pthread: retry timer thread creation w/o attr on EINVAL

normal	2017-05-25 03:59:24 +0900 (Thu, 25 May 2017)

  New Revision: 58874

  https://svn.ruby-lang.org/cgi-bin/viewvc.cgi?view=revision&revision=58874

  Log:
    thread_pthread: retry timer thread creation w/o attr on EINVAL
    
    Setting a small stack size can fail due to having 3rd-party
    libraries (e.g. libkqueue) loaded, if those libraries use
    thread-local-storage (__thread) heavily.  This causes
    pthread_create to fail with small stacks; even if our
    timer_thread function does not hit any of the TLS-using code
    paths.
    
    Today, some RubyGems are capable of using libkqueue (or __thread
    storage directly), and future versions of Ruby may use kqueue
    internally.
    
    cf. https://www.akkadia.org/drepper/tls.pdf

  Modified files:
    trunk/thread_pthread.c
Index: thread_pthread.c
===================================================================
--- thread_pthread.c	(revision 58873)
+++ thread_pthread.c	(revision 58874)
@@ -1587,6 +1587,7 @@ rb_thread_create_timer_thread(void) https://github.com/ruby/ruby/blob/trunk/thread_pthread.c#L1587
 	int err;
 #ifdef HAVE_PTHREAD_ATTR_INIT
 	pthread_attr_t attr;
+	rb_vm_t *vm = GET_VM();
 
 	err = pthread_attr_init(&attr);
 	if (err != 0) {
@@ -1623,10 +1624,20 @@ rb_thread_create_timer_thread(void) https://github.com/ruby/ruby/blob/trunk/thread_pthread.c#L1624
 	    rb_bug("rb_thread_create_timer_thread: Timer thread was already created\n");
 	}
 #ifdef HAVE_PTHREAD_ATTR_INIT
-	err = pthread_create(&timer_thread.id, &attr, thread_timer, &GET_VM()->gvl);
+	err = pthread_create(&timer_thread.id, &attr, thread_timer, &vm->gvl);
 	pthread_attr_destroy(&attr);
+
+	if (err == EINVAL) {
+	    /*
+	     * Even if we are careful with our own stack use in thread_timer(),
+	     * any third-party libraries (eg libkqueue) which rely on __thread
+	     * storage can cause small stack sizes to fail.  So lets hope the
+	     * default stack size is enough for them:
+	     */
+	    err = pthread_create(&timer_thread.id, NULL, thread_timer, &vm->gvl);
+	}
 #else
-	err = pthread_create(&timer_thread.id, NULL, thread_timer, &GET_VM()->gvl);
+	err = pthread_create(&timer_thread.id, NULL, thread_timer, &vm->gvl);
 #endif
 	if (err != 0) {
 	    rb_warn("pthread_create failed for timer: %s, scheduling broken",

--
ML: ruby-changes@q...
Info: http://www.atdot.net/~ko1/quickml/

[前][次][番号順一覧][スレッド一覧]