Commit 32f4de0a authored by Heikki Linnakangas's avatar Heikki Linnakangas

Write exact xlog position of timeline switch in the timeline history file.

This allows us to do some more rigorous sanity checking for various
incorrect point-in-time recovery scenarios, and provides more information
for debugging purposes. It will also come handy in the upcoming patch to
allow timeline switches to be replicated by streaming replication.
parent a84c30dd
...@@ -12,10 +12,10 @@ ...@@ -12,10 +12,10 @@
* *
* Each line in the file represents a timeline switch: * Each line in the file represents a timeline switch:
* *
* <parentTLI> <xlogfname> <reason> * <parentTLI> <switchpoint> <reason>
* *
* parentTLI ID of the parent timeline * parentTLI ID of the parent timeline
* xlogfname filename of the WAL segment where the switch happened * switchpoint XLogRecPtr of the WAL position where the switch happened
* reason human-readable explanation of why the timeline was changed * reason human-readable explanation of why the timeline was changed
* *
* The fields are separated by tabs. Lines beginning with # are comments, and * The fields are separated by tabs. Lines beginning with # are comments, and
...@@ -56,10 +56,18 @@ readTimeLineHistory(TimeLineID targetTLI) ...@@ -56,10 +56,18 @@ readTimeLineHistory(TimeLineID targetTLI)
char histfname[MAXFNAMELEN]; char histfname[MAXFNAMELEN];
char fline[MAXPGPATH]; char fline[MAXPGPATH];
FILE *fd; FILE *fd;
TimeLineHistoryEntry *entry;
TimeLineID lasttli = 0;
XLogRecPtr prevend;
/* Timeline 1 does not have a history file, so no need to check */ /* Timeline 1 does not have a history file, so no need to check */
if (targetTLI == 1) if (targetTLI == 1)
return list_make1_int((int) targetTLI); {
entry = (TimeLineHistoryEntry *) palloc(sizeof(TimeLineHistoryEntry));
entry->tli = targetTLI;
entry->begin = entry->end = InvalidXLogRecPtr;
return list_make1(entry);
}
if (InArchiveRecovery) if (InArchiveRecovery)
{ {
...@@ -77,7 +85,10 @@ readTimeLineHistory(TimeLineID targetTLI) ...@@ -77,7 +85,10 @@ readTimeLineHistory(TimeLineID targetTLI)
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not open file \"%s\": %m", path))); errmsg("could not open file \"%s\": %m", path)));
/* Not there, so assume no parents */ /* Not there, so assume no parents */
return list_make1_int((int) targetTLI); entry = (TimeLineHistoryEntry *) palloc(sizeof(TimeLineHistoryEntry));
entry->tli = targetTLI;
entry->begin = entry->end = InvalidXLogRecPtr;
return list_make1(entry);
} }
result = NIL; result = NIL;
...@@ -85,12 +96,15 @@ readTimeLineHistory(TimeLineID targetTLI) ...@@ -85,12 +96,15 @@ readTimeLineHistory(TimeLineID targetTLI)
/* /*
* Parse the file... * Parse the file...
*/ */
prevend = InvalidXLogRecPtr;
while (fgets(fline, sizeof(fline), fd) != NULL) while (fgets(fline, sizeof(fline), fd) != NULL)
{ {
/* skip leading whitespace and check for # comment */ /* skip leading whitespace and check for # comment */
char *ptr; char *ptr;
char *endptr;
TimeLineID tli; TimeLineID tli;
uint32 switchpoint_hi;
uint32 switchpoint_lo;
int nfields;
for (ptr = fline; *ptr; ptr++) for (ptr = fline; *ptr; ptr++)
{ {
...@@ -100,38 +114,56 @@ readTimeLineHistory(TimeLineID targetTLI) ...@@ -100,38 +114,56 @@ readTimeLineHistory(TimeLineID targetTLI)
if (*ptr == '\0' || *ptr == '#') if (*ptr == '\0' || *ptr == '#')
continue; continue;
/* expect a numeric timeline ID as first field of line */ nfields = sscanf(fline, "%u\t%X/%X", &tli, &switchpoint_hi, &switchpoint_lo);
tli = (TimeLineID) strtoul(ptr, &endptr, 0);
if (endptr == ptr) if (nfields < 1)
{
/* expect a numeric timeline ID as first field of line */
ereport(FATAL, ereport(FATAL,
(errmsg("syntax error in history file: %s", fline), (errmsg("syntax error in history file: %s", fline),
errhint("Expected a numeric timeline ID."))); errhint("Expected a numeric timeline ID.")));
}
if (nfields != 3)
ereport(FATAL,
(errmsg("syntax error in history file: %s", fline),
errhint("Expected an XLOG switchpoint location.")));
if (result && if (result && tli <= lasttli)
tli <= (TimeLineID) linitial_int(result))
ereport(FATAL, ereport(FATAL,
(errmsg("invalid data in history file: %s", fline), (errmsg("invalid data in history file: %s", fline),
errhint("Timeline IDs must be in increasing sequence."))); errhint("Timeline IDs must be in increasing sequence.")));
lasttli = tli;
entry = (TimeLineHistoryEntry *) palloc(sizeof(TimeLineHistoryEntry));
entry->tli = tli;
entry->begin = prevend;
entry->end = ((uint64) (switchpoint_hi)) << 32 | (uint64) switchpoint_lo;
prevend = entry->end;
/* Build list with newest item first */ /* Build list with newest item first */
result = lcons_int((int) tli, result); result = lcons(entry, result);
/* we ignore the remainder of each line */ /* we ignore the remainder of each line */
} }
FreeFile(fd); FreeFile(fd);
if (result && if (result && targetTLI <= lasttli)
targetTLI <= (TimeLineID) linitial_int(result))
ereport(FATAL, ereport(FATAL,
(errmsg("invalid data in history file \"%s\"", path), (errmsg("invalid data in history file \"%s\"", path),
errhint("Timeline IDs must be less than child timeline's ID."))); errhint("Timeline IDs must be less than child timeline's ID.")));
result = lcons_int((int) targetTLI, result); /*
* Create one more entry for the "tip" of the timeline, which has no
* entry in the history file.
*/
entry = (TimeLineHistoryEntry *) palloc(sizeof(TimeLineHistoryEntry));
entry->tli = targetTLI;
entry->begin = prevend;
entry->end = InvalidXLogRecPtr;
ereport(DEBUG3, result = lcons(entry, result);
(errmsg_internal("history of timeline %u is %s",
targetTLI, nodeToString(result))));
return result; return result;
} }
...@@ -214,7 +246,7 @@ findNewestTimeLine(TimeLineID startTLI) ...@@ -214,7 +246,7 @@ findNewestTimeLine(TimeLineID startTLI)
* *
* newTLI: ID of the new timeline * newTLI: ID of the new timeline
* parentTLI: ID of its immediate parent * parentTLI: ID of its immediate parent
* endTLI et al: ID of the last used WAL file, for annotation purposes * switchpoint: XLOG position where the system switched to the new timeline
* reason: human-readable explanation of why the timeline was switched * reason: human-readable explanation of why the timeline was switched
* *
* Currently this is only used at the end recovery, and so there are no locking * Currently this is only used at the end recovery, and so there are no locking
...@@ -223,12 +255,11 @@ findNewestTimeLine(TimeLineID startTLI) ...@@ -223,12 +255,11 @@ findNewestTimeLine(TimeLineID startTLI)
*/ */
void void
writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
TimeLineID endTLI, XLogSegNo endLogSegNo, char *reason) XLogRecPtr switchpoint, char *reason)
{ {
char path[MAXPGPATH]; char path[MAXPGPATH];
char tmppath[MAXPGPATH]; char tmppath[MAXPGPATH];
char histfname[MAXFNAMELEN]; char histfname[MAXFNAMELEN];
char xlogfname[MAXFNAMELEN];
char buffer[BLCKSZ]; char buffer[BLCKSZ];
int srcfd; int srcfd;
int fd; int fd;
...@@ -313,13 +344,11 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, ...@@ -313,13 +344,11 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
* If we did have a parent file, insert an extra newline just in case the * If we did have a parent file, insert an extra newline just in case the
* parent file failed to end with one. * parent file failed to end with one.
*/ */
XLogFileName(xlogfname, endTLI, endLogSegNo);
snprintf(buffer, sizeof(buffer), snprintf(buffer, sizeof(buffer),
"%s%u\t%s\t%s\n", "%s%u\t%X/%X\t%s\n",
(srcfd < 0) ? "" : "\n", (srcfd < 0) ? "" : "\n",
parentTLI, parentTLI,
xlogfname, (uint32) (switchpoint >> 32), (uint32) (switchpoint),
reason); reason);
nbytes = strlen(buffer); nbytes = strlen(buffer);
...@@ -380,3 +409,70 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, ...@@ -380,3 +409,70 @@ writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
TLHistoryFileName(histfname, newTLI); TLHistoryFileName(histfname, newTLI);
XLogArchiveNotify(histfname); XLogArchiveNotify(histfname);
} }
/*
* Returns true if 'expectedTLEs' contains a timeline with id 'tli'
*/
bool
tliInHistory(TimeLineID tli, List *expectedTLEs)
{
ListCell *cell;
foreach(cell, expectedTLEs)
{
if (((TimeLineHistoryEntry *) lfirst(cell))->tli == tli)
return true;
}
return false;
}
/*
* Returns the ID of the timeline in use at a particular point in time, in
* the given timeline history.
*/
TimeLineID
tliOfPointInHistory(XLogRecPtr ptr, List *history)
{
ListCell *cell;
foreach(cell, history)
{
TimeLineHistoryEntry *tle = (TimeLineHistoryEntry *) lfirst(cell);
if ((XLogRecPtrIsInvalid(tle->begin) || XLByteLE(tle->begin, ptr)) &&
(XLogRecPtrIsInvalid(tle->end) || XLByteLT(ptr, tle->end)))
{
/* found it */
return tle->tli;
}
}
/* shouldn't happen. */
elog(ERROR, "timeline history was not contiguous");
return 0; /* keep compiler quiet */
}
/*
* Returns the point in history where we branched off the given timeline.
* Returns InvalidXLogRecPtr if the timeline is current (= we have not
* branched off from it), and throws an error if the timeline is not part of
* this server's history.
*/
XLogRecPtr
tliSwitchPoint(TimeLineID tli, List *history)
{
ListCell *cell;
foreach (cell, history)
{
TimeLineHistoryEntry *tle = (TimeLineHistoryEntry *) lfirst(cell);
if (tle->tli == tli)
return tle->end;
}
ereport(ERROR,
(errmsg("requested timeline %u is not in this server's history",
tli)));
return InvalidXLogRecPtr; /* keep compiler quiet */
}
...@@ -226,7 +226,7 @@ static bool recoveryStopAfter; ...@@ -226,7 +226,7 @@ static bool recoveryStopAfter;
* *
* recoveryTargetIsLatest: was the requested target timeline 'latest'? * recoveryTargetIsLatest: was the requested target timeline 'latest'?
* *
* expectedTLIs: an integer list of recoveryTargetTLI and the TLIs of * expectedTLEs: a list of TimeLineHistoryEntries for recoveryTargetTLI and the timelines of
* its known parents, newest first (so recoveryTargetTLI is always the * its known parents, newest first (so recoveryTargetTLI is always the
* first list member). Only these TLIs are expected to be seen in the WAL * first list member). Only these TLIs are expected to be seen in the WAL
* segments we read, and indeed only these TLIs will be considered as * segments we read, and indeed only these TLIs will be considered as
...@@ -240,7 +240,7 @@ static bool recoveryStopAfter; ...@@ -240,7 +240,7 @@ static bool recoveryStopAfter;
*/ */
static TimeLineID recoveryTargetTLI; static TimeLineID recoveryTargetTLI;
static bool recoveryTargetIsLatest = false; static bool recoveryTargetIsLatest = false;
static List *expectedTLIs; static List *expectedTLEs;
static TimeLineID curFileTLI; static TimeLineID curFileTLI;
/* /*
...@@ -2515,7 +2515,7 @@ InstallXLogFileSegment(XLogSegNo *segno, char *tmppath, ...@@ -2515,7 +2515,7 @@ InstallXLogFileSegment(XLogSegNo *segno, char *tmppath,
/* /*
* Prefer link() to rename() here just to be really sure that we don't * Prefer link() to rename() here just to be really sure that we don't
* overwrite an existing logfile. However, there shouldn't be one, so * overwrite an existing file. However, there shouldn't be one, so
* rename() is an acceptable substitute except for the truly paranoid. * rename() is an acceptable substitute except for the truly paranoid.
*/ */
#if HAVE_WORKING_LINK #if HAVE_WORKING_LINK
...@@ -2716,7 +2716,7 @@ XLogFileRead(XLogSegNo segno, int emode, TimeLineID tli, ...@@ -2716,7 +2716,7 @@ XLogFileRead(XLogSegNo segno, int emode, TimeLineID tli,
/* /*
* Open a logfile segment for reading (during recovery). * Open a logfile segment for reading (during recovery).
* *
* This version searches for the segment with any TLI listed in expectedTLIs. * This version searches for the segment with any TLI listed in expectedTLEs.
*/ */
static int static int
XLogFileReadAnyTLI(XLogSegNo segno, int emode, int source) XLogFileReadAnyTLI(XLogSegNo segno, int emode, int source)
...@@ -2727,7 +2727,7 @@ XLogFileReadAnyTLI(XLogSegNo segno, int emode, int source) ...@@ -2727,7 +2727,7 @@ XLogFileReadAnyTLI(XLogSegNo segno, int emode, int source)
/* /*
* Loop looking for a suitable timeline ID: we might need to read any of * Loop looking for a suitable timeline ID: we might need to read any of
* the timelines listed in expectedTLIs. * the timelines listed in expectedTLEs.
* *
* We expect curFileTLI on entry to be the TLI of the preceding file in * We expect curFileTLI on entry to be the TLI of the preceding file in
* sequence, or 0 if there was no predecessor. We do not allow curFileTLI * sequence, or 0 if there was no predecessor. We do not allow curFileTLI
...@@ -2735,9 +2735,9 @@ XLogFileReadAnyTLI(XLogSegNo segno, int emode, int source) ...@@ -2735,9 +2735,9 @@ XLogFileReadAnyTLI(XLogSegNo segno, int emode, int source)
* parent timeline extends to higher segment numbers than the child we * parent timeline extends to higher segment numbers than the child we
* want to read. * want to read.
*/ */
foreach(cell, expectedTLIs) foreach(cell, expectedTLEs)
{ {
TimeLineID tli = (TimeLineID) lfirst_int(cell); TimeLineID tli = ((TimeLineHistoryEntry *) lfirst(cell))->tli;
if (tli < curFileTLI) if (tli < curFileTLI)
break; /* don't bother looking at too-old TLIs */ break; /* don't bother looking at too-old TLIs */
...@@ -3344,7 +3344,7 @@ ReadRecord(XLogRecPtr *RecPtr, int emode, bool fetching_ckpt) ...@@ -3344,7 +3344,7 @@ ReadRecord(XLogRecPtr *RecPtr, int emode, bool fetching_ckpt)
/* /*
* Since we are going to a random position in WAL, forget any prior * Since we are going to a random position in WAL, forget any prior
* state about what timeline we were in, and allow it to be any * state about what timeline we were in, and allow it to be any
* timeline in expectedTLIs. We also set a flag to allow curFileTLI * timeline in expectedTLEs. We also set a flag to allow curFileTLI
* to go backwards (but we can't reset that variable right here, since * to go backwards (but we can't reset that variable right here, since
* we might not change files at all). * we might not change files at all).
*/ */
...@@ -3675,7 +3675,7 @@ ValidXLogPageHeader(XLogPageHeader hdr, int emode, bool segmentonly) ...@@ -3675,7 +3675,7 @@ ValidXLogPageHeader(XLogPageHeader hdr, int emode, bool segmentonly)
/* /*
* Check page TLI is one of the expected values. * Check page TLI is one of the expected values.
*/ */
if (!list_member_int(expectedTLIs, (int) hdr->xlp_tli)) if (!tliInHistory(hdr->xlp_tli, expectedTLEs))
{ {
ereport(emode_for_corrupt_record(emode, recaddr), ereport(emode_for_corrupt_record(emode, recaddr),
(errmsg("unexpected timeline ID %u in log segment %s, offset %u", (errmsg("unexpected timeline ID %u in log segment %s, offset %u",
...@@ -3812,57 +3812,86 @@ ValidXLogRecordHeader(XLogRecPtr *RecPtr, XLogRecord *record, int emode, ...@@ -3812,57 +3812,86 @@ ValidXLogRecordHeader(XLogRecPtr *RecPtr, XLogRecord *record, int emode,
static bool static bool
rescanLatestTimeLine(void) rescanLatestTimeLine(void)
{ {
List *newExpectedTLEs;
bool found;
ListCell *cell;
TimeLineID newtarget; TimeLineID newtarget;
TimeLineHistoryEntry *currentTle = NULL;
/* use volatile pointer to prevent code rearrangement */
volatile XLogCtlData *xlogctl = XLogCtl;
newtarget = findNewestTimeLine(recoveryTargetTLI); newtarget = findNewestTimeLine(recoveryTargetTLI);
if (newtarget != recoveryTargetTLI) if (newtarget == recoveryTargetTLI)
{ {
/* /* No new timelines found */
* Determine the list of expected TLIs for the new TLI return false;
*/ }
List *newExpectedTLIs;
newExpectedTLIs = readTimeLineHistory(newtarget);
/* /*
* If the current timeline is not part of the history of the new * Determine the list of expected TLIs for the new TLI
* timeline, we cannot proceed to it. */
*
* XXX This isn't foolproof: The new timeline might have forked from
* the current one, but before the current recovery location. In that
* case we will still switch to the new timeline and proceed replaying
* from it even though the history doesn't match what we already
* replayed. That's not good. We will likely notice at the next online
* checkpoint, as the TLI won't match what we expected, but it's not
* guaranteed. The admin needs to make sure that doesn't happen.
*/
if (!list_member_int(newExpectedTLIs,
(int) recoveryTargetTLI))
ereport(LOG,
(errmsg("new timeline %u is not a child of database system timeline %u",
newtarget,
ThisTimeLineID)));
else
{
/* use volatile pointer to prevent code rearrangement */
volatile XLogCtlData *xlogctl = XLogCtl;
/* Switch target */ newExpectedTLEs = readTimeLineHistory(newtarget);
recoveryTargetTLI = newtarget;
list_free(expectedTLIs);
expectedTLIs = newExpectedTLIs;
SpinLockAcquire(&xlogctl->info_lck); /*
xlogctl->RecoveryTargetTLI = recoveryTargetTLI; * If the current timeline is not part of the history of the new
SpinLockRelease(&xlogctl->info_lck); * timeline, we cannot proceed to it.
*/
found = false;
foreach (cell, newExpectedTLEs)
{
currentTle = (TimeLineHistoryEntry *) lfirst(cell);
ereport(LOG, if (currentTle->tli == recoveryTargetTLI)
(errmsg("new target timeline is %u", {
recoveryTargetTLI))); found = true;
return true; break;
} }
} }
return false; if (!found)
{
ereport(LOG,
(errmsg("new timeline %u is not a child of database system timeline %u",
newtarget,
ThisTimeLineID)));
return false;
}
/*
* The current timeline was found in the history file, but check that the
* next timeline was forked off from it *after* the current recovery
* location.
*/
if (XLByteLT(currentTle->end, EndRecPtr))
{
ereport(LOG,
(errmsg("new timeline %u forked off current database system timeline %u before current recovery point %X/%X",
newtarget,
ThisTimeLineID,
(uint32) (EndRecPtr >> 32), (uint32) EndRecPtr)));
return false;
}
/* The new timeline history seems valid. Switch target */
recoveryTargetTLI = newtarget;
list_free_deep(expectedTLEs);
expectedTLEs = newExpectedTLEs;
SpinLockAcquire(&xlogctl->info_lck);
xlogctl->RecoveryTargetTLI = recoveryTargetTLI;
SpinLockRelease(&xlogctl->info_lck);
ereport(LOG,
(errmsg("new target timeline is %u",
recoveryTargetTLI)));
/*
* Wake up any walsenders to notice that we have a new target timeline.
*/
if (AllowCascadeReplication())
WalSndWakeup();
return true;
} }
/* /*
...@@ -5300,26 +5329,41 @@ StartupXLOG(void) ...@@ -5300,26 +5329,41 @@ StartupXLOG(void)
readRecoveryCommandFile(); readRecoveryCommandFile();
/* Now we can determine the list of expected TLIs */ /* Now we can determine the list of expected TLIs */
expectedTLIs = readTimeLineHistory(recoveryTargetTLI); expectedTLEs = readTimeLineHistory(recoveryTargetTLI);
/* /*
* If pg_control's timeline is not in expectedTLIs, then we cannot * If the location of the checkpoint record is not on the expected
* proceed: the backup is not part of the history of the requested * timeline in the history of the requested timeline, we cannot proceed:
* timeline. * the backup is not part of the history of the requested timeline.
*/ */
if (!list_member_int(expectedTLIs, if (tliOfPointInHistory(ControlFile->checkPoint, expectedTLEs) !=
(int) ControlFile->checkPointCopy.ThisTimeLineID)) ControlFile->checkPointCopy.ThisTimeLineID)
{
XLogRecPtr switchpoint;
/*
* tliSwitchPoint will throw an error if the checkpoint's timeline
* is not in expectedTLEs at all.
*/
switchpoint = tliSwitchPoint(ControlFile->checkPointCopy.ThisTimeLineID, expectedTLEs);
ereport(FATAL, ereport(FATAL,
(errmsg("requested timeline %u is not a child of database system timeline %u", (errmsg("requested timeline %u is not a child of this server's history",
recoveryTargetTLI, recoveryTargetTLI),
ControlFile->checkPointCopy.ThisTimeLineID))); errdetail("Latest checkpoint is at %X/%X on timeline %u, but in the history of the requested timeline, the server forked off from that timeline at %X/%X",
(uint32) (ControlFile->checkPoint >> 32),
(uint32) ControlFile->checkPoint,
ControlFile->checkPointCopy.ThisTimeLineID,
(uint32) (switchpoint >> 32),
(uint32) switchpoint)));
}
/* /*
* The min recovery point should be part of the requested timeline's * The min recovery point should be part of the requested timeline's
* history, too. * history, too.
*/ */
if (!XLogRecPtrIsInvalid(ControlFile->minRecoveryPoint) && if (!XLogRecPtrIsInvalid(ControlFile->minRecoveryPoint) &&
!list_member_int(expectedTLIs, ControlFile->minRecoveryPointTLI)) tliOfPointInHistory(ControlFile->minRecoveryPoint - 1, expectedTLEs) !=
ControlFile->minRecoveryPointTLI)
ereport(FATAL, ereport(FATAL,
(errmsg("requested timeline %u does not contain minimum recovery point %X/%X on timeline %u", (errmsg("requested timeline %u does not contain minimum recovery point %X/%X on timeline %u",
recoveryTargetTLI, recoveryTargetTLI,
...@@ -6026,8 +6070,8 @@ StartupXLOG(void) ...@@ -6026,8 +6070,8 @@ StartupXLOG(void)
(errmsg("selected new timeline ID: %u", ThisTimeLineID))); (errmsg("selected new timeline ID: %u", ThisTimeLineID)));
/* /*
* Write comment to history file to explain why and where timeline * Create a comment for the history file to explain why and where
* changed. Comment varies according to the recovery target used. * timeline changed.
*/ */
if (recoveryTarget == RECOVERY_TARGET_XID) if (recoveryTarget == RECOVERY_TARGET_XID)
snprintf(reason, sizeof(reason), snprintf(reason, sizeof(reason),
...@@ -6047,7 +6091,7 @@ StartupXLOG(void) ...@@ -6047,7 +6091,7 @@ StartupXLOG(void)
snprintf(reason, sizeof(reason), "no recovery target specified"); snprintf(reason, sizeof(reason), "no recovery target specified");
writeTimeLineHistory(ThisTimeLineID, recoveryTargetTLI, writeTimeLineHistory(ThisTimeLineID, recoveryTargetTLI,
curFileTLI, endLogSegNo, reason); EndRecPtr, reason);
} }
/* Save the selected TimeLineID in shared memory, too */ /* Save the selected TimeLineID in shared memory, too */
...@@ -7916,8 +7960,7 @@ xlog_redo(XLogRecPtr lsn, XLogRecord *record) ...@@ -7916,8 +7960,7 @@ xlog_redo(XLogRecPtr lsn, XLogRecord *record)
* decrease. * decrease.
*/ */
if (checkPoint.ThisTimeLineID < ThisTimeLineID || if (checkPoint.ThisTimeLineID < ThisTimeLineID ||
!list_member_int(expectedTLIs, !tliInHistory(checkPoint.ThisTimeLineID, expectedTLEs))
(int) checkPoint.ThisTimeLineID))
ereport(PANIC, ereport(PANIC,
(errmsg("unexpected timeline ID %u (after %u) in checkpoint record", (errmsg("unexpected timeline ID %u (after %u) in checkpoint record",
checkPoint.ThisTimeLineID, ThisTimeLineID))); checkPoint.ThisTimeLineID, ThisTimeLineID)));
......
...@@ -14,10 +14,28 @@ ...@@ -14,10 +14,28 @@
#include "access/xlogdefs.h" #include "access/xlogdefs.h"
#include "nodes/pg_list.h" #include "nodes/pg_list.h"
/*
* A list of these structs describes the timeline history of the server. Each
* TimeLineHistoryEntry represents a piece of WAL belonging to the history,
* from newest to oldest. All WAL positions between 'begin' and 'end' belong to
* the timeline represented by the entry. Together the 'begin' and 'end'
* pointers of all the entries form a contiguous line from beginning of time
* to infinity.
*/
typedef struct
{
TimeLineID tli;
XLogRecPtr begin; /* inclusive */
XLogRecPtr end; /* exclusive, 0 means infinity */
} TimeLineHistoryEntry;
extern List *readTimeLineHistory(TimeLineID targetTLI); extern List *readTimeLineHistory(TimeLineID targetTLI);
extern bool existsTimeLineHistory(TimeLineID probeTLI); extern bool existsTimeLineHistory(TimeLineID probeTLI);
extern TimeLineID findNewestTimeLine(TimeLineID startTLI); extern TimeLineID findNewestTimeLine(TimeLineID startTLI);
extern void writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI, extern void writeTimeLineHistory(TimeLineID newTLI, TimeLineID parentTLI,
TimeLineID endTLI, XLogSegNo endLogSegNo, char *reason); XLogRecPtr switchpoint, char *reason);
extern bool tliInHistory(TimeLineID tli, List *expectedTLIs);
extern TimeLineID tliOfPointInHistory(XLogRecPtr ptr, List *history);
extern XLogRecPtr tliSwitchPoint(TimeLineID tli, List *history);
#endif /* TIMELINE_H */ #endif /* TIMELINE_H */
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