Commit 272c2ab9 authored by Alvaro Herrera's avatar Alvaro Herrera

Change some bogus PageGetLSN calls to BufferGetLSNAtomic

As src/backend/access/transam/README says, PageGetLSN may only be called
by processes holding either exclusive lock on buffer, or a shared lock
on buffer plus buffer header lock.  Therefore any place that only holds
a shared buffer lock must use BufferGetLSNAtomic instead of PageGetLSN,
which internally obtains buffer header lock prior to reading the LSN.

A few callsites failed to comply with this rule.  This was detected by
running all tests under a new (not committed) assertion that verifies
PageGetLSN locking contract.  All but one of the callsites that failed
the assertion are fixed by this patch.  Remaining callsites were
inspected manually and determined not to need any change.

The exception (unfixed callsite) is in TestForOldSnapshot, which only
has a Page argument, making it impossible to access the corresponding
Buffer from it.  Fixing that seems a much larger patch that will have to
be done separately; and that's just as well, since it was only
introduced in 9.6 and other bugs are much older.

Some of these bugs are ancient; backpatch all the way back to 9.3.

Authors: Jacob Champion, Asim Praveen, Ashwin Agrawal
Reviewed-by: Michaël Paquier
Discussion: https://postgr.es/m/CABAq_6GXgQDVu3u12mK9O5Xt5abBZWQ0V40LZCE+oUf95XyNFg@mail.gmail.com
parent 11b623dd
...@@ -640,7 +640,8 @@ gistdoinsert(Relation r, IndexTuple itup, Size freespace, GISTSTATE *giststate) ...@@ -640,7 +640,8 @@ gistdoinsert(Relation r, IndexTuple itup, Size freespace, GISTSTATE *giststate)
} }
stack->page = (Page) BufferGetPage(stack->buffer); stack->page = (Page) BufferGetPage(stack->buffer);
stack->lsn = PageGetLSN(stack->page); stack->lsn = xlocked ?
PageGetLSN(stack->page) : BufferGetLSNAtomic(stack->buffer);
Assert(!RelationNeedsWAL(state.r) || !XLogRecPtrIsInvalid(stack->lsn)); Assert(!RelationNeedsWAL(state.r) || !XLogRecPtrIsInvalid(stack->lsn));
/* /*
...@@ -890,7 +891,7 @@ gistFindPath(Relation r, BlockNumber child, OffsetNumber *downlinkoffnum) ...@@ -890,7 +891,7 @@ gistFindPath(Relation r, BlockNumber child, OffsetNumber *downlinkoffnum)
break; break;
} }
top->lsn = PageGetLSN(page); top->lsn = BufferGetLSNAtomic(buffer);
/* /*
* If F_FOLLOW_RIGHT is set, the page to the right doesn't have a * If F_FOLLOW_RIGHT is set, the page to the right doesn't have a
......
...@@ -61,7 +61,7 @@ gistkillitems(IndexScanDesc scan) ...@@ -61,7 +61,7 @@ gistkillitems(IndexScanDesc scan)
* read. killedItems could be not valid so LP_DEAD hints applying is not * read. killedItems could be not valid so LP_DEAD hints applying is not
* safe. * safe.
*/ */
if (PageGetLSN(page) != so->curPageLSN) if (BufferGetLSNAtomic(buffer) != so->curPageLSN)
{ {
UnlockReleaseBuffer(buffer); UnlockReleaseBuffer(buffer);
so->numKilled = 0; /* reset counter */ so->numKilled = 0; /* reset counter */
...@@ -384,7 +384,7 @@ gistScanPage(IndexScanDesc scan, GISTSearchItem *pageItem, double *myDistances, ...@@ -384,7 +384,7 @@ gistScanPage(IndexScanDesc scan, GISTSearchItem *pageItem, double *myDistances,
* safe to apply LP_DEAD hints to the page later. This allows us to drop * safe to apply LP_DEAD hints to the page later. This allows us to drop
* the pin for MVCC scans, which allows vacuum to avoid blocking. * the pin for MVCC scans, which allows vacuum to avoid blocking.
*/ */
so->curPageLSN = PageGetLSN(page); so->curPageLSN = BufferGetLSNAtomic(buffer);
/* /*
* check all tuples on page * check all tuples on page
......
...@@ -249,7 +249,7 @@ gistbulkdelete(IndexVacuumInfo *info, IndexBulkDeleteResult *stats, ...@@ -249,7 +249,7 @@ gistbulkdelete(IndexVacuumInfo *info, IndexBulkDeleteResult *stats,
ptr = (GistBDItem *) palloc(sizeof(GistBDItem)); ptr = (GistBDItem *) palloc(sizeof(GistBDItem));
ptr->blkno = ItemPointerGetBlockNumber(&(idxtuple->t_tid)); ptr->blkno = ItemPointerGetBlockNumber(&(idxtuple->t_tid));
ptr->parentlsn = PageGetLSN(page); ptr->parentlsn = BufferGetLSNAtomic(buffer);
ptr->next = stack->next; ptr->next = stack->next;
stack->next = ptr; stack->next = ptr;
......
...@@ -1224,7 +1224,7 @@ _bt_readpage(IndexScanDesc scan, ScanDirection dir, OffsetNumber offnum) ...@@ -1224,7 +1224,7 @@ _bt_readpage(IndexScanDesc scan, ScanDirection dir, OffsetNumber offnum)
* safe to apply LP_DEAD hints to the page later. This allows us to drop * safe to apply LP_DEAD hints to the page later. This allows us to drop
* the pin for MVCC scans, which allows vacuum to avoid blocking. * the pin for MVCC scans, which allows vacuum to avoid blocking.
*/ */
so->currPos.lsn = PageGetLSN(page); so->currPos.lsn = BufferGetLSNAtomic(so->currPos.buf);
/* /*
* we must save the page's right-link while scanning it; this tells us * we must save the page's right-link while scanning it; this tells us
......
...@@ -1772,7 +1772,7 @@ _bt_killitems(IndexScanDesc scan) ...@@ -1772,7 +1772,7 @@ _bt_killitems(IndexScanDesc scan)
return; return;
page = BufferGetPage(buf); page = BufferGetPage(buf);
if (PageGetLSN(page) == so->currPos.lsn) if (BufferGetLSNAtomic(buf) == so->currPos.lsn)
so->currPos.buf = buf; so->currPos.buf = buf;
else else
{ {
......
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