Commit 21934612 authored by Jeff Davis's avatar Jeff Davis

Fix race condition where heap_delete() fails to pin VM page.

Similar to 5f12bc94dc, the code must re-check PageIsAllVisible() after
buffer lock is re-acquired. Backpatching to the same version, 12.

Discussion: https://postgr.es/m/CAEP4nAw9jYQDKd_5Y+-s2E4YiUJq1vqiikFjYGpLShtp-K3gag@mail.gmail.com
Reported-by: Robins Tharakan
Reviewed-by: Robins Tharakan
Backpatch-through: 12
parent e4514aaf
...@@ -2773,6 +2773,15 @@ heap_delete(Relation relation, ItemPointer tid, ...@@ -2773,6 +2773,15 @@ heap_delete(Relation relation, ItemPointer tid,
LockBuffer(buffer, BUFFER_LOCK_EXCLUSIVE); LockBuffer(buffer, BUFFER_LOCK_EXCLUSIVE);
lp = PageGetItemId(page, ItemPointerGetOffsetNumber(tid));
Assert(ItemIdIsNormal(lp));
tp.t_tableOid = RelationGetRelid(relation);
tp.t_data = (HeapTupleHeader) PageGetItem(page, lp);
tp.t_len = ItemIdGetLength(lp);
tp.t_self = *tid;
l1:
/* /*
* If we didn't pin the visibility map page and the page has become all * If we didn't pin the visibility map page and the page has become all
* visible while we were busy locking the buffer, we'll have to unlock and * visible while we were busy locking the buffer, we'll have to unlock and
...@@ -2786,15 +2795,6 @@ heap_delete(Relation relation, ItemPointer tid, ...@@ -2786,15 +2795,6 @@ heap_delete(Relation relation, ItemPointer tid,
LockBuffer(buffer, BUFFER_LOCK_EXCLUSIVE); LockBuffer(buffer, BUFFER_LOCK_EXCLUSIVE);
} }
lp = PageGetItemId(page, ItemPointerGetOffsetNumber(tid));
Assert(ItemIdIsNormal(lp));
tp.t_tableOid = RelationGetRelid(relation);
tp.t_data = (HeapTupleHeader) PageGetItem(page, lp);
tp.t_len = ItemIdGetLength(lp);
tp.t_self = *tid;
l1:
result = HeapTupleSatisfiesUpdate(&tp, cid, buffer); result = HeapTupleSatisfiesUpdate(&tp, cid, buffer);
if (result == TM_Invisible) if (result == TM_Invisible)
...@@ -2853,8 +2853,12 @@ l1: ...@@ -2853,8 +2853,12 @@ l1:
* If xwait had just locked the tuple then some other xact * If xwait had just locked the tuple then some other xact
* could update this tuple before we get to this point. Check * could update this tuple before we get to this point. Check
* for xmax change, and start over if so. * for xmax change, and start over if so.
*
* We also must start over if we didn't pin the VM page, and
* the page has become all visible.
*/ */
if (xmax_infomask_changed(tp.t_data->t_infomask, infomask) || if ((vmbuffer == InvalidBuffer && PageIsAllVisible(page)) ||
xmax_infomask_changed(tp.t_data->t_infomask, infomask) ||
!TransactionIdEquals(HeapTupleHeaderGetRawXmax(tp.t_data), !TransactionIdEquals(HeapTupleHeaderGetRawXmax(tp.t_data),
xwait)) xwait))
goto l1; goto l1;
...@@ -2886,8 +2890,12 @@ l1: ...@@ -2886,8 +2890,12 @@ l1:
* xwait is done, but if xwait had just locked the tuple then some * xwait is done, but if xwait had just locked the tuple then some
* other xact could update this tuple before we get to this point. * other xact could update this tuple before we get to this point.
* Check for xmax change, and start over if so. * Check for xmax change, and start over if so.
*
* We also must start over if we didn't pin the VM page, and the
* page has become all visible.
*/ */
if (xmax_infomask_changed(tp.t_data->t_infomask, infomask) || if ((vmbuffer == InvalidBuffer && PageIsAllVisible(page)) ||
xmax_infomask_changed(tp.t_data->t_infomask, infomask) ||
!TransactionIdEquals(HeapTupleHeaderGetRawXmax(tp.t_data), !TransactionIdEquals(HeapTupleHeaderGetRawXmax(tp.t_data),
xwait)) xwait))
goto l1; goto l1;
......
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