Commit bab7823a authored by Fujii Masao's avatar Fujii Masao

Fix pg_xlogdump so that it handles cross-page XLP_FIRST_IS_CONTRECORD record.

Previously pg_xlogdump failed to dump the contents of the WAL file
if the file starts with the continuation WAL record which spans
more than one pages. Since pg_xlogdump assumed that the continuation
record always fits on a page, it could not find the valid WAL record to
start reading from in that case.

This patch changes pg_xlogdump so that it can handle a continuation
WAL record which crosses a page boundary and find the valid record
to start reading from.

Back-patch to 9.3 where pg_xlogdump was introduced.

Author: Pavan Deolasee
Reviewed-By: Michael Paquier and Craig Ringer
Discussion: CABOikdPsPByMiG6J01DKq6om2+BNkxHTPkOyqHM2a4oYwGKsqQ@mail.gmail.com
parent b899ccbb
...@@ -866,21 +866,38 @@ XLogRecPtr ...@@ -866,21 +866,38 @@ XLogRecPtr
XLogFindNextRecord(XLogReaderState *state, XLogRecPtr RecPtr) XLogFindNextRecord(XLogReaderState *state, XLogRecPtr RecPtr)
{ {
XLogReaderState saved_state = *state; XLogReaderState saved_state = *state;
XLogRecPtr targetPagePtr;
XLogRecPtr tmpRecPtr; XLogRecPtr tmpRecPtr;
int targetRecOff;
XLogRecPtr found = InvalidXLogRecPtr; XLogRecPtr found = InvalidXLogRecPtr;
uint32 pageHeaderSize;
XLogPageHeader header; XLogPageHeader header;
int readLen;
char *errormsg; char *errormsg;
Assert(!XLogRecPtrIsInvalid(RecPtr)); Assert(!XLogRecPtrIsInvalid(RecPtr));
targetRecOff = RecPtr % XLOG_BLCKSZ; /*
* skip over potential continuation data, keeping in mind that it may span
* multiple pages
*/
tmpRecPtr = RecPtr;
while (true)
{
XLogRecPtr targetPagePtr;
int targetRecOff;
uint32 pageHeaderSize;
int readLen;
/*
* Compute targetRecOff. It should typically be equal or greater than
* short page-header since a valid record can't start anywhere before
* that, except when caller has explicitly specified the offset that
* falls somewhere there or when we are skipping multi-page
* continuation record. It doesn't matter though because
* ReadPageInternal() is prepared to handle that and will read at least
* short page-header worth of data
*/
targetRecOff = tmpRecPtr % XLOG_BLCKSZ;
/* scroll back to page boundary */ /* scroll back to page boundary */
targetPagePtr = RecPtr - targetRecOff; targetPagePtr = tmpRecPtr - targetRecOff;
/* Read the page containing the record */ /* Read the page containing the record */
readLen = ReadPageInternal(state, targetPagePtr, targetRecOff); readLen = ReadPageInternal(state, targetPagePtr, targetRecOff);
...@@ -899,13 +916,33 @@ XLogFindNextRecord(XLogReaderState *state, XLogRecPtr RecPtr) ...@@ -899,13 +916,33 @@ XLogFindNextRecord(XLogReaderState *state, XLogRecPtr RecPtr)
/* skip over potential continuation data */ /* skip over potential continuation data */
if (header->xlp_info & XLP_FIRST_IS_CONTRECORD) if (header->xlp_info & XLP_FIRST_IS_CONTRECORD)
{ {
/* record headers are MAXALIGN'ed */ /*
* If the length of the remaining continuation data is more than
* what can fit in this page, the continuation record crosses over
* this page. Read the next page and try again. xlp_rem_len in the
* next page header will contain the remaining length of the
* continuation data
*
* Note that record headers are MAXALIGN'ed
*/
if (MAXALIGN(header->xlp_rem_len) > (XLOG_BLCKSZ - pageHeaderSize))
tmpRecPtr = targetPagePtr + XLOG_BLCKSZ;
else
{
/*
* The previous continuation record ends in this page. Set
* tmpRecPtr to point to the first valid record
*/
tmpRecPtr = targetPagePtr + pageHeaderSize tmpRecPtr = targetPagePtr + pageHeaderSize
+ MAXALIGN(header->xlp_rem_len); + MAXALIGN(header->xlp_rem_len);
break;
}
} }
else else
{ {
tmpRecPtr = targetPagePtr + pageHeaderSize; tmpRecPtr = targetPagePtr + pageHeaderSize;
break;
}
} }
/* /*
......
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