Commit 7714c638 authored by Heikki Linnakangas's avatar Heikki Linnakangas

Remove extra copies of LogwrtResult.

This simplifies the code a little bit. The new rule is that to update
XLogCtl->LogwrtResult, you must hold both WALWriteLock and info_lck, whereas
before we had two copies, one that was protected by WALWriteLock and another
protected by info_lck. The code that updates them was already holding both
locks, so merging the two is trivial.

The third copy, XLogCtl->Insert.LogwrtResult, was not totally redundant, it
was used in AdvanceXLInsertBuffer to update the backend-local copy, before
acquiring the info_lck to read the up-to-date value. But the value of that
seems dubious; at best it's saving one spinlock acquisition per completed
WAL page, which is not significant compared to all the other work involved.
And in practice, it's probably not saving even that much.
parent 3b682df3
...@@ -289,35 +289,16 @@ static XLogRecPtr RedoStartLSN = {0, 0}; ...@@ -289,35 +289,16 @@ static XLogRecPtr RedoStartLSN = {0, 0};
* These structs are identical but are declared separately to indicate their * These structs are identical but are declared separately to indicate their
* slightly different functions. * slightly different functions.
* *
* We do a lot of pushups to minimize the amount of access to lockable * To read XLogCtl->LogwrtResult, you must hold either info_lck or
* shared memory values. There are actually three shared-memory copies of * WALWriteLock. To update it, you need to hold both locks. The point of
* LogwrtResult, plus one unshared copy in each backend. Here's how it works: * this arrangement is that the value can be examined by code that already
* XLogCtl->LogwrtResult is protected by info_lck * holds WALWriteLock without needing to grab info_lck as well. In addition
* XLogCtl->Write.LogwrtResult is protected by WALWriteLock * to the shared variable, each backend has a private copy of LogwrtResult,
* XLogCtl->Insert.LogwrtResult is protected by WALInsertLock * which is updated when convenient.
* One must hold the associated lock to read or write any of these, but
* of course no lock is needed to read/write the unshared LogwrtResult.
*
* XLogCtl->LogwrtResult and XLogCtl->Write.LogwrtResult are both "always
* right", since both are updated by a write or flush operation before
* it releases WALWriteLock. The point of keeping XLogCtl->Write.LogwrtResult
* is that it can be examined/modified by code that already holds WALWriteLock
* without needing to grab info_lck as well.
*
* XLogCtl->Insert.LogwrtResult may lag behind the reality of the other two,
* but is updated when convenient. Again, it exists for the convenience of
* code that is already holding WALInsertLock but not the other locks.
*
* The unshared LogwrtResult may lag behind any or all of these, and again
* is updated when convenient.
* *
* The request bookkeeping is simpler: there is a shared XLogCtl->LogwrtRqst * The request bookkeeping is simpler: there is a shared XLogCtl->LogwrtRqst
* (protected by info_lck), but we don't need to cache any copies of it. * (protected by info_lck), but we don't need to cache any copies of it.
* *
* Note that this all works because the request and result positions can only
* advance forward, never back up, and so we can easily determine which of two
* values is "more up to date".
*
* info_lck is only held long enough to read/update the protected variables, * info_lck is only held long enough to read/update the protected variables,
* so it's a plain spinlock. The other locks are held longer (potentially * so it's a plain spinlock. The other locks are held longer (potentially
* over I/O operations), so we use LWLocks for them. These locks are: * over I/O operations), so we use LWLocks for them. These locks are:
...@@ -354,7 +335,6 @@ typedef struct XLogwrtResult ...@@ -354,7 +335,6 @@ typedef struct XLogwrtResult
*/ */
typedef struct XLogCtlInsert typedef struct XLogCtlInsert
{ {
XLogwrtResult LogwrtResult; /* a recent value of LogwrtResult */
XLogRecPtr PrevRecord; /* start of previously-inserted record */ XLogRecPtr PrevRecord; /* start of previously-inserted record */
int curridx; /* current block index in cache */ int curridx; /* current block index in cache */
XLogPageHeader currpage; /* points to header of block in cache */ XLogPageHeader currpage; /* points to header of block in cache */
...@@ -388,7 +368,6 @@ typedef struct XLogCtlInsert ...@@ -388,7 +368,6 @@ typedef struct XLogCtlInsert
*/ */
typedef struct XLogCtlWrite typedef struct XLogCtlWrite
{ {
XLogwrtResult LogwrtResult; /* current value of LogwrtResult */
int curridx; /* cache index of next block to write */ int curridx; /* cache index of next block to write */
pg_time_t lastSegSwitchTime; /* time of last xlog segment switch */ pg_time_t lastSegSwitchTime; /* time of last xlog segment switch */
} XLogCtlWrite; } XLogCtlWrite;
...@@ -403,7 +382,6 @@ typedef struct XLogCtlData ...@@ -403,7 +382,6 @@ typedef struct XLogCtlData
/* Protected by info_lck: */ /* Protected by info_lck: */
XLogwrtRqst LogwrtRqst; XLogwrtRqst LogwrtRqst;
XLogwrtResult LogwrtResult;
uint32 ckptXidEpoch; /* nextXID & epoch of latest checkpoint */ uint32 ckptXidEpoch; /* nextXID & epoch of latest checkpoint */
TransactionId ckptXid; TransactionId ckptXid;
XLogRecPtr asyncXactLSN; /* LSN of newest async commit/abort */ XLogRecPtr asyncXactLSN; /* LSN of newest async commit/abort */
...@@ -413,6 +391,12 @@ typedef struct XLogCtlData ...@@ -413,6 +391,12 @@ typedef struct XLogCtlData
/* Protected by WALWriteLock: */ /* Protected by WALWriteLock: */
XLogCtlWrite Write; XLogCtlWrite Write;
/*
* Protected by info_lck and WALWriteLock (you must hold either lock to
* read it, but both to update)
*/
XLogwrtResult LogwrtResult;
/* /*
* These values do not change after startup, although the pointed-to pages * These values do not change after startup, although the pointed-to pages
* and xlblocks values certainly do. Permission to read/write the pages * and xlblocks values certainly do. Permission to read/write the pages
...@@ -1015,7 +999,7 @@ begin:; ...@@ -1015,7 +999,7 @@ begin:;
} }
LWLockAcquire(WALWriteLock, LW_EXCLUSIVE); LWLockAcquire(WALWriteLock, LW_EXCLUSIVE);
LogwrtResult = XLogCtl->Write.LogwrtResult; LogwrtResult = XLogCtl->LogwrtResult;
if (!XLByteLE(RecPtr, LogwrtResult.Flush)) if (!XLByteLE(RecPtr, LogwrtResult.Flush))
{ {
XLogwrtRqst FlushRqst; XLogwrtRqst FlushRqst;
...@@ -1188,8 +1172,6 @@ begin:; ...@@ -1188,8 +1172,6 @@ begin:;
SpinLockRelease(&xlogctl->info_lck); SpinLockRelease(&xlogctl->info_lck);
} }
Write->LogwrtResult = LogwrtResult;
LWLockRelease(WALWriteLock); LWLockRelease(WALWriteLock);
updrqst = false; /* done already */ updrqst = false; /* done already */
...@@ -1477,7 +1459,6 @@ static bool ...@@ -1477,7 +1459,6 @@ static bool
AdvanceXLInsertBuffer(bool new_segment) AdvanceXLInsertBuffer(bool new_segment)
{ {
XLogCtlInsert *Insert = &XLogCtl->Insert; XLogCtlInsert *Insert = &XLogCtl->Insert;
XLogCtlWrite *Write = &XLogCtl->Write;
int nextidx = NextBufIdx(Insert->curridx); int nextidx = NextBufIdx(Insert->curridx);
bool update_needed = true; bool update_needed = true;
XLogRecPtr OldPageRqstPtr; XLogRecPtr OldPageRqstPtr;
...@@ -1485,10 +1466,6 @@ AdvanceXLInsertBuffer(bool new_segment) ...@@ -1485,10 +1466,6 @@ AdvanceXLInsertBuffer(bool new_segment)
XLogRecPtr NewPageEndPtr; XLogRecPtr NewPageEndPtr;
XLogPageHeader NewPage; XLogPageHeader NewPage;
/* Use Insert->LogwrtResult copy if it's more fresh */
if (XLByteLT(LogwrtResult.Write, Insert->LogwrtResult.Write))
LogwrtResult = Insert->LogwrtResult;
/* /*
* Get ending-offset of the buffer page we need to replace (this may be * Get ending-offset of the buffer page we need to replace (this may be
* zero if the buffer hasn't been used yet). Fall through if it's already * zero if the buffer hasn't been used yet). Fall through if it's already
...@@ -1516,21 +1493,19 @@ AdvanceXLInsertBuffer(bool new_segment) ...@@ -1516,21 +1493,19 @@ AdvanceXLInsertBuffer(bool new_segment)
update_needed = false; /* Did the shared-request update */ update_needed = false; /* Did the shared-request update */
if (XLByteLE(OldPageRqstPtr, LogwrtResult.Write)) /*
{ * Now that we have an up-to-date LogwrtResult value, see if we still
/* OK, someone wrote it already */ * need to write it or if someone else already did.
Insert->LogwrtResult = LogwrtResult; */
} if (!XLByteLE(OldPageRqstPtr, LogwrtResult.Write))
else
{ {
/* Must acquire write lock */ /* Must acquire write lock */
LWLockAcquire(WALWriteLock, LW_EXCLUSIVE); LWLockAcquire(WALWriteLock, LW_EXCLUSIVE);
LogwrtResult = Write->LogwrtResult; LogwrtResult = XLogCtl->LogwrtResult;
if (XLByteLE(OldPageRqstPtr, LogwrtResult.Write)) if (XLByteLE(OldPageRqstPtr, LogwrtResult.Write))
{ {
/* OK, someone wrote it already */ /* OK, someone wrote it already */
LWLockRelease(WALWriteLock); LWLockRelease(WALWriteLock);
Insert->LogwrtResult = LogwrtResult;
} }
else else
{ {
...@@ -1544,7 +1519,6 @@ AdvanceXLInsertBuffer(bool new_segment) ...@@ -1544,7 +1519,6 @@ AdvanceXLInsertBuffer(bool new_segment)
WriteRqst.Flush.xrecoff = 0; WriteRqst.Flush.xrecoff = 0;
XLogWrite(WriteRqst, false, false); XLogWrite(WriteRqst, false, false);
LWLockRelease(WALWriteLock); LWLockRelease(WALWriteLock);
Insert->LogwrtResult = LogwrtResult;
TRACE_POSTGRESQL_WAL_BUFFER_WRITE_DIRTY_DONE(); TRACE_POSTGRESQL_WAL_BUFFER_WRITE_DIRTY_DONE();
} }
} }
...@@ -1697,7 +1671,7 @@ XLogWrite(XLogwrtRqst WriteRqst, bool flexible, bool xlog_switch) ...@@ -1697,7 +1671,7 @@ XLogWrite(XLogwrtRqst WriteRqst, bool flexible, bool xlog_switch)
/* /*
* Update local LogwrtResult (caller probably did this already, but...) * Update local LogwrtResult (caller probably did this already, but...)
*/ */
LogwrtResult = Write->LogwrtResult; LogwrtResult = XLogCtl->LogwrtResult;
/* /*
* Since successive pages in the xlog cache are consecutively allocated, * Since successive pages in the xlog cache are consecutively allocated,
...@@ -1931,8 +1905,6 @@ XLogWrite(XLogwrtRqst WriteRqst, bool flexible, bool xlog_switch) ...@@ -1931,8 +1905,6 @@ XLogWrite(XLogwrtRqst WriteRqst, bool flexible, bool xlog_switch)
xlogctl->LogwrtRqst.Flush = LogwrtResult.Flush; xlogctl->LogwrtRqst.Flush = LogwrtResult.Flush;
SpinLockRelease(&xlogctl->info_lck); SpinLockRelease(&xlogctl->info_lck);
} }
Write->LogwrtResult = LogwrtResult;
} }
/* /*
...@@ -2126,7 +2098,7 @@ XLogFlush(XLogRecPtr record) ...@@ -2126,7 +2098,7 @@ XLogFlush(XLogRecPtr record)
continue; continue;
} }
/* Got the lock */ /* Got the lock */
LogwrtResult = XLogCtl->Write.LogwrtResult; LogwrtResult = XLogCtl->LogwrtResult;
if (!XLByteLE(record, LogwrtResult.Flush)) if (!XLByteLE(record, LogwrtResult.Flush))
{ {
/* try to write/flush later additions to XLOG as well */ /* try to write/flush later additions to XLOG as well */
...@@ -2268,7 +2240,7 @@ XLogBackgroundFlush(void) ...@@ -2268,7 +2240,7 @@ XLogBackgroundFlush(void)
/* now wait for the write lock */ /* now wait for the write lock */
LWLockAcquire(WALWriteLock, LW_EXCLUSIVE); LWLockAcquire(WALWriteLock, LW_EXCLUSIVE);
LogwrtResult = XLogCtl->Write.LogwrtResult; LogwrtResult = XLogCtl->LogwrtResult;
if (!XLByteLE(WriteRqstPtr, LogwrtResult.Flush)) if (!XLByteLE(WriteRqstPtr, LogwrtResult.Flush))
{ {
XLogwrtRqst WriteRqst; XLogwrtRqst WriteRqst;
...@@ -6831,8 +6803,6 @@ StartupXLOG(void) ...@@ -6831,8 +6803,6 @@ StartupXLOG(void)
LogwrtResult.Write = LogwrtResult.Flush = EndOfLog; LogwrtResult.Write = LogwrtResult.Flush = EndOfLog;
XLogCtl->Write.LogwrtResult = LogwrtResult;
Insert->LogwrtResult = LogwrtResult;
XLogCtl->LogwrtResult = LogwrtResult; XLogCtl->LogwrtResult = LogwrtResult;
XLogCtl->LogwrtRqst.Write = EndOfLog; XLogCtl->LogwrtRqst.Write = EndOfLog;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment