aboutsummaryrefslogtreecommitdiff
path: root/src/backend/access/transam/xlog.c
diff options
context:
space:
mode:
authorAmit Kapila <akapila@postgresql.org>2018-09-13 16:10:59 +0530
committerAmit Kapila <akapila@postgresql.org>2018-09-13 16:10:59 +0530
commit47a589c1fe35176b38ffcdb2a95b4fa2ef45add2 (patch)
tree8856d5fa06dc116c0ce3bd4a4b4bd8951cae18a1 /src/backend/access/transam/xlog.c
parent93a508efb2456517af6d8756ddff4a427aee4a0c (diff)
downloadpostgresql-47a589c1fe35176b38ffcdb2a95b4fa2ef45add2.tar.gz
postgresql-47a589c1fe35176b38ffcdb2a95b4fa2ef45add2.zip
Attach FPI to the first record after full_page_writes is turned on.
XLogInsert fails to attach a required FPI to the first record after full_page_writes is turned on by the last checkpoint. This bug got introduced in 9.5 due to code rearrangement in commits 2c03216d83 and 2076db2aea. Fix it by ensuring that XLogInsertRecord performs a recomputation when the given record is generated with FPW as off but found that the flag has been turned on while actually inserting the record. Reported-by: Kyotaro Horiguchi Author: Kyotaro Horiguchi Reviewed-by: Amit Kapila Backpatch-through: 9.5 where this problem was introduced Discussion: https://postgr.es/m/20180420.151043.74298611.horiguchi.kyotaro@lab.ntt.co.jp
Diffstat (limited to 'src/backend/access/transam/xlog.c')
-rw-r--r--src/backend/access/transam/xlog.c19
1 files changed, 13 insertions, 6 deletions
diff --git a/src/backend/access/transam/xlog.c b/src/backend/access/transam/xlog.c
index 0e95b3cef81..b4ea4d40544 100644
--- a/src/backend/access/transam/xlog.c
+++ b/src/backend/access/transam/xlog.c
@@ -909,7 +909,7 @@ static void WALInsertLockUpdateInsertingAt(XLogRecPtr insertingAt);
*
* If 'fpw_lsn' is valid, it is the oldest LSN among the pages that this
* WAL record applies to, that were not included in the record as full page
- * images. If fpw_lsn >= RedoRecPtr, the function does not perform the
+ * images. If fpw_lsn <= RedoRecPtr, the function does not perform the
* insertion and returns InvalidXLogRecPtr. The caller can then recalculate
* which pages need a full-page image, and retry. If fpw_lsn is invalid, the
* record is always inserted.
@@ -936,6 +936,7 @@ XLogInsertRecord(XLogRecData *rdata, XLogRecPtr fpw_lsn)
rechdr->xl_info == XLOG_SWITCH);
XLogRecPtr StartPos;
XLogRecPtr EndPos;
+ bool prevDoPageWrites = doPageWrites;
/* we assume that all of the record header is in the first chunk */
Assert(rdata->len >= SizeOfXLogRecord);
@@ -983,10 +984,14 @@ XLogInsertRecord(XLogRecData *rdata, XLogRecPtr fpw_lsn)
WALInsertLockAcquire();
/*
- * Check to see if my copy of RedoRecPtr or doPageWrites is out of date.
- * If so, may have to go back and have the caller recompute everything.
- * This can only happen just after a checkpoint, so it's better to be slow
- * in this case and fast otherwise.
+ * Check to see if my copy of RedoRecPtr is out of date. If so, may have
+ * to go back and have the caller recompute everything. This can only
+ * happen just after a checkpoint, so it's better to be slow in this case
+ * and fast otherwise.
+ *
+ * Also check to see if fullPageWrites or forcePageWrites was just turned
+ * on; if we weren't already doing full-page writes then go back and
+ * recompute.
*
* If we aren't doing full-page writes then RedoRecPtr doesn't actually
* affect the contents of the XLOG record, so we'll update our local copy
@@ -1001,7 +1006,9 @@ XLogInsertRecord(XLogRecData *rdata, XLogRecPtr fpw_lsn)
}
doPageWrites = (Insert->fullPageWrites || Insert->forcePageWrites);
- if (fpw_lsn != InvalidXLogRecPtr && fpw_lsn <= RedoRecPtr && doPageWrites)
+ if (doPageWrites &&
+ (!prevDoPageWrites ||
+ (fpw_lsn != InvalidXLogRecPtr && fpw_lsn <= RedoRecPtr)))
{
/*
* Oops, some buffer now needs to be backed up that the caller didn't