Commit f786e91a authored by Tom Lane's avatar Tom Lane

Improve underdocumented btree_xlog_delete_get_latestRemovedXid() code.

As noted by Noah Misch, btree_xlog_delete_get_latestRemovedXid is
critically dependent on the assumption that it's examining a consistent
state of the database.  This was undocumented though, so the
seemingly-unrelated check for no active HS sessions might be thought to be
merely an optional optimization.  Improve comments, and add an explicit
check of reachedConsistency just to be sure.

This function returns InvalidTransactionId (thereby killing all HS
transactions) in several cases that are not nearly unlikely enough for my
taste.  This commit doesn't attempt to fix those deficiencies, just
document them.

Back-patch to 9.2, not from any real functional need but just to keep the
branches more closely synced to simplify possible future back-patching.
parent c1793f2e
...@@ -581,15 +581,33 @@ btree_xlog_delete_get_latestRemovedXid(XLogRecord *record) ...@@ -581,15 +581,33 @@ btree_xlog_delete_get_latestRemovedXid(XLogRecord *record)
/* /*
* If there's nothing running on the standby we don't need to derive a * If there's nothing running on the standby we don't need to derive a
* full latestRemovedXid value, so use a fast path out of here. That * full latestRemovedXid value, so use a fast path out of here. This
* returns InvalidTransactionId, and so will conflict with users, but * returns InvalidTransactionId, and so will conflict with all HS
* since we just worked out that's zero people, its OK. * transactions; but since we just worked out that that's zero people,
* it's OK.
*
* XXX There is a race condition here, which is that a new backend might
* start just after we look. If so, it cannot need to conflict, but this
* coding will result in throwing a conflict anyway.
*/ */
if (CountDBBackends(InvalidOid) == 0) if (CountDBBackends(InvalidOid) == 0)
return latestRemovedXid; return latestRemovedXid;
/* /*
* Get index page * In what follows, we have to examine the previous state of the index
* page, as well as the heap page(s) it points to. This is only valid if
* WAL replay has reached a consistent database state; which means that
* the preceding check is not just an optimization, but is *necessary*.
* We won't have let in any user sessions before we reach consistency.
*/
if (!reachedConsistency)
elog(PANIC, "btree_xlog_delete_get_latestRemovedXid: cannot operate with inconsistent data");
/*
* Get index page. If the DB is consistent, this should not fail, nor
* should any of the heap page fetches below. If one does, we return
* InvalidTransactionId to cancel all HS transactions. That's probably
* overkill, but it's safe, and certainly better than panicking here.
*/ */
ibuffer = XLogReadBuffer(xlrec->node, xlrec->block, false); ibuffer = XLogReadBuffer(xlrec->node, xlrec->block, false);
if (!BufferIsValid(ibuffer)) if (!BufferIsValid(ibuffer))
...@@ -671,12 +689,11 @@ btree_xlog_delete_get_latestRemovedXid(XLogRecord *record) ...@@ -671,12 +689,11 @@ btree_xlog_delete_get_latestRemovedXid(XLogRecord *record)
UnlockReleaseBuffer(ibuffer); UnlockReleaseBuffer(ibuffer);
/* /*
* Note that if all heap tuples were LP_DEAD then we will be returning * XXX If all heap tuples were LP_DEAD then we will be returning
* InvalidTransactionId here. That can happen if we are re-replaying this * InvalidTransactionId here, causing conflict for all HS
* record type, though that will be before the consistency point and will * transactions. That should happen very rarely (reasoning please?). Also
* not cause problems. It should happen very rarely after the consistency * note that caller can't tell the difference between this case and the
* point, though note that we can't tell the difference between this and * fast path exit above. May need to change that in future.
* the fast path exit above. May need to change that in future.
*/ */
return latestRemovedXid; return latestRemovedXid;
} }
...@@ -940,6 +957,10 @@ btree_redo(XLogRecPtr lsn, XLogRecord *record) ...@@ -940,6 +957,10 @@ btree_redo(XLogRecPtr lsn, XLogRecord *record)
{ {
uint8 info = record->xl_info & ~XLR_INFO_MASK; uint8 info = record->xl_info & ~XLR_INFO_MASK;
/*
* If we have any conflict processing to do, it must happen before we
* update the page.
*/
if (InHotStandby) if (InHotStandby)
{ {
switch (info) switch (info)
......
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