Commit f946069e authored by Andres Freund's avatar Andres Freund

Use TransactionXmin instead of RecentGlobalXmin in heap_abort_speculative().

There's a very low risk that RecentGlobalXmin could be far enough in
the past to be older than relfrozenxid, or even wrapped
around. Luckily the consequences of that having happened wouldn't be
too bad - the page wouldn't be pruned for a while.

Avoid that risk by using TransactionXmin instead. As that's announced
via MyPgXact->xmin, it is protected against wrapping around (see code
comments for details around relfrozenxid).

Author: Andres Freund
Discussion: https://postgr.es/m/20200328213023.s4eyijhdosuc4vcj@alap3.anarazel.de
Backpatch: 9.5-
parent 549a3e23
...@@ -5581,6 +5581,7 @@ heap_abort_speculative(Relation relation, ItemPointer tid) ...@@ -5581,6 +5581,7 @@ heap_abort_speculative(Relation relation, ItemPointer tid)
Page page; Page page;
BlockNumber block; BlockNumber block;
Buffer buffer; Buffer buffer;
TransactionId prune_xid;
Assert(ItemPointerIsValid(tid)); Assert(ItemPointerIsValid(tid));
...@@ -5623,13 +5624,21 @@ heap_abort_speculative(Relation relation, ItemPointer tid) ...@@ -5623,13 +5624,21 @@ heap_abort_speculative(Relation relation, ItemPointer tid)
START_CRIT_SECTION(); START_CRIT_SECTION();
/* /*
* The tuple will become DEAD immediately. Flag that this page * The tuple will become DEAD immediately. Flag that this page is a
* immediately is a candidate for pruning by setting xmin to * candidate for pruning by setting xmin to TransactionXmin. While not
* RecentGlobalXmin. That's not pretty, but it doesn't seem worth * immediately prunable, it is the oldest xid we can cheaply determine
* inventing a nicer API for this. * that's safe against wraparound / being older than the table's
* relfrozenxid. To defend against the unlikely case of a new relation
* having a newer relfrozenxid than our TransactionXmin, use relfrozenxid
* if so (vacuum can't subsequently move relfrozenxid to beyond
* TransactionXmin, so there's no race here).
*/ */
Assert(TransactionIdIsValid(RecentGlobalXmin)); Assert(TransactionIdIsValid(TransactionXmin));
PageSetPrunable(page, RecentGlobalXmin); if (TransactionIdPrecedes(TransactionXmin, relation->rd_rel->relfrozenxid))
prune_xid = relation->rd_rel->relfrozenxid;
else
prune_xid = TransactionXmin;
PageSetPrunable(page, prune_xid);
/* store transaction information of xact deleting the tuple */ /* store transaction information of xact deleting the tuple */
tp.t_data->t_infomask &= ~(HEAP_XMAX_BITS | HEAP_MOVED); tp.t_data->t_infomask &= ~(HEAP_XMAX_BITS | HEAP_MOVED);
......
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