diff options
author | jhb <jhb@FreeBSD.org> | 2003-12-09 21:09:04 +0000 |
---|---|---|
committer | jhb <jhb@FreeBSD.org> | 2003-12-09 21:09:04 +0000 |
commit | 66cc89fadfbdc13372fd21d89b6a300f5a2f35d0 (patch) | |
tree | e36dcfd14da390fbb45217cc26f2728afc87360a /sys/kern | |
parent | e6cf44c49dace75cb69cfec772707422391cbdfa (diff) | |
download | FreeBSD-src-66cc89fadfbdc13372fd21d89b6a300f5a2f35d0.zip FreeBSD-src-66cc89fadfbdc13372fd21d89b6a300f5a2f35d0.tar.gz |
Revert the previous race fix and replace it with a more general fix. The
case of a turnstile having no threads is just one instance of the more
general case where the thread we are examining has been partially awakened
already in that it has been removed from the turnstile's blocked list but
still has TDI_LOCK set. We detect that case by checking to see if the
thread has already had a turnstile reassigned to it.
Diffstat (limited to 'sys/kern')
-rw-r--r-- | sys/kern/subr_turnstile.c | 17 |
1 files changed, 9 insertions, 8 deletions
diff --git a/sys/kern/subr_turnstile.c b/sys/kern/subr_turnstile.c index 8f8de89..baf5d8e 100644 --- a/sys/kern/subr_turnstile.c +++ b/sys/kern/subr_turnstile.c @@ -233,11 +233,16 @@ propagate_priority(struct thread *td) mtx_lock_spin(&tc->tc_lock); /* - * If this turnstile has no threads on its blocked queue - * then it's possible that it was just woken up on another - * CPU. If so, we are done. + * This thread may not be blocked on this turnstile anymore + * but instead might already be woken up on another CPU + * that is waiting on sched_lock in turnstile_unpend() to + * finish waking this thread up. We can detect this case + * by checking to see if this thread has been given a + * turnstile by either turnstile_signal() or + * turnstile_wakeup(). In this case, treat the thread as + * if it was already running. */ - if (TAILQ_EMPTY(&ts->ts_blocked)) { + if (td->td_turnstile != NULL) { mtx_unlock_spin(&tc->tc_lock); return; } @@ -300,9 +305,7 @@ init_turnstiles(void) NULL, MTX_SPIN); } mtx_init(&td_contested_lock, "td_contested", NULL, MTX_SPIN); -#ifdef INVARIANTS thread0.td_turnstile = NULL; -#endif } static void @@ -468,9 +471,7 @@ turnstile_wait(struct turnstile *ts, struct lock_object *lock, LIST_INSERT_HEAD(&ts->ts_free, td->td_turnstile, ts_hash); MPASS(owner == ts->ts_owner); } -#ifdef INVARIANTS td->td_turnstile = NULL; -#endif mtx_unlock_spin(&tc->tc_lock); mtx_lock_spin(&sched_lock); |