]> git.ipfire.org Git - thirdparty/postgresql.git/commitdiff
At end of recovery, reset all sinval-managed caches.
authorNoah Misch <noah@leadboat.com>
Fri, 25 Oct 2024 13:51:06 +0000 (06:51 -0700)
committerNoah Misch <noah@leadboat.com>
Fri, 25 Oct 2024 13:51:06 +0000 (06:51 -0700)
An inplace update's invalidation messages are part of its transaction's
commit record.  However, the update survives even if its transaction
aborts or we stop recovery before replaying its transaction commit.
After recovery, a backend that started in recovery could update the row
without incorporating the inplace update.  That could result in a table
with an index, yet relhasindex=f.  That is a source of index corruption.

This bulk invalidation avoids the functional consequences.  A future
change can fix the !RecoveryInProgress() scenario without changing the
WAL format.  Back-patch to v17 - v12 (all supported versions).  v18 will
instead add invalidations to WAL.

Discussion: https://postgr.es/m/20240618152349.7f.nmisch@google.com

src/backend/access/transam/xlog.c
src/backend/storage/ipc/sinvaladt.c
src/include/storage/sinvaladt.h

index 7f1360262776ba2e7e002ba5b25abbe66ac37961..c1dd8b543289c9e9eaee1c6fc7ca7483cd74eb27 100644 (file)
@@ -92,6 +92,7 @@
 #include "storage/proc.h"
 #include "storage/procarray.h"
 #include "storage/reinit.h"
+#include "storage/sinvaladt.h"
 #include "storage/spin.h"
 #include "storage/sync.h"
 #include "utils/guc_hooks.h"
@@ -6037,6 +6038,30 @@ StartupXLOG(void)
        XLogCtl->LogwrtRqst.Write = EndOfLog;
        XLogCtl->LogwrtRqst.Flush = EndOfLog;
 
+       /*
+        * Invalidate all sinval-managed caches before READ WRITE transactions
+        * begin.  The xl_heap_inplace WAL record doesn't store sufficient data
+        * for invalidations.  The commit record, if any, has the invalidations.
+        * However, the inplace update is permanent, whether or not we reach a
+        * commit record.  Fortunately, read-only transactions tolerate caches not
+        * reflecting the latest inplace updates.  Read-only transactions
+        * experience the notable inplace updates as follows:
+        *
+        * - relhasindex=true affects readers only after the CREATE INDEX
+        * transaction commit makes an index fully available to them.
+        *
+        * - datconnlimit=DATCONNLIMIT_INVALID_DB affects readers only at
+        * InitPostgres() time, and that read does not use a cache.
+        *
+        * - relfrozenxid, datfrozenxid, relminmxid, and datminmxid have no effect
+        * on readers.
+        *
+        * Hence, hot standby queries (all READ ONLY) function correctly without
+        * the missing invalidations.  This avoided changing the WAL format in
+        * back branches.
+        */
+       SIResetAll();
+
        /*
         * Preallocate additional log files, if wanted.
         */
index b486d8ddd1df64b226f2ff6403f471f88e1176b9..271212987e0cea44f119b1ce1080653bdec5af9f 100644 (file)
@@ -683,6 +683,48 @@ SICleanupQueue(bool callerHasWriteLock, int minFree)
        }
 }
 
+/*
+ * SIResetAll
+ *             Mark all active backends as "reset"
+ *
+ * Use this when we don't know what needs to be invalidated.  It's a
+ * cluster-wide InvalidateSystemCaches().  This was a back-branch-only remedy
+ * to avoid a WAL format change.
+ *
+ * The implementation is like SICleanupQueue(false, MAXNUMMESSAGES + 1), with
+ * one addition.  SICleanupQueue() assumes minFree << MAXNUMMESSAGES, so it
+ * assumes hasMessages==true for any backend it resets.  We're resetting even
+ * fully-caught-up backends, so we set hasMessages.
+ */
+void
+SIResetAll(void)
+{
+       SISeg      *segP = shmInvalBuffer;
+       int                     i;
+
+       LWLockAcquire(SInvalWriteLock, LW_EXCLUSIVE);
+       LWLockAcquire(SInvalReadLock, LW_EXCLUSIVE);
+
+       for (i = 0; i < segP->numProcs; i++)
+       {
+               ProcState  *stateP = &segP->procState[segP->pgprocnos[i]];
+
+               Assert(stateP->procPid != 0);
+               if (stateP->sendOnly)
+                       continue;
+
+               /* Consuming the reset will update "nextMsgNum" and "signaled". */
+               stateP->resetState = true;
+               stateP->hasMessages = true;
+       }
+
+       segP->minMsgNum = segP->maxMsgNum;
+       segP->nextThreshold = CLEANUP_MIN;
+
+       LWLockRelease(SInvalReadLock);
+       LWLockRelease(SInvalWriteLock);
+}
+
 
 /*
  * GetNextLocalTransactionId --- allocate a new LocalTransactionId
index c3c97b3f8b7ed6883817261773e53ef20d63a6eb..7c8f450db7c3095fea109bc950522eaefd88fc38 100644 (file)
@@ -35,6 +35,7 @@ extern void SharedInvalBackendInit(bool sendOnly);
 extern void SIInsertDataEntries(const SharedInvalidationMessage *data, int n);
 extern int     SIGetDataEntries(SharedInvalidationMessage *data, int datasize);
 extern void SICleanupQueue(bool callerHasWriteLock, int minFree);
+extern void SIResetAll(void);
 
 extern LocalTransactionId GetNextLocalTransactionId(void);