Commit 2393c7d1 authored by Alvaro Herrera's avatar Alvaro Herrera

Fix a couple of bugs in MultiXactId freezing

Both heap_freeze_tuple() and heap_tuple_needs_freeze() neglected to look
into a multixact to check the members against cutoff_xid.  This means
that a very old Xid could survive hidden within a multi, possibly
outliving its CLOG storage.  In the distant future, this would cause
clog lookup failures:
ERROR:  could not access status of transaction 3883960912
DETAIL:  Could not open file "pg_clog/0E78": No such file or directory.

This mostly was problematic when the updating transaction aborted, since
in that case the row wouldn't get pruned away earlier in vacuum and the
multixact could possibly survive for a long time.  In many cases, data
that is inaccessible for this reason way can be brought back
heuristically.

As a second bug, heap_freeze_tuple() didn't properly handle multixacts
that need to be frozen according to cutoff_multi, but whose updater xid
is still alive.  Instead of preserving the update Xid, it just set Xmax
invalid, which leads to both old and new tuple versions becoming
visible.  This is pretty rare in practice, but a real threat
nonetheless.  Existing corrupted rows, unfortunately, cannot be repaired
in an automated fashion.

Existing physical replicas might have already incorrectly frozen tuples
because of different behavior than in master, which might only become
apparent in the future once pg_multixact/ is truncated; it is
recommended that all clones be rebuilt after upgrading.

Following code analysis caused by bug report by J Smith in message
CADFUPgc5bmtv-yg9znxV-vcfkb+JPRqs7m2OesQXaM_4Z1JpdQ@mail.gmail.com
and privately by F-Secure.

Backpatch to 9.3, where freezing of MultiXactIds was introduced.

