]> granicus.if.org Git - postgresql/commitdiff
Prevent potentially hazardous compiler/cpu reordering during lwlock release.
authorAndres Freund <andres@anarazel.de>
Fri, 19 Dec 2014 13:29:52 +0000 (14:29 +0100)
committerAndres Freund <andres@anarazel.de>
Fri, 19 Dec 2014 13:32:47 +0000 (14:32 +0100)
In LWLockRelease() (and in 9.4+ LWLockUpdateVar()) we release enqueued
waiters using PGSemaphoreUnlock(). As there are other sources of such
unlocks backends only wake up if MyProc->lwWaiting is set to false;
which is only done in the aforementioned functions.

Before this commit there were dangers because the store to lwWaitLink
could become visible before the store to lwWaitLink. This could both
happen due to compiler reordering (on most compilers) and on some
platforms due to the CPU reordering stores.

The possible consequence of this is that a backend stops waiting
before lwWaitLink is set to NULL. If that backend then tries to
acquire another lock and has to wait there the list could become
corrupted once the lwWaitLink store is finally performed.

Add a write memory barrier to prevent that issue.

Unfortunately the barrier support has been only added in 9.2. Given
that the issue has not knowingly been observed in praxis it seems
sufficient to prohibit compiler reordering using volatile for 9.0 and
9.1. Actual problems due to compiler reordering are more likely
anyway.

Discussion: 20140210134625.GA15246@awork2.anarazel.de

src/backend/storage/lmgr/lwlock.c

index 88d9cbec21568791126acef6ace2a7025e0bf3ff..196b1110326b285b030c9c5ce306549d279206f6 100644 (file)
@@ -35,6 +35,7 @@
 #include "miscadmin.h"
 #include "pg_trace.h"
 #include "replication/slot.h"
+#include "storage/barrier.h"
 #include "storage/ipc.h"
 #include "storage/predicate.h"
 #include "storage/proc.h"
@@ -1102,6 +1103,8 @@ LWLockUpdateVar(LWLock *l, uint64 *valptr, uint64 val)
                proc = head;
                head = proc->lwWaitLink;
                proc->lwWaitLink = NULL;
+               /* check comment in LWLockRelease() about this barrier */
+               pg_write_barrier();
                proc->lwWaiting = false;
                PGSemaphoreUnlock(&proc->sem);
        }
@@ -1222,6 +1225,17 @@ LWLockRelease(LWLock *l)
                proc = head;
                head = proc->lwWaitLink;
                proc->lwWaitLink = NULL;
+               /*
+                * Guarantee that lwWaiting being unset only becomes visible once the
+                * unlink from the link has completed. Otherwise the target backend
+                * could be woken up for other reason and enqueue for a new lock - if
+                * that happens before the list unlink happens, the list would end up
+                * being corrupted.
+                *
+                * The barrier pairs with the SpinLockAcquire() when enqueing for
+                * another lock.
+                */
+               pg_write_barrier();
                proc->lwWaiting = false;
                PGSemaphoreUnlock(&proc->sem);
        }