Commit 77a1d1e7 authored by Robert Haas's avatar Robert Haas

Department of second thoughts: remove PD_ALL_FROZEN.

Commit a892234f added a second bit per
page to the visibility map, which still seems like a good idea, but it
also added a second page-level bit alongside PD_ALL_VISIBLE to track
whether the visibility map bit was set.  That no longer seems like a
clever plan, because we don't really need that bit for anything.  We
always clear both bits when the page is modified anyway.

Patch by me, reviewed by Kyotaro Horiguchi and Masahiko Sawada.
parent ba0a198f
...@@ -7855,10 +7855,7 @@ heap_xlog_visible(XLogReaderState *record) ...@@ -7855,10 +7855,7 @@ heap_xlog_visible(XLogReaderState *record)
*/ */
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
if (xlrec->flags & VISIBILITYMAP_ALL_VISIBLE) PageSetAllVisible(page);
PageSetAllVisible(page);
if (xlrec->flags & VISIBILITYMAP_ALL_FROZEN)
PageSetAllFrozen(page);
MarkBufferDirty(buffer); MarkBufferDirty(buffer);
} }
......
...@@ -39,15 +39,15 @@ ...@@ -39,15 +39,15 @@
* *
* When we *set* a visibility map during VACUUM, we must write WAL. This may * When we *set* a visibility map during VACUUM, we must write WAL. This may
* seem counterintuitive, since the bit is basically a hint: if it is clear, * seem counterintuitive, since the bit is basically a hint: if it is clear,
* it may still be the case that every tuple on the page is all-visible or * it may still be the case that every tuple on the page is visible to all
* all-frozen we just don't know that for certain. The difficulty is that * transactions; we just don't know that for certain. The difficulty is that
* there are two bits which are typically set together: the PD_ALL_VISIBLE * there are two bits which are typically set together: the PD_ALL_VISIBLE bit
* or PD_ALL_FROZEN bit on the page itself, and the corresponding visibility * on the page itself, and the visibility map bit. If a crash occurs after the
* map bit. If a crash occurs after the visibility map page makes it to disk * visibility map page makes it to disk and before the updated heap page makes
* and before the updated heap page makes it to disk, redo must set the bit on * it to disk, redo must set the bit on the heap page. Otherwise, the next
* the heap page. Otherwise, the next insert, update, or delete on the heap * insert, update, or delete on the heap page will fail to realize that the
* page will fail to realize that the visibility map bit must be cleared, * visibility map bit must be cleared, possibly causing index-only scans to
* possibly causing index-only scans to return wrong answers. * return wrong answers.
* *
* VACUUM will normally skip pages for which the visibility map bit is set; * VACUUM will normally skip pages for which the visibility map bit is set;
* such pages can't contain any dead tuples and therefore don't need vacuuming. * such pages can't contain any dead tuples and therefore don't need vacuuming.
...@@ -251,11 +251,10 @@ visibilitymap_pin_ok(BlockNumber heapBlk, Buffer buf) ...@@ -251,11 +251,10 @@ visibilitymap_pin_ok(BlockNumber heapBlk, Buffer buf)
* to InvalidTransactionId when a page that is already all-visible is being * to InvalidTransactionId when a page that is already all-visible is being
* marked all-frozen. * marked all-frozen.
* *
* Caller is expected to set the heap page's PD_ALL_VISIBLE or PD_ALL_FROZEN * Caller is expected to set the heap page's PD_ALL_VISIBLE bit before calling
* bit before calling this function. Except in recovery, caller should also * this function. Except in recovery, caller should also pass the heap
* pass the heap buffer and flags which indicates what flag we want to set. * buffer. When checksums are enabled and we're not in recovery, we must add
* When checksums are enabled and we're not in recovery, we must add the heap * the heap buffer to the WAL chain to protect it from being torn.
* buffer to the WAL chain to protect it from being torn.
* *
* You must pass a buffer containing the correct map page to this function. * You must pass a buffer containing the correct map page to this function.
* Call visibilitymap_pin first to pin the right one. This function doesn't do * Call visibilitymap_pin first to pin the right one. This function doesn't do
...@@ -315,10 +314,8 @@ visibilitymap_set(Relation rel, BlockNumber heapBlk, Buffer heapBuf, ...@@ -315,10 +314,8 @@ visibilitymap_set(Relation rel, BlockNumber heapBlk, Buffer heapBuf,
{ {
Page heapPage = BufferGetPage(heapBuf); Page heapPage = BufferGetPage(heapBuf);
/* Caller is expected to set page-level bits first. */ /* caller is expected to set PD_ALL_VISIBLE first */
Assert((flags & VISIBILITYMAP_ALL_VISIBLE) == 0 || PageIsAllVisible(heapPage)); Assert(PageIsAllVisible(heapPage));
Assert((flags & VISIBILITYMAP_ALL_FROZEN) == 0 || PageIsAllFrozen(heapPage));
PageSetLSN(heapPage, recptr); PageSetLSN(heapPage, recptr);
} }
} }
......
...@@ -766,7 +766,6 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats, ...@@ -766,7 +766,6 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats,
log_newpage_buffer(buf, true); log_newpage_buffer(buf, true);
PageSetAllVisible(page); PageSetAllVisible(page);
PageSetAllFrozen(page);
visibilitymap_set(onerel, blkno, buf, InvalidXLogRecPtr, visibilitymap_set(onerel, blkno, buf, InvalidXLogRecPtr,
vmbuffer, InvalidTransactionId, vmbuffer, InvalidTransactionId,
VISIBILITYMAP_ALL_VISIBLE | VISIBILITYMAP_ALL_FROZEN); VISIBILITYMAP_ALL_VISIBLE | VISIBILITYMAP_ALL_FROZEN);
...@@ -1024,6 +1023,9 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats, ...@@ -1024,6 +1023,9 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats,
{ {
uint8 flags = VISIBILITYMAP_ALL_VISIBLE; uint8 flags = VISIBILITYMAP_ALL_VISIBLE;
if (all_frozen)
flags |= VISIBILITYMAP_ALL_FROZEN;
/* /*
* It should never be the case that the visibility map page is set * It should never be the case that the visibility map page is set
* while the page-level bit is clear, but the reverse is allowed * while the page-level bit is clear, but the reverse is allowed
...@@ -1038,11 +1040,6 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats, ...@@ -1038,11 +1040,6 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats,
* rare cases after a crash, it is not worth optimizing. * rare cases after a crash, it is not worth optimizing.
*/ */
PageSetAllVisible(page); PageSetAllVisible(page);
if (all_frozen)
{
PageSetAllFrozen(page);
flags |= VISIBILITYMAP_ALL_FROZEN;
}
MarkBufferDirty(buf); MarkBufferDirty(buf);
visibilitymap_set(onerel, blkno, buf, InvalidXLogRecPtr, visibilitymap_set(onerel, blkno, buf, InvalidXLogRecPtr,
vmbuffer, visibility_cutoff_xid, flags); vmbuffer, visibility_cutoff_xid, flags);
...@@ -1093,10 +1090,6 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats, ...@@ -1093,10 +1090,6 @@ lazy_scan_heap(Relation onerel, LVRelStats *vacrelstats,
else if (all_visible_according_to_vm && all_visible && all_frozen && else if (all_visible_according_to_vm && all_visible && all_frozen &&
!VM_ALL_FROZEN(onerel, blkno, &vmbuffer)) !VM_ALL_FROZEN(onerel, blkno, &vmbuffer))
{ {
/* Page is marked all-visible but should be all-frozen */
PageSetAllFrozen(page);
MarkBufferDirty(buf);
/* /*
* We can pass InvalidTransactionId as the cutoff XID here, * We can pass InvalidTransactionId as the cutoff XID here,
* because setting the all-frozen bit doesn't cause recovery * because setting the all-frozen bit doesn't cause recovery
...@@ -1344,11 +1337,7 @@ lazy_vacuum_page(Relation onerel, BlockNumber blkno, Buffer buffer, ...@@ -1344,11 +1337,7 @@ lazy_vacuum_page(Relation onerel, BlockNumber blkno, Buffer buffer,
*/ */
if (heap_page_is_all_visible(onerel, buffer, &visibility_cutoff_xid, if (heap_page_is_all_visible(onerel, buffer, &visibility_cutoff_xid,
&all_frozen)) &all_frozen))
{
PageSetAllVisible(page); PageSetAllVisible(page);
if (all_frozen)
PageSetAllFrozen(page);
}
/* /*
* All the changes to the heap page have been done. If the all-visible * All the changes to the heap page have been done. If the all-visible
......
...@@ -178,10 +178,8 @@ typedef PageHeaderData *PageHeader; ...@@ -178,10 +178,8 @@ typedef PageHeaderData *PageHeader;
* tuple? */ * tuple? */
#define PD_ALL_VISIBLE 0x0004 /* all tuples on page are visible to #define PD_ALL_VISIBLE 0x0004 /* all tuples on page are visible to
* everyone */ * everyone */
#define PD_ALL_FROZEN 0x0008 /* all tuples on page are completely
frozen */
#define PD_VALID_FLAG_BITS 0x000F /* OR of all valid pd_flags bits */ #define PD_VALID_FLAG_BITS 0x0007 /* OR of all valid pd_flags bits */
/* /*
* Page layout version number 0 is for pre-7.3 Postgres releases. * Page layout version number 0 is for pre-7.3 Postgres releases.
...@@ -369,12 +367,7 @@ typedef PageHeaderData *PageHeader; ...@@ -369,12 +367,7 @@ typedef PageHeaderData *PageHeader;
#define PageSetAllVisible(page) \ #define PageSetAllVisible(page) \
(((PageHeader) (page))->pd_flags |= PD_ALL_VISIBLE) (((PageHeader) (page))->pd_flags |= PD_ALL_VISIBLE)
#define PageClearAllVisible(page) \ #define PageClearAllVisible(page) \
(((PageHeader) (page))->pd_flags &= ~(PD_ALL_VISIBLE | PD_ALL_FROZEN)) (((PageHeader) (page))->pd_flags &= ~PD_ALL_VISIBLE)
#define PageIsAllFrozen(page) \
(((PageHeader) (page))->pd_flags & PD_ALL_FROZEN)
#define PageSetAllFrozen(page) \
(((PageHeader) (page))->pd_flags |= PD_ALL_FROZEN)
#define PageIsPrunable(page, oldestxmin) \ #define PageIsPrunable(page, oldestxmin) \
( \ ( \
......
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