Analysis and patch by Andres Freund, with some tweaks by Álvaro.
parent 1ce150b7
...@@ -5264,6 +5264,8 @@ heap_inplace_update(Relation relation, HeapTuple tuple) ...@@ -5264,6 +5264,8 @@ heap_inplace_update(Relation relation, HeapTuple tuple)
* so we need no external state checks to decide what to do. (This is good * so we need no external state checks to decide what to do. (This is good
* because this function is applied during WAL recovery, when we don't have * because this function is applied during WAL recovery, when we don't have
* access to any such state, and can't depend on the hint bits to be set.) * access to any such state, and can't depend on the hint bits to be set.)
* There is an exception we make which is to assume GetMultiXactIdMembers can
* be called during recovery.
* *
* Similarly, cutoff_multi must be less than or equal to the smallest * Similarly, cutoff_multi must be less than or equal to the smallest
* MultiXactId used by any transaction currently open. * MultiXactId used by any transaction currently open.
...@@ -5279,14 +5281,19 @@ heap_inplace_update(Relation relation, HeapTuple tuple) ...@@ -5279,14 +5281,19 @@ heap_inplace_update(Relation relation, HeapTuple tuple)
* exclusive lock ensures no other backend is in process of checking the * exclusive lock ensures no other backend is in process of checking the
* tuple status. Also, getting exclusive lock makes it safe to adjust the * tuple status. Also, getting exclusive lock makes it safe to adjust the
* infomask bits. * infomask bits.
*
* NB: Cannot rely on hint bits here, they might not be set after a crash or
* on a standby.
*/ */
bool bool
heap_freeze_tuple(HeapTupleHeader tuple, TransactionId cutoff_xid, heap_freeze_tuple(HeapTupleHeader tuple, TransactionId cutoff_xid,
MultiXactId cutoff_multi) MultiXactId cutoff_multi)
{ {
bool changed = false; bool changed = false;
bool freeze_xmax = false;
TransactionId xid; TransactionId xid;
/* Process xmin */
xid = HeapTupleHeaderGetXmin(tuple); xid = HeapTupleHeaderGetXmin(tuple);
if (TransactionIdIsNormal(xid) && if (TransactionIdIsNormal(xid) &&
TransactionIdPrecedes(xid, cutoff_xid)) TransactionIdPrecedes(xid, cutoff_xid))
...@@ -5303,16 +5310,112 @@ heap_freeze_tuple(HeapTupleHeader tuple, TransactionId cutoff_xid, ...@@ -5303,16 +5310,112 @@ heap_freeze_tuple(HeapTupleHeader tuple, TransactionId cutoff_xid,
} }
/* /*
* Note that this code handles IS_MULTI Xmax values, too, but only to mark * Process xmax. To thoroughly examine the current Xmax value we need to
* the tuple as not updated if the multixact is below the cutoff Multixact * resolve a MultiXactId to its member Xids, in case some of them are
* given; it doesn't remove dead members of a very old multixact. * below the given cutoff for Xids. In that case, those values might need
* freezing, too. Also, if a multi needs freezing, we cannot simply take
* it out --- if there's a live updater Xid, it needs to be kept.
*
* Make sure to keep heap_tuple_needs_freeze in sync with this.
*/ */
xid = HeapTupleHeaderGetRawXmax(tuple); xid = HeapTupleHeaderGetRawXmax(tuple);
if ((tuple->t_infomask & HEAP_XMAX_IS_MULTI) ?
(MultiXactIdIsValid(xid) && if (tuple->t_infomask & HEAP_XMAX_IS_MULTI)
MultiXactIdPrecedes(xid, cutoff_multi)) : {
(TransactionIdIsNormal(xid) && if (!MultiXactIdIsValid(xid))
TransactionIdPrecedes(xid, cutoff_xid))) {
/* no xmax set, ignore */
;
}
else if (MultiXactIdPrecedes(xid, cutoff_multi))
{
/*
* This old multi cannot possibly be running. If it was a locker
* only, it can be removed without much further thought; but if it
* contained an update, we need to preserve it.
*/
if (HEAP_XMAX_IS_LOCKED_ONLY(tuple->t_infomask))
freeze_xmax = true;
else
{
TransactionId update_xid;
update_xid = HeapTupleGetUpdateXid(tuple);
/*
* The multixact has an update hidden within. Get rid of it.
*
* If the update_xid is below the cutoff_xid, it necessarily
* must be an aborted transaction. In a primary server, such
* an Xmax would have gotten marked invalid by
* HeapTupleSatisfiesVacuum, but in a replica that is not
* called before we are, so deal with it in the same way.
*
* If not below the cutoff_xid, then the tuple would have been
* pruned by vacuum, if the update committed long enough ago,
* and we wouldn't be freezing it; so it's either recently
* committed, or in-progress. Deal with this by setting the
* Xmax to the update Xid directly and remove the IS_MULTI
* bit. (We know there cannot be running lockers in this
* multi, because it's below the cutoff_multi value.)
*/
if (TransactionIdPrecedes(update_xid, cutoff_xid))
{
Assert(InRecovery || TransactionIdDidAbort(update_xid));
freeze_xmax = true;
}
else
{
Assert(InRecovery || !TransactionIdIsInProgress(update_xid));
tuple->t_infomask &= ~HEAP_XMAX_BITS;
HeapTupleHeaderSetXmax(tuple, update_xid);
changed = true;
}
}
}
else if (HEAP_XMAX_IS_LOCKED_ONLY(tuple->t_infomask))
{
/* newer than the cutoff, so don't touch it */
;
}
else
{
TransactionId update_xid;
/*
* This is a multixact which is not marked LOCK_ONLY, but which
* is newer than the cutoff_multi. If the update_xid is below the
* cutoff_xid point, then we can just freeze the Xmax in the
* tuple, removing it altogether. This seems simple, but there
* are several underlying assumptions:
*
* 1. A tuple marked by an multixact containing a very old
* committed update Xid would have been pruned away by vacuum; we
* wouldn't be freezing this tuple at all.
*
* 2. There cannot possibly be any live locking members remaining
* in the multixact. This is because if they were alive, the
* update's Xid would had been considered, via the lockers'
* snapshot's Xmin, as part the cutoff_xid.
*
* 3. We don't create new MultiXacts via MultiXactIdExpand() that
* include a very old aborted update Xid: in that function we only
* include update Xids corresponding to transactions that are
* committed or in-progress.
*/
update_xid = HeapTupleGetUpdateXid(tuple);
if (TransactionIdPrecedes(update_xid, cutoff_xid))
freeze_xmax = true;
}
}
else if (TransactionIdIsNormal(xid) &&
TransactionIdPrecedes(xid, cutoff_xid))
{
freeze_xmax = true;
}
if (freeze_xmax)
{ {
HeapTupleHeaderSetXmax(tuple, InvalidTransactionId); HeapTupleHeaderSetXmax(tuple, InvalidTransactionId);
...@@ -5632,6 +5735,9 @@ ConditionalMultiXactIdWait(MultiXactId multi, MultiXactStatus status, ...@@ -5632,6 +5735,9 @@ ConditionalMultiXactIdWait(MultiXactId multi, MultiXactStatus status,
* *
* It doesn't matter whether the tuple is alive or dead, we are checking * It doesn't matter whether the tuple is alive or dead, we are checking
* to see if a tuple needs to be removed or frozen to avoid wraparound. * to see if a tuple needs to be removed or frozen to avoid wraparound.
*
* NB: Cannot rely on hint bits here, they might not be set after a crash or
* on a standby.
*/ */
bool bool
heap_tuple_needs_freeze(HeapTupleHeader tuple, TransactionId cutoff_xid, heap_tuple_needs_freeze(HeapTupleHeader tuple, TransactionId cutoff_xid,
...@@ -5644,24 +5750,42 @@ heap_tuple_needs_freeze(HeapTupleHeader tuple, TransactionId cutoff_xid, ...@@ -5644,24 +5750,42 @@ heap_tuple_needs_freeze(HeapTupleHeader tuple, TransactionId cutoff_xid,
TransactionIdPrecedes(xid, cutoff_xid)) TransactionIdPrecedes(xid, cutoff_xid))
return true; return true;
if (!(tuple->t_infomask & HEAP_XMAX_INVALID)) /*
* The considerations for multixacts are complicated; look at
* heap_freeze_tuple for justifications. This routine had better be in
* sync with that one!
*/
if (tuple->t_infomask & HEAP_XMAX_IS_MULTI)
{ {
if (!(tuple->t_infomask & HEAP_XMAX_IS_MULTI)) MultiXactId multi;
multi = HeapTupleHeaderGetRawXmax(tuple);
if (!MultiXactIdIsValid(multi))
{ {
xid = HeapTupleHeaderGetRawXmax(tuple); /* no xmax set, ignore */
if (TransactionIdIsNormal(xid) && ;
TransactionIdPrecedes(xid, cutoff_xid)) }
else if (MultiXactIdPrecedes(multi, cutoff_multi))
return true; return true;
else if (HEAP_XMAX_IS_LOCKED_ONLY(tuple->t_infomask))
{
/* only-locker multis don't need internal examination */
;
} }
else else
{ {
MultiXactId multi; if (TransactionIdPrecedes(HeapTupleGetUpdateXid(tuple),
cutoff_xid))
multi = HeapTupleHeaderGetRawXmax(tuple);
if (MultiXactIdPrecedes(multi, cutoff_multi))
return true; return true;
} }
} }
else
{
xid = HeapTupleHeaderGetRawXmax(tuple);
if (TransactionIdIsNormal(xid) &&
TransactionIdPrecedes(xid, cutoff_xid))
return true;
}
if (tuple->t_infomask & HEAP_MOVED) if (tuple->t_infomask & HEAP_MOVED)
{ {
......
...@@ -434,11 +434,14 @@ MultiXactIdExpand(MultiXactId multi, TransactionId xid, MultiXactStatus status) ...@@ -434,11 +434,14 @@ MultiXactIdExpand(MultiXactId multi, TransactionId xid, MultiXactStatus status)
* Determine which of the members of the MultiXactId are still of * Determine which of the members of the MultiXactId are still of
* interest. This is any running transaction, and also any transaction * interest. This is any running transaction, and also any transaction
* that grabbed something stronger than just a lock and was committed. (An * that grabbed something stronger than just a lock and was committed. (An
* update that aborted is of no interest here.) * update that aborted is of no interest here; and having more than one
* update Xid in a multixact would cause errors elsewhere.)
* *
* (Removing dead members is just an optimization, but a useful one. Note * Removing dead members is not just an optimization: freezing of tuples
* we have the same race condition here as above: j could be 0 at the end * whose Xmax are multis depends on this behavior.
* of the loop.) *
* Note we have the same race condition here as above: j could be 0 at the
* end of the loop.
*/ */
newMembers = (MultiXactMember *) newMembers = (MultiXactMember *)
palloc(sizeof(MultiXactMember) * (nmembers + 1)); palloc(sizeof(MultiXactMember) * (nmembers + 1));
...@@ -1052,7 +1055,8 @@ GetMultiXactIdMembers(MultiXactId multi, MultiXactMember **members, ...@@ -1052,7 +1055,8 @@ GetMultiXactIdMembers(MultiXactId multi, MultiXactMember **members,
debug_elog3(DEBUG2, "GetMembers: asked for %u", multi); debug_elog3(DEBUG2, "GetMembers: asked for %u", multi);
Assert(MultiXactIdIsValid(multi)); if (!MultiXactIdIsValid(multi))
return -1;
/* See if the MultiXactId is in the local cache */ /* See if the MultiXactId is in the local cache */
length = mXactCacheGetById(multi, members); length = mXactCacheGetById(multi, members);
......
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