Bug #14945
closed[PATCH] thread.c (blocking_region_end): clear ubf before unregister_ubf_list
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
Updated by normalperson (Eric Wong) over 6 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 6 years ago
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) about 6 years ago
:+1:
One off topic question. list_empty()
is thread-safe?
Updated by ko1 (Koichi Sasada) about 6 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) about 6 years ago
ko1@atdot.net 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);