Project

General

Profile

Actions

Bug #14945

closed

[PATCH] thread.c (blocking_region_end): clear ubf before unregister_ubf_list

Added by normalperson (Eric Wong) over 5 years ago. Updated over 5 years ago.

Status:
Closed
Assignee:
-
Target version:
-
[ruby-core:88141]

Description

thread.c (blocking_region_end): clear ubf before unregister_ubf_list

If we keep ubf set after unregistering, there is a window for
other threads (including timer thread) to put this thread back
on the ubf_list right away. Entering ubf_list unexpectedly
after GVL acquisition may cause spurious wakeup and trigger
unexpected behavior.

Finally, clear ubf before acquiring GVL, to since ubf is useless
during GVL acquisition anyways and we don't want to waste cycles
in other threads calling ubf for useless work.

I found this bug while rewriting GVL to handle timer-thread duty
and eliminate separate timer thread.


Files

Actions #1

Updated by normalperson (Eric Wong) over 5 years ago

  • Status changed from Open to Closed

Applied in changeset trunk|r64083.


thread.c (blocking_region_end): clear ubf before unregister_ubf_list

If we keep ubf set after unregistering, there is a window for
other threads (including timer thread) to put this thread back
on the ubf_list right away. Entering ubf_list unexpectedly
after GVL acquisition may cause spurious wakeup and trigger
unexpected behavior.

Finally, clear ubf before acquiring GVL, to since ubf is useless
during GVL acquisition anyways and we don't want to waste cycles
in other threads calling ubf for useless work.

[ruby-core:88141] [Bug #14945]

Updated by normalperson (Eric Wong) over 5 years ago

https://bugs.ruby-lang.org/issues/14945

Actually, r64083 may not be completely safe, either, as
the list_empty checks are dangerous, at least w/o GVL.

However, this order must be enforced:

unblock_function_clear
unregister_ubf_list

Working on fixes...

Updated by ko1 (Koichi Sasada) over 5 years ago

:+1:

One off topic question. list_empty() is thread-safe?

Updated by ko1 (Koichi Sasada) over 5 years ago

ko1 (Koichi Sasada) wrote:

One off topic question. list_empty() is thread-safe?

Sorry it should be safe (I misread as other operation. sorry).

Updated by normalperson (Eric Wong) over 5 years ago

wrote:

ko1 (Koichi Sasada) wrote:

One off topic question. list_empty() is thread-safe?

Sorry it should be safe (I misread as other operation. sorry).

Right, good question, though... for register_ubf_list
and unregister_ubf_list, they are safe and I can commit
below patch to clarify.

However, list_empty in ubf_threads_empty I'm not 100% sure
about...

diff --git a/thread_pthread.c b/thread_pthread.c
index 29805ef2df..febec0c1d1 100644
--- a/thread_pthread.c
+++ b/thread_pthread.c
@@ -1125,6 +1125,15 @@ register_ubf_list(rb_thread_t *th)
{
struct list_node *node = &th->native_thread_data.ubf_list;

+    /*
+     * list_empty check is safe here without ubf_list_lock held
+     * because th->interrupt_lock is already held by
+     * rb_threadptr_interrupt_common.
+     *
+     *  rb_threadptr_interrupt_common
+     *    ubf_select (== th->unblock.func)
+     *      register_ubf_list (this function)
+     */
if (list_empty((struct list_head*)node)) {
rb_native_mutex_lock(&ubf_list_lock);
	list_add(&ubf_list_head, node);
@@ -1141,6 +1150,11 @@ unregister_ubf_list(rb_thread_t *th)
/* we can't allow re-entry into ubf_list_head */
VM_ASSERT(th->unblock.func == 0);

+    /*
+     * list_empty check is safe here without ubf_list_lock held
+     * because we already cleared th->unblock.func while
+     * th->interrupt_lock was held.
+     */
if (!list_empty((struct list_head*)node)) {
rb_native_mutex_lock(&ubf_list_lock);
	list_del_init(node);

Actions

Also available in: Atom PDF

Like0
Like0Like0Like0Like0Like0