From: Heikki Linnakangas Date: Mon, 10 Feb 2014 13:14:07 +0000 (+0200) Subject: Fix WakeupWaiters() to not wake up an exclusive locker unnecessarily. X-Git-Tag: REL9_4_BETA1~514 X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=d699ba41349e4ef397222a7223606fa03f4c4870;p=postgresql Fix WakeupWaiters() to not wake up an exclusive locker unnecessarily. WakeupWaiters() is supposed to wake up all LW_WAIT_UNTIL_FREE waiters of the slot, but the loop incorrectly also woke up the first LW_EXCLUSIVE waiter, if there was no LW_WAIT_UNTIL_FREE waiters in the queue. Noted by Andres Freund. This code is new in 9.4, so no backpatching. --- diff --git a/src/backend/access/transam/xlog.c b/src/backend/access/transam/xlog.c index 7f63185b1c..508970a751 100644 --- a/src/backend/access/transam/xlog.c +++ b/src/backend/access/transam/xlog.c @@ -1842,15 +1842,14 @@ WakeupWaiters(XLogRecPtr EndPos) slot->xlogInsertingAt = EndPos; /* - * See if there are any waiters that need to be woken up. + * See if there are any LW_WAIT_UNTIL_FREE waiters that need to be woken + * up. They are always in the front of the queue. */ head = slot->head; - if (head != NULL) + if (head != NULL && head->lwWaitMode == LW_WAIT_UNTIL_FREE) { proc = head; - - /* LW_WAIT_UNTIL_FREE waiters are always in the front of the queue */ next = proc->lwWaitLink; while (next && next->lwWaitMode == LW_WAIT_UNTIL_FREE) { @@ -1862,6 +1861,8 @@ WakeupWaiters(XLogRecPtr EndPos) slot->head = next; proc->lwWaitLink = NULL; } + else + head = NULL; /* We are done updating shared state of the lock itself. */ SpinLockRelease(&slot->mutex);