Commit 5749f6ef authored by Tom Lane's avatar Tom Lane

Rewrite btree vacuuming to fold the former bulkdelete and cleanup operations

into a single mostly-physical-order scan of the index.  This requires some
ticklish interlocking considerations, but should create no material
performance impact on normal index operations (at least given the
already-committed changes to make scans work a page at a time).  VACUUM
itself should get significantly faster in any index that's degenerated to a
very nonlinear page order.  Also, we save one pass over the index entirely,
except in the case where there were no deletions to do and so only one pass
happened anyway.

Original patch by Heikki Linnakangas, rework by Tom Lane.
parent 09cb5c0e
$PostgreSQL: pgsql/src/backend/access/nbtree/README,v 1.11 2006/05/07 01:21:30 tgl Exp $ $PostgreSQL: pgsql/src/backend/access/nbtree/README,v 1.12 2006/05/08 00:00:09 tgl Exp $
This directory contains a correct implementation of Lehman and Yao's This directory contains a correct implementation of Lehman and Yao's
high-concurrency B-tree management algorithm (P. Lehman and S. Yao, high-concurrency B-tree management algorithm (P. Lehman and S. Yao,
...@@ -293,10 +293,32 @@ as part of the atomic update for the delete (either way, the metapage has ...@@ -293,10 +293,32 @@ as part of the atomic update for the delete (either way, the metapage has
to be the last page locked in the update to avoid deadlock risks). This to be the last page locked in the update to avoid deadlock risks). This
avoids race conditions if two such operations are executing concurrently. avoids race conditions if two such operations are executing concurrently.
VACUUM needs to do a linear scan of an index to search for empty leaf VACUUM needs to do a linear scan of an index to search for deleted pages
pages and half-dead parent pages that can be deleted, as well as deleted that can be reclaimed because they are older than all open transactions.
pages that can be reclaimed because they are older than all open For efficiency's sake, we'd like to use the same linear scan to search for
transactions. deletable tuples. Before Postgres 8.2, btbulkdelete scanned the leaf pages
in index order, but it is possible to visit them in physical order instead.
The tricky part of this is to avoid missing any deletable tuples in the
presence of concurrent page splits: a page split could easily move some
tuples from a page not yet passed over by the sequential scan to a
lower-numbered page already passed over. (This wasn't a concern for the
index-order scan, because splits always split right.) To implement this,
we provide a "vacuum cycle ID" mechanism that makes it possible to
determine whether a page has been split since the current btbulkdelete
cycle started. If btbulkdelete finds a page that has been split since
it started, and has a right-link pointing to a lower page number, then
it temporarily suspends its sequential scan and visits that page instead.
It must continue to follow right-links and vacuum dead tuples until
reaching a page that either hasn't been split since btbulkdelete started,
or is above the location of the outer sequential scan. Then it can resume
the sequential scan. This ensures that all tuples are visited. It may be
that some tuples are visited twice, but that has no worse effect than an
inaccurate index tuple count (and we can't guarantee an accurate count
anyway in the face of concurrent activity). Note that this still works
if the has-been-recently-split test has a small probability of false
positives, so long as it never gives a false negative. This makes it
possible to implement the test with a small counter value stored on each
index page.
WAL considerations WAL considerations
------------------ ------------------
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/access/nbtree/nbtinsert.c,v 1.136 2006/04/25 22:46:05 tgl Exp $ * $PostgreSQL: pgsql/src/backend/access/nbtree/nbtinsert.c,v 1.137 2006/05/08 00:00:09 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -700,14 +700,18 @@ _bt_split(Relation rel, Buffer buf, OffsetNumber firstright, ...@@ -700,14 +700,18 @@ _bt_split(Relation rel, Buffer buf, OffsetNumber firstright,
ropaque = (BTPageOpaque) PageGetSpecialPointer(rightpage); ropaque = (BTPageOpaque) PageGetSpecialPointer(rightpage);
/* if we're splitting this page, it won't be the root when we're done */ /* if we're splitting this page, it won't be the root when we're done */
/* also, clear the SPLIT_END flag in both pages */
lopaque->btpo_flags = oopaque->btpo_flags; lopaque->btpo_flags = oopaque->btpo_flags;
lopaque->btpo_flags &= ~BTP_ROOT; lopaque->btpo_flags &= ~(BTP_ROOT | BTP_SPLIT_END);
ropaque->btpo_flags = lopaque->btpo_flags; ropaque->btpo_flags = lopaque->btpo_flags;
lopaque->btpo_prev = oopaque->btpo_prev; lopaque->btpo_prev = oopaque->btpo_prev;
lopaque->btpo_next = BufferGetBlockNumber(rbuf); lopaque->btpo_next = BufferGetBlockNumber(rbuf);
ropaque->btpo_prev = BufferGetBlockNumber(buf); ropaque->btpo_prev = BufferGetBlockNumber(buf);
ropaque->btpo_next = oopaque->btpo_next; ropaque->btpo_next = oopaque->btpo_next;
lopaque->btpo.level = ropaque->btpo.level = oopaque->btpo.level; lopaque->btpo.level = ropaque->btpo.level = oopaque->btpo.level;
/* Since we already have write-lock on both pages, ok to read cycleid */
lopaque->btpo_cycleid = _bt_vacuum_cycleid(rel);
ropaque->btpo_cycleid = lopaque->btpo_cycleid;
/* /*
* If the page we're splitting is not the rightmost page at its level in * If the page we're splitting is not the rightmost page at its level in
...@@ -836,6 +840,21 @@ _bt_split(Relation rel, Buffer buf, OffsetNumber firstright, ...@@ -836,6 +840,21 @@ _bt_split(Relation rel, Buffer buf, OffsetNumber firstright,
sopaque = (BTPageOpaque) PageGetSpecialPointer(spage); sopaque = (BTPageOpaque) PageGetSpecialPointer(spage);
if (sopaque->btpo_prev != ropaque->btpo_prev) if (sopaque->btpo_prev != ropaque->btpo_prev)
elog(PANIC, "right sibling's left-link doesn't match"); elog(PANIC, "right sibling's left-link doesn't match");
/*
* Check to see if we can set the SPLIT_END flag in the right-hand
* split page; this can save some I/O for vacuum since it need not
* proceed to the right sibling. We can set the flag if the right
* sibling has a different cycleid: that means it could not be part
* of a group of pages that were all split off from the same ancestor
* page. If you're confused, imagine that page A splits to A B and
* then again, yielding A C B, while vacuum is in progress. Tuples
* originally in A could now be in either B or C, hence vacuum must
* examine both pages. But if D, our right sibling, has a different
* cycleid then it could not contain any tuples that were in A when
* the vacuum started.
*/
if (sopaque->btpo_cycleid != ropaque->btpo_cycleid)
ropaque->btpo_flags |= BTP_SPLIT_END;
} }
/* /*
...@@ -1445,6 +1464,7 @@ _bt_newroot(Relation rel, Buffer lbuf, Buffer rbuf) ...@@ -1445,6 +1464,7 @@ _bt_newroot(Relation rel, Buffer lbuf, Buffer rbuf)
rootopaque->btpo_flags = BTP_ROOT; rootopaque->btpo_flags = BTP_ROOT;
rootopaque->btpo.level = rootopaque->btpo.level =
((BTPageOpaque) PageGetSpecialPointer(lpage))->btpo.level + 1; ((BTPageOpaque) PageGetSpecialPointer(lpage))->btpo.level + 1;
rootopaque->btpo_cycleid = 0;
/* update metapage data */ /* update metapage data */
metad->btm_root = rootblknum; metad->btm_root = rootblknum;
......
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/access/nbtree/nbtpage.c,v 1.96 2006/04/25 22:46:05 tgl Exp $ * $PostgreSQL: pgsql/src/backend/access/nbtree/nbtpage.c,v 1.97 2006/05/08 00:00:10 tgl Exp $
* *
* NOTES * NOTES
* Postgres btree pages look like ordinary relation pages. The opaque * Postgres btree pages look like ordinary relation pages. The opaque
...@@ -206,6 +206,7 @@ _bt_getroot(Relation rel, int access) ...@@ -206,6 +206,7 @@ _bt_getroot(Relation rel, int access)
rootopaque->btpo_prev = rootopaque->btpo_next = P_NONE; rootopaque->btpo_prev = rootopaque->btpo_next = P_NONE;
rootopaque->btpo_flags = (BTP_LEAF | BTP_ROOT); rootopaque->btpo_flags = (BTP_LEAF | BTP_ROOT);
rootopaque->btpo.level = 0; rootopaque->btpo.level = 0;
rootopaque->btpo_cycleid = 0;
/* NO ELOG(ERROR) till meta is updated */ /* NO ELOG(ERROR) till meta is updated */
START_CRIT_SECTION(); START_CRIT_SECTION();
...@@ -544,7 +545,7 @@ _bt_getbuf(Relation rel, BlockNumber blkno, int access) ...@@ -544,7 +545,7 @@ _bt_getbuf(Relation rel, BlockNumber blkno, int access)
* Release the file-extension lock; it's now OK for someone else to * Release the file-extension lock; it's now OK for someone else to
* extend the relation some more. Note that we cannot release this * extend the relation some more. Note that we cannot release this
* lock before we have buffer lock on the new page, or we risk a race * lock before we have buffer lock on the new page, or we risk a race
* condition against btvacuumcleanup --- see comments therein. * condition against btvacuumscan --- see comments therein.
*/ */
if (needLock) if (needLock)
UnlockRelationForExtension(rel, ExclusiveLock); UnlockRelationForExtension(rel, ExclusiveLock);
...@@ -608,7 +609,7 @@ _bt_pageinit(Page page, Size size) ...@@ -608,7 +609,7 @@ _bt_pageinit(Page page, Size size)
/* /*
* _bt_page_recyclable() -- Is an existing page recyclable? * _bt_page_recyclable() -- Is an existing page recyclable?
* *
* This exists to make sure _bt_getbuf and btvacuumcleanup have the same * This exists to make sure _bt_getbuf and btvacuumscan have the same
* policy about whether a page is safe to re-use. * policy about whether a page is safe to re-use.
*/ */
bool bool
...@@ -651,6 +652,7 @@ _bt_delitems(Relation rel, Buffer buf, ...@@ -651,6 +652,7 @@ _bt_delitems(Relation rel, Buffer buf,
OffsetNumber *itemnos, int nitems) OffsetNumber *itemnos, int nitems)
{ {
Page page = BufferGetPage(buf); Page page = BufferGetPage(buf);
BTPageOpaque opaque;
/* No ereport(ERROR) until changes are logged */ /* No ereport(ERROR) until changes are logged */
START_CRIT_SECTION(); START_CRIT_SECTION();
...@@ -658,6 +660,13 @@ _bt_delitems(Relation rel, Buffer buf, ...@@ -658,6 +660,13 @@ _bt_delitems(Relation rel, Buffer buf,
/* Fix the page */ /* Fix the page */
PageIndexMultiDelete(page, itemnos, nitems); PageIndexMultiDelete(page, itemnos, nitems);
/*
* We can clear the vacuum cycle ID since this page has certainly
* been processed by the current vacuum scan.
*/
opaque = (BTPageOpaque) PageGetSpecialPointer(page);
opaque->btpo_cycleid = 0;
MarkBufferDirty(buf); MarkBufferDirty(buf);
/* XLOG stuff */ /* XLOG stuff */
......
...@@ -12,7 +12,7 @@ ...@@ -12,7 +12,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/access/nbtree/nbtree.c,v 1.147 2006/05/07 01:21:30 tgl Exp $ * $PostgreSQL: pgsql/src/backend/access/nbtree/nbtree.c,v 1.148 2006/05/08 00:00:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -47,6 +47,20 @@ typedef struct ...@@ -47,6 +47,20 @@ typedef struct
double indtuples; double indtuples;
} BTBuildState; } BTBuildState;
/* Working state needed by btvacuumpage */
typedef struct
{
IndexVacuumInfo *info;
IndexBulkDeleteResult *stats;
IndexBulkDeleteCallback callback;
void *callback_state;
BTCycleId cycleid;
BlockNumber *freePages;
int nFreePages;
int maxFreePages;
MemoryContext pagedelcontext;
} BTVacState;
static void btbuildCallback(Relation index, static void btbuildCallback(Relation index,
HeapTuple htup, HeapTuple htup,
...@@ -54,6 +68,11 @@ static void btbuildCallback(Relation index, ...@@ -54,6 +68,11 @@ static void btbuildCallback(Relation index,
bool *isnull, bool *isnull,
bool tupleIsAlive, bool tupleIsAlive,
void *state); void *state);
static void btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats,
IndexBulkDeleteCallback callback, void *callback_state,
BTCycleId cycleid);
static void btvacuumpage(BTVacState *vstate, BlockNumber blkno,
BlockNumber orig_blkno);
/* /*
...@@ -492,192 +511,257 @@ Datum ...@@ -492,192 +511,257 @@ Datum
btbulkdelete(PG_FUNCTION_ARGS) btbulkdelete(PG_FUNCTION_ARGS)
{ {
IndexVacuumInfo *info = (IndexVacuumInfo *) PG_GETARG_POINTER(0); IndexVacuumInfo *info = (IndexVacuumInfo *) PG_GETARG_POINTER(0);
IndexBulkDeleteResult *stats = (IndexBulkDeleteResult *) PG_GETARG_POINTER(1); IndexBulkDeleteResult * volatile stats = (IndexBulkDeleteResult *) PG_GETARG_POINTER(1);
IndexBulkDeleteCallback callback = (IndexBulkDeleteCallback) PG_GETARG_POINTER(2); IndexBulkDeleteCallback callback = (IndexBulkDeleteCallback) PG_GETARG_POINTER(2);
void *callback_state = (void *) PG_GETARG_POINTER(3); void *callback_state = (void *) PG_GETARG_POINTER(3);
Relation rel = info->index; Relation rel = info->index;
double tuples_removed = 0; BTCycleId cycleid;
OffsetNumber deletable[MaxOffsetNumber];
int ndeletable;
Buffer buf;
/* /* allocate stats if first time through, else re-use existing struct */
* The outer loop iterates over index leaf pages, the inner over items on if (stats == NULL)
* a leaf page. We issue just one _bt_delitems() call per page, so as to stats = (IndexBulkDeleteResult *) palloc0(sizeof(IndexBulkDeleteResult));
* minimize WAL traffic.
*
* Note that we exclusive-lock every leaf page containing data items, in
* sequence left to right. It sounds attractive to only exclusive-lock
* those containing items we need to delete, but unfortunately that is not
* safe: we could then pass a stopped indexscan, which could in rare cases
* lead to deleting items that the indexscan will still return later.
* (See discussion in nbtree/README.) We can skip obtaining exclusive
* lock on empty pages though, since no indexscan could be stopped on
* those. (Note: this presumes that a split couldn't have left either
* page totally empty.)
*/
buf = _bt_get_endpoint(rel, 0, false);
if (BufferIsValid(buf)) /* check for empty index */ /* Establish the vacuum cycle ID to use for this scan */
PG_TRY();
{ {
for (;;) cycleid = _bt_start_vacuum(rel);
{
Page page;
BTPageOpaque opaque;
OffsetNumber offnum,
minoff,
maxoff;
BlockNumber nextpage;
ndeletable = 0; btvacuumscan(info, stats, callback, callback_state, cycleid);
page = BufferGetPage(buf);
opaque = (BTPageOpaque) PageGetSpecialPointer(page); _bt_end_vacuum(rel);
minoff = P_FIRSTDATAKEY(opaque); }
maxoff = PageGetMaxOffsetNumber(page); PG_CATCH();
/* We probably cannot see deleted pages, but skip 'em if so */
if (minoff <= maxoff && !P_ISDELETED(opaque))
{ {
/* /* Make sure shared memory gets cleaned up */
* Trade in the initial read lock for a super-exclusive write _bt_end_vacuum(rel);
* lock on this page. PG_RE_THROW();
*/ }
LockBuffer(buf, BUFFER_LOCK_UNLOCK); PG_END_TRY();
LockBufferForCleanup(buf);
/* PG_RETURN_POINTER(stats);
* Recompute minoff/maxoff, both of which could have changed }
* while we weren't holding the lock.
/*
* Post-VACUUM cleanup.
*
* Result: a palloc'd struct containing statistical info for VACUUM displays.
*/ */
minoff = P_FIRSTDATAKEY(opaque); Datum
maxoff = PageGetMaxOffsetNumber(page); btvacuumcleanup(PG_FUNCTION_ARGS)
{
IndexVacuumInfo *info = (IndexVacuumInfo *) PG_GETARG_POINTER(0);
IndexBulkDeleteResult *stats = (IndexBulkDeleteResult *) PG_GETARG_POINTER(1);
/* /*
* Scan over all items to see which ones need deleted * If btbulkdelete was called, we need not do anything, just return
* according to the callback function. * the stats from the latest btbulkdelete call. If it wasn't called,
* we must still do a pass over the index, to recycle any newly-recyclable
* pages and to obtain index statistics.
*
* Since we aren't going to actually delete any leaf items, there's no
* need to go through all the vacuum-cycle-ID pushups.
*/ */
for (offnum = minoff; if (stats == NULL)
offnum <= maxoff;
offnum = OffsetNumberNext(offnum))
{
IndexTuple itup;
ItemPointer htup;
itup = (IndexTuple)
PageGetItem(page, PageGetItemId(page, offnum));
htup = &(itup->t_tid);
if (callback(htup, callback_state))
{ {
deletable[ndeletable++] = offnum; stats = (IndexBulkDeleteResult *) palloc0(sizeof(IndexBulkDeleteResult));
tuples_removed += 1; btvacuumscan(info, stats, NULL, NULL, 0);
}
}
} }
/* Apply any needed deletes */ /*
if (ndeletable > 0) * During a non-FULL vacuum it's quite possible for us to be fooled by
_bt_delitems(rel, buf, deletable, ndeletable); * concurrent page splits into double-counting some index tuples, so
* disbelieve any total that exceeds the underlying heap's count.
/* Fetch nextpage link before releasing the buffer */ * (We can't check this during btbulkdelete.)
nextpage = opaque->btpo_next; */
_bt_relbuf(rel, buf); if (!info->vacuum_full)
{
/* call vacuum_delay_point while not holding any buffer lock */ if (stats->num_index_tuples > info->num_heap_tuples)
vacuum_delay_point(); stats->num_index_tuples = info->num_heap_tuples;
/* And advance to next page, if any */
if (nextpage == P_NONE)
break;
buf = _bt_getbuf(rel, nextpage, BT_READ);
}
} }
/* return statistics */
if (stats == NULL)
stats = (IndexBulkDeleteResult *) palloc0(sizeof(IndexBulkDeleteResult));
stats->tuples_removed += tuples_removed;
/* btvacuumcleanup will fill in num_pages and num_index_tuples */
PG_RETURN_POINTER(stats); PG_RETURN_POINTER(stats);
} }
/* /*
* Post-VACUUM cleanup. * btvacuumscan --- scan the index for VACUUMing purposes
* *
* Here, we scan looking for pages we can delete or return to the freelist. * This combines the functions of looking for leaf tuples that are deletable
* according to the vacuum callback, looking for empty pages that can be
* deleted, and looking for old deleted pages that can be recycled. Both
* btbulkdelete and btvacuumcleanup invoke this (the latter only if no
* btbulkdelete call occurred).
* *
* Result: a palloc'd struct containing statistical info for VACUUM displays. * The caller is responsible for initially allocating/zeroing a stats struct
* and for obtaining a vacuum cycle ID if necessary.
*/ */
Datum static void
btvacuumcleanup(PG_FUNCTION_ARGS) btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats,
IndexBulkDeleteCallback callback, void *callback_state,
BTCycleId cycleid)
{ {
IndexVacuumInfo *info = (IndexVacuumInfo *) PG_GETARG_POINTER(0);
IndexBulkDeleteResult *stats = (IndexBulkDeleteResult *) PG_GETARG_POINTER(1);
Relation rel = info->index; Relation rel = info->index;
BTVacState vstate;
BlockNumber num_pages; BlockNumber num_pages;
BlockNumber blkno; BlockNumber blkno;
BlockNumber *freePages;
int nFreePages,
maxFreePages;
double num_index_tuples = 0;
BlockNumber pages_deleted = 0;
MemoryContext mycontext;
MemoryContext oldcontext;
bool needLock; bool needLock;
/* Set up all-zero stats if btbulkdelete wasn't called */ /*
if (stats == NULL) * Reset counts that will be incremented during the scan; needed in
stats = (IndexBulkDeleteResult *) palloc0(sizeof(IndexBulkDeleteResult)); * case of multiple scans during a single VACUUM command
*/
stats->num_index_tuples = 0;
stats->pages_deleted = 0;
/* Set up info to pass down to btvacuumpage */
vstate.info = info;
vstate.stats = stats;
vstate.callback = callback;
vstate.callback_state = callback_state;
vstate.cycleid = cycleid;
vstate.freePages = NULL; /* temporarily */
vstate.nFreePages = 0;
vstate.maxFreePages = 0;
/* Create a temporary memory context to run _bt_pagedel in */
vstate.pagedelcontext = AllocSetContextCreate(CurrentMemoryContext,
"_bt_pagedel",
ALLOCSET_DEFAULT_MINSIZE,
ALLOCSET_DEFAULT_INITSIZE,
ALLOCSET_DEFAULT_MAXSIZE);
/* /*
* First find out the number of pages in the index. We must acquire the * The outer loop iterates over all index pages except the metapage,
* relation-extension lock while doing this to avoid a race condition: if * in physical order (we hope the kernel will cooperate in providing
* someone else is extending the relation, there is a window where * read-ahead for speed). It is critical that we visit all leaf pages,
* bufmgr/smgr have created a new all-zero page but it hasn't yet been * including ones added after we start the scan, else we might fail to
* write-locked by _bt_getbuf(). If we manage to scan such a page here, * delete some deletable tuples. Hence, we must repeatedly check the
* we'll improperly assume it can be recycled. Taking the lock * relation length. We must acquire the relation-extension lock while
* synchronizes things enough to prevent a problem: either num_pages won't * doing so to avoid a race condition: if someone else is extending the
* include the new page, or _bt_getbuf already has write lock on the * relation, there is a window where bufmgr/smgr have created a new
* buffer and it will be fully initialized before we can examine it. (See * all-zero page but it hasn't yet been write-locked by _bt_getbuf().
* also vacuumlazy.c, which has the same issue.) * If we manage to scan such a page here, we'll improperly assume it can
* be recycled. Taking the lock synchronizes things enough to prevent a
* problem: either num_pages won't include the new page, or _bt_getbuf
* already has write lock on the buffer and it will be fully initialized
* before we can examine it. (See also vacuumlazy.c, which has the same
* issue.) Also, we need not worry if a page is added immediately after
* we look; the page splitting code already has write-lock on the left
* page before it adds a right page, so we must already have processed
* any tuples due to be moved into such a page.
* *
* We can skip locking for new or temp relations, however, since no one * We can skip locking for new or temp relations, however, since no one
* else could be accessing them. * else could be accessing them.
*/ */
needLock = !RELATION_IS_LOCAL(rel); needLock = !RELATION_IS_LOCAL(rel);
blkno = BTREE_METAPAGE + 1;
for (;;)
{
/* Get the current relation length */
if (needLock) if (needLock)
LockRelationForExtension(rel, ExclusiveLock); LockRelationForExtension(rel, ExclusiveLock);
num_pages = RelationGetNumberOfBlocks(rel); num_pages = RelationGetNumberOfBlocks(rel);
if (needLock) if (needLock)
UnlockRelationForExtension(rel, ExclusiveLock); UnlockRelationForExtension(rel, ExclusiveLock);
/* Allocate freePages after we read num_pages the first time */
if (vstate.freePages == NULL)
{
/* No point in remembering more than MaxFSMPages pages */ /* No point in remembering more than MaxFSMPages pages */
maxFreePages = MaxFSMPages; vstate.maxFreePages = MaxFSMPages;
if ((BlockNumber) maxFreePages > num_pages) if ((BlockNumber) vstate.maxFreePages > num_pages)
maxFreePages = (int) num_pages; vstate.maxFreePages = (int) num_pages;
freePages = (BlockNumber *) palloc(maxFreePages * sizeof(BlockNumber)); vstate.freePages = (BlockNumber *)
nFreePages = 0; palloc(vstate.maxFreePages * sizeof(BlockNumber));
}
/* Create a temporary memory context to run _bt_pagedel in */ /* Quit if we've scanned the whole relation */
mycontext = AllocSetContextCreate(CurrentMemoryContext, if (blkno >= num_pages)
"_bt_pagedel", break;
ALLOCSET_DEFAULT_MINSIZE, /* Iterate over pages, then loop back to recheck length */
ALLOCSET_DEFAULT_INITSIZE, for (; blkno < num_pages; blkno++)
ALLOCSET_DEFAULT_MAXSIZE); {
btvacuumpage(&vstate, blkno, blkno);
}
}
/* /*
* Scan through all pages of index, except metapage. (Any pages added * During VACUUM FULL, we truncate off any recyclable pages at the end of
* after we start the scan will not be examined; this should be fine, * the index. In a normal vacuum it'd be unsafe to do this except by
* since they can't possibly be empty.) * acquiring exclusive lock on the index and then rechecking all the
* pages; doesn't seem worth it.
*/ */
for (blkno = BTREE_METAPAGE + 1; blkno < num_pages; blkno++) if (info->vacuum_full && vstate.nFreePages > 0)
{
BlockNumber new_pages = num_pages;
while (vstate.nFreePages > 0 &&
vstate.freePages[vstate.nFreePages - 1] == new_pages - 1)
{
new_pages--;
stats->pages_deleted--;
vstate.nFreePages--;
}
if (new_pages != num_pages)
{ {
/*
* Okay to truncate.
*/
RelationTruncate(rel, new_pages);
/* update statistics */
stats->pages_removed += num_pages - new_pages;
num_pages = new_pages;
}
}
/*
* Update the shared Free Space Map with the info we now have about free
* pages in the index, discarding any old info the map may have. We do not
* need to sort the page numbers; they're in order already.
*/
RecordIndexFreeSpace(&rel->rd_node, vstate.nFreePages, vstate.freePages);
pfree(vstate.freePages);
MemoryContextDelete(vstate.pagedelcontext);
/* update statistics */
stats->num_pages = num_pages;
stats->pages_free = vstate.nFreePages;
}
/*
* btvacuumpage --- VACUUM one page
*
* This processes a single page for btvacuumscan(). In some cases we
* must go back and re-examine previously-scanned pages; this routine
* recurses when necessary to handle that case.
*
* blkno is the page to process. orig_blkno is the highest block number
* reached by the outer btvacuumscan loop (the same as blkno, unless we
* are recursing to re-examine a previous page).
*/
static void
btvacuumpage(BTVacState *vstate, BlockNumber blkno, BlockNumber orig_blkno)
{
IndexVacuumInfo *info = vstate->info;
IndexBulkDeleteResult *stats = vstate->stats;
IndexBulkDeleteCallback callback = vstate->callback;
void *callback_state = vstate->callback_state;
Relation rel = info->index;
bool delete_now;
BlockNumber recurse_to;
Buffer buf; Buffer buf;
Page page; Page page;
BTPageOpaque opaque; BTPageOpaque opaque;
restart:
delete_now = false;
recurse_to = P_NONE;
/* call vacuum_delay_point while not holding any buffer lock */
vacuum_delay_point(); vacuum_delay_point();
/* /*
...@@ -691,33 +775,156 @@ btvacuumcleanup(PG_FUNCTION_ARGS) ...@@ -691,33 +775,156 @@ btvacuumcleanup(PG_FUNCTION_ARGS)
opaque = (BTPageOpaque) PageGetSpecialPointer(page); opaque = (BTPageOpaque) PageGetSpecialPointer(page);
if (!PageIsNew(page)) if (!PageIsNew(page))
_bt_checkpage(rel, buf); _bt_checkpage(rel, buf);
/*
* If we are recursing, the only case we want to do anything with is
* a live leaf page having the current vacuum cycle ID. Any other state
* implies we already saw the page (eg, deleted it as being empty).
* In particular, we don't want to risk adding it to freePages twice.
*/
if (blkno != orig_blkno)
{
if (_bt_page_recyclable(page) ||
P_ISDELETED(opaque) ||
(opaque->btpo_flags & BTP_HALF_DEAD) ||
!P_ISLEAF(opaque) ||
opaque->btpo_cycleid != vstate->cycleid)
{
_bt_relbuf(rel, buf);
return;
}
}
/* Page is valid, see what to do with it */
if (_bt_page_recyclable(page)) if (_bt_page_recyclable(page))
{ {
/* Okay to recycle this page */ /* Okay to recycle this page */
if (nFreePages < maxFreePages) if (vstate->nFreePages < vstate->maxFreePages)
freePages[nFreePages++] = blkno; vstate->freePages[vstate->nFreePages++] = blkno;
pages_deleted++; stats->pages_deleted++;
} }
else if (P_ISDELETED(opaque)) else if (P_ISDELETED(opaque))
{ {
/* Already deleted, but can't recycle yet */ /* Already deleted, but can't recycle yet */
pages_deleted++; stats->pages_deleted++;
}
else if (opaque->btpo_flags & BTP_HALF_DEAD)
{
/* Half-dead, try to delete */
delete_now = true;
}
else if (P_ISLEAF(opaque))
{
OffsetNumber deletable[MaxOffsetNumber];
int ndeletable;
OffsetNumber offnum,
minoff,
maxoff;
/*
* Trade in the initial read lock for a super-exclusive write
* lock on this page. We must get such a lock on every leaf page
* over the course of the vacuum scan, whether or not it actually
* contains any deletable tuples --- see nbtree/README.
*/
LockBuffer(buf, BUFFER_LOCK_UNLOCK);
LockBufferForCleanup(buf);
/*
* Check whether we need to recurse back to earlier pages. What
* we are concerned about is a page split that happened since we
* started the vacuum scan. If the split moved some tuples to a
* lower page then we might have missed 'em. If so, set up for
* tail recursion. (Must do this before possibly clearing
* btpo_cycleid below!)
*/
if (vstate->cycleid != 0 &&
opaque->btpo_cycleid == vstate->cycleid &&
!(opaque->btpo_flags & BTP_SPLIT_END) &&
!P_RIGHTMOST(opaque) &&
opaque->btpo_next < orig_blkno)
recurse_to = opaque->btpo_next;
/*
* Scan over all items to see which ones need deleted
* according to the callback function.
*/
ndeletable = 0;
minoff = P_FIRSTDATAKEY(opaque);
maxoff = PageGetMaxOffsetNumber(page);
if (callback)
{
for (offnum = minoff;
offnum <= maxoff;
offnum = OffsetNumberNext(offnum))
{
IndexTuple itup;
ItemPointer htup;
itup = (IndexTuple) PageGetItem(page,
PageGetItemId(page, offnum));
htup = &(itup->t_tid);
if (callback(htup, callback_state))
deletable[ndeletable++] = offnum;
}
} }
else if ((opaque->btpo_flags & BTP_HALF_DEAD) ||
P_FIRSTDATAKEY(opaque) > PageGetMaxOffsetNumber(page)) /*
* Apply any needed deletes. We issue just one _bt_delitems()
* call per page, so as to minimize WAL traffic.
*/
if (ndeletable > 0)
{ {
/* Empty, try to delete */ _bt_delitems(rel, buf, deletable, ndeletable);
stats->tuples_removed += ndeletable;
/* must recompute maxoff */
maxoff = PageGetMaxOffsetNumber(page);
}
else
{
/*
* If the page has been split during this vacuum cycle, it seems
* worth expending a write to clear btpo_cycleid even if we don't
* have any deletions to do. (If we do, _bt_delitems takes care
* of this.) This ensures we won't process the page again.
*
* We treat this like a hint-bit update because there's no need
* to WAL-log it.
*/
if (vstate->cycleid != 0 &&
opaque->btpo_cycleid == vstate->cycleid)
{
opaque->btpo_cycleid = 0;
SetBufferCommitInfoNeedsSave(buf);
}
}
/*
* If it's now empty, try to delete; else count the live tuples.
* We don't delete when recursing, though, to avoid putting entries
* into freePages out-of-order (doesn't seem worth any extra code to
* handle the case).
*/
if (minoff > maxoff)
delete_now = (blkno == orig_blkno);
else
stats->num_index_tuples += maxoff - minoff + 1;
}
if (delete_now)
{
MemoryContext oldcontext;
int ndel; int ndel;
/* Run pagedel in a temp context to avoid memory leakage */ /* Run pagedel in a temp context to avoid memory leakage */
MemoryContextReset(mycontext); MemoryContextReset(vstate->pagedelcontext);
oldcontext = MemoryContextSwitchTo(mycontext); oldcontext = MemoryContextSwitchTo(vstate->pagedelcontext);
ndel = _bt_pagedel(rel, buf, info->vacuum_full); ndel = _bt_pagedel(rel, buf, info->vacuum_full);
/* count only this page, else may double-count parent */ /* count only this page, else may double-count parent */
if (ndel) if (ndel)
pages_deleted++; stats->pages_deleted++;
/* /*
* During VACUUM FULL it's okay to recycle deleted pages * During VACUUM FULL it's okay to recycle deleted pages
...@@ -730,68 +937,26 @@ btvacuumcleanup(PG_FUNCTION_ARGS) ...@@ -730,68 +937,26 @@ btvacuumcleanup(PG_FUNCTION_ARGS)
*/ */
if (ndel && info->vacuum_full) if (ndel && info->vacuum_full)
{ {
if (nFreePages < maxFreePages) if (vstate->nFreePages < vstate->maxFreePages)
freePages[nFreePages++] = blkno; vstate->freePages[vstate->nFreePages++] = blkno;
} }
MemoryContextSwitchTo(oldcontext); MemoryContextSwitchTo(oldcontext);
continue; /* pagedel released buffer */ /* pagedel released buffer, so we shouldn't */
}
else if (P_ISLEAF(opaque))
{
/* Count the index entries of live leaf pages */
num_index_tuples += PageGetMaxOffsetNumber(page) + 1 -
P_FIRSTDATAKEY(opaque);
} }
else
_bt_relbuf(rel, buf); _bt_relbuf(rel, buf);
}
/* /*
* During VACUUM FULL, we truncate off any recyclable pages at the end of * This is really tail recursion, but if the compiler is too stupid
* the index. In a normal vacuum it'd be unsafe to do this except by * to optimize it as such, we'd eat an uncomfortably large amount of
* acquiring exclusive lock on the index and then rechecking all the * stack space per recursion level (due to the deletable[] array).
* pages; doesn't seem worth it. * A failure is improbable since the number of levels isn't likely to be
* large ... but just in case, let's hand-optimize into a loop.
*/ */
if (info->vacuum_full && nFreePages > 0) if (recurse_to != P_NONE)
{ {
BlockNumber new_pages = num_pages; blkno = recurse_to;
goto restart;
while (nFreePages > 0 && freePages[nFreePages - 1] == new_pages - 1)
{
new_pages--;
pages_deleted--;
nFreePages--;
} }
if (new_pages != num_pages)
{
/*
* Okay to truncate.
*/
RelationTruncate(rel, new_pages);
/* update statistics */
stats->pages_removed = num_pages - new_pages;
num_pages = new_pages;
}
}
/*
* Update the shared Free Space Map with the info we now have about free
* pages in the index, discarding any old info the map may have. We do not
* need to sort the page numbers; they're in order already.
*/
RecordIndexFreeSpace(&rel->rd_node, nFreePages, freePages);
pfree(freePages);
MemoryContextDelete(mycontext);
/* update statistics */
stats->num_pages = num_pages;
stats->num_index_tuples = num_index_tuples;
stats->pages_deleted = pages_deleted;
stats->pages_free = nFreePages;
PG_RETURN_POINTER(stats);
} }
...@@ -56,7 +56,7 @@ ...@@ -56,7 +56,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/access/nbtree/nbtsort.c,v 1.100 2006/03/10 20:18:15 tgl Exp $ * $PostgreSQL: pgsql/src/backend/access/nbtree/nbtsort.c,v 1.101 2006/05/08 00:00:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -244,6 +244,7 @@ _bt_blnewpage(uint32 level) ...@@ -244,6 +244,7 @@ _bt_blnewpage(uint32 level)
opaque->btpo_prev = opaque->btpo_next = P_NONE; opaque->btpo_prev = opaque->btpo_next = P_NONE;
opaque->btpo.level = level; opaque->btpo.level = level;
opaque->btpo_flags = (level > 0) ? 0 : BTP_LEAF; opaque->btpo_flags = (level > 0) ? 0 : BTP_LEAF;
opaque->btpo_cycleid = 0;
/* Make the P_HIKEY line pointer appear allocated */ /* Make the P_HIKEY line pointer appear allocated */
((PageHeader) page)->pd_lower += sizeof(ItemIdData); ((PageHeader) page)->pd_lower += sizeof(ItemIdData);
......
...@@ -8,17 +8,20 @@ ...@@ -8,17 +8,20 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/access/nbtree/nbtutils.c,v 1.73 2006/05/07 01:21:30 tgl Exp $ * $PostgreSQL: pgsql/src/backend/access/nbtree/nbtutils.c,v 1.74 2006/05/08 00:00:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#include "postgres.h" #include "postgres.h"
#include <time.h>
#include "access/genam.h" #include "access/genam.h"
#include "access/nbtree.h" #include "access/nbtree.h"
#include "catalog/catalog.h" #include "catalog/catalog.h"
#include "executor/execdebug.h" #include "executor/execdebug.h"
#include "miscadmin.h"
static void _bt_mark_scankey_required(ScanKey skey); static void _bt_mark_scankey_required(ScanKey skey);
...@@ -820,11 +823,13 @@ _bt_check_rowcompare(ScanKey skey, IndexTuple tuple, TupleDesc tupdesc, ...@@ -820,11 +823,13 @@ _bt_check_rowcompare(ScanKey skey, IndexTuple tuple, TupleDesc tupdesc,
* delete. We cope with cases where items have moved right due to insertions. * delete. We cope with cases where items have moved right due to insertions.
* If an item has moved off the current page due to a split, we'll fail to * If an item has moved off the current page due to a split, we'll fail to
* find it and do nothing (this is not an error case --- we assume the item * find it and do nothing (this is not an error case --- we assume the item
* will eventually get marked in a future indexscan). Likewise, if the item * will eventually get marked in a future indexscan). Note that because we
* has moved left due to deletions or disappeared itself, we'll not find it, * hold pin on the target page continuously from initially reading the items
* but these cases are not worth optimizing. (Since deletions are only done * until applying this function, VACUUM cannot have deleted any items from
* by VACUUM, any deletion makes it highly likely that the dead item has been * the page, and so there is no need to search left from the recorded offset.
* removed itself, and therefore searching left is not worthwhile.) * (This observation also guarantees that the item is still the right one
* to delete, which might otherwise be questionable since heap TIDs can get
* recycled.)
*/ */
void void
_bt_killitems(IndexScanDesc scan, bool haveLock) _bt_killitems(IndexScanDesc scan, bool haveLock)
...@@ -890,3 +895,187 @@ _bt_killitems(IndexScanDesc scan, bool haveLock) ...@@ -890,3 +895,187 @@ _bt_killitems(IndexScanDesc scan, bool haveLock)
*/ */
so->numKilled = 0; so->numKilled = 0;
} }
/*
* The following routines manage a shared-memory area in which we track
* assignment of "vacuum cycle IDs" to currently-active btree vacuuming
* operations. There is a single counter which increments each time we
* start a vacuum to assign it a cycle ID. Since multiple vacuums could
* be active concurrently, we have to track the cycle ID for each active
* vacuum; this requires at most MaxBackends entries (usually far fewer).
* We assume at most one vacuum can be active for a given index.
*
* Access to the shared memory area is controlled by BtreeVacuumLock.
* In principle we could use a separate lmgr locktag for each index,
* but a single LWLock is much cheaper, and given the short time that
* the lock is ever held, the concurrency hit should be minimal.
*/
typedef struct BTOneVacInfo
{
LockRelId relid; /* global identifier of an index */
BTCycleId cycleid; /* cycle ID for its active VACUUM */
} BTOneVacInfo;
typedef struct BTVacInfo
{
BTCycleId cycle_ctr; /* cycle ID most recently assigned */
int num_vacuums; /* number of currently active VACUUMs */
int max_vacuums; /* allocated length of vacuums[] array */
BTOneVacInfo vacuums[1]; /* VARIABLE LENGTH ARRAY */
} BTVacInfo;
static BTVacInfo *btvacinfo;
/*
* _bt_vacuum_cycleid --- get the active vacuum cycle ID for an index,
* or zero if there is no active VACUUM
*
* Note: for correct interlocking, the caller must already hold pin and
* exclusive lock on each buffer it will store the cycle ID into. This
* ensures that even if a VACUUM starts immediately afterwards, it cannot
* process those pages until the page split is complete.
*/
BTCycleId
_bt_vacuum_cycleid(Relation rel)
{
BTCycleId result = 0;
int i;
/* Share lock is enough since this is a read-only operation */
LWLockAcquire(BtreeVacuumLock, LW_SHARED);
for (i = 0; i < btvacinfo->num_vacuums; i++)
{
BTOneVacInfo *vac = &btvacinfo->vacuums[i];
if (vac->relid.relId == rel->rd_lockInfo.lockRelId.relId &&
vac->relid.dbId == rel->rd_lockInfo.lockRelId.dbId)
{
result = vac->cycleid;
break;
}
}
LWLockRelease(BtreeVacuumLock);
return result;
}
/*
* _bt_start_vacuum --- assign a cycle ID to a just-starting VACUUM operation
*
* Note: the caller must guarantee (via PG_TRY) that it will eventually call
* _bt_end_vacuum, else we'll permanently leak an array slot.
*/
BTCycleId
_bt_start_vacuum(Relation rel)
{
BTCycleId result;
int i;
BTOneVacInfo *vac;
LWLockAcquire(BtreeVacuumLock, LW_EXCLUSIVE);
/* Assign the next cycle ID, being careful to avoid zero */
do {
result = ++(btvacinfo->cycle_ctr);
} while (result == 0);
/* Let's just make sure there's no entry already for this index */
for (i = 0; i < btvacinfo->num_vacuums; i++)
{
vac = &btvacinfo->vacuums[i];
if (vac->relid.relId == rel->rd_lockInfo.lockRelId.relId &&
vac->relid.dbId == rel->rd_lockInfo.lockRelId.dbId)
elog(ERROR, "multiple active vacuums for index \"%s\"",
RelationGetRelationName(rel));
}
/* OK, add an entry */
if (btvacinfo->num_vacuums >= btvacinfo->max_vacuums)
elog(ERROR, "out of btvacinfo slots");
vac = &btvacinfo->vacuums[btvacinfo->num_vacuums];
vac->relid = rel->rd_lockInfo.lockRelId;
vac->cycleid = result;
btvacinfo->num_vacuums++;
LWLockRelease(BtreeVacuumLock);
return result;
}
/*
* _bt_end_vacuum --- mark a btree VACUUM operation as done
*
* Note: this is deliberately coded not to complain if no entry is found;
* this allows the caller to put PG_TRY around the start_vacuum operation.
*/
void
_bt_end_vacuum(Relation rel)
{
int i;
LWLockAcquire(BtreeVacuumLock, LW_EXCLUSIVE);
/* Find the array entry */
for (i = 0; i < btvacinfo->num_vacuums; i++)
{
BTOneVacInfo *vac = &btvacinfo->vacuums[i];
if (vac->relid.relId == rel->rd_lockInfo.lockRelId.relId &&
vac->relid.dbId == rel->rd_lockInfo.lockRelId.dbId)
{
/* Remove it by shifting down the last entry */
*vac = btvacinfo->vacuums[btvacinfo->num_vacuums - 1];
btvacinfo->num_vacuums--;
break;
}
}
LWLockRelease(BtreeVacuumLock);
}
/*
* BTreeShmemSize --- report amount of shared memory space needed
*/
Size
BTreeShmemSize(void)
{
Size size;
size = offsetof(BTVacInfo, vacuums[0]);
size = add_size(size, mul_size(MaxBackends, sizeof(BTOneVacInfo)));
return size;
}
/*
* BTreeShmemInit --- initialize this module's shared memory
*/
void
BTreeShmemInit(void)
{
bool found;
btvacinfo = (BTVacInfo *) ShmemInitStruct("BTree Vacuum State",
BTreeShmemSize(),
&found);
if (!IsUnderPostmaster)
{
/* Initialize shared memory area */
Assert(!found);
/*
* It doesn't really matter what the cycle counter starts at, but
* having it always start the same doesn't seem good. Seed with
* low-order bits of time() instead.
*/
btvacinfo->cycle_ctr = (BTCycleId) time(NULL);
btvacinfo->num_vacuums = 0;
btvacinfo->max_vacuums = MaxBackends;
}
else
Assert(found);
}
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/access/nbtree/nbtxlog.c,v 1.32 2006/04/13 03:53:05 tgl Exp $ * $PostgreSQL: pgsql/src/backend/access/nbtree/nbtxlog.c,v 1.33 2006/05/08 00:00:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -249,6 +249,7 @@ btree_xlog_split(bool onleft, bool isroot, ...@@ -249,6 +249,7 @@ btree_xlog_split(bool onleft, bool isroot,
pageop->btpo_next = rightsib; pageop->btpo_next = rightsib;
pageop->btpo.level = xlrec->level; pageop->btpo.level = xlrec->level;
pageop->btpo_flags = (xlrec->level == 0) ? BTP_LEAF : 0; pageop->btpo_flags = (xlrec->level == 0) ? BTP_LEAF : 0;
pageop->btpo_cycleid = 0;
_bt_restore_page(page, _bt_restore_page(page,
(char *) xlrec + SizeOfBtreeSplit, (char *) xlrec + SizeOfBtreeSplit,
...@@ -281,6 +282,7 @@ btree_xlog_split(bool onleft, bool isroot, ...@@ -281,6 +282,7 @@ btree_xlog_split(bool onleft, bool isroot,
pageop->btpo_next = xlrec->rightblk; pageop->btpo_next = xlrec->rightblk;
pageop->btpo.level = xlrec->level; pageop->btpo.level = xlrec->level;
pageop->btpo_flags = (xlrec->level == 0) ? BTP_LEAF : 0; pageop->btpo_flags = (xlrec->level == 0) ? BTP_LEAF : 0;
pageop->btpo_cycleid = 0;
_bt_restore_page(page, _bt_restore_page(page,
(char *) xlrec + SizeOfBtreeSplit + xlrec->leftlen, (char *) xlrec + SizeOfBtreeSplit + xlrec->leftlen,
...@@ -506,6 +508,7 @@ btree_xlog_delete_page(bool ismeta, ...@@ -506,6 +508,7 @@ btree_xlog_delete_page(bool ismeta,
pageop->btpo_next = rightsib; pageop->btpo_next = rightsib;
pageop->btpo.xact = FrozenTransactionId; pageop->btpo.xact = FrozenTransactionId;
pageop->btpo_flags = BTP_DELETED; pageop->btpo_flags = BTP_DELETED;
pageop->btpo_cycleid = 0;
PageSetLSN(page, lsn); PageSetLSN(page, lsn);
PageSetTLI(page, ThisTimeLineID); PageSetTLI(page, ThisTimeLineID);
...@@ -548,6 +551,7 @@ btree_xlog_newroot(XLogRecPtr lsn, XLogRecord *record) ...@@ -548,6 +551,7 @@ btree_xlog_newroot(XLogRecPtr lsn, XLogRecord *record)
pageop->btpo.level = xlrec->level; pageop->btpo.level = xlrec->level;
if (xlrec->level == 0) if (xlrec->level == 0)
pageop->btpo_flags |= BTP_LEAF; pageop->btpo_flags |= BTP_LEAF;
pageop->btpo_cycleid = 0;
if (record->xl_len > SizeOfBtreeNewroot) if (record->xl_len > SizeOfBtreeNewroot)
{ {
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/storage/ipc/ipci.c,v 1.82 2006/03/05 15:58:37 momjian Exp $ * $PostgreSQL: pgsql/src/backend/storage/ipc/ipci.c,v 1.83 2006/05/08 00:00:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
#include "access/clog.h" #include "access/clog.h"
#include "access/multixact.h" #include "access/multixact.h"
#include "access/nbtree.h"
#include "access/subtrans.h" #include "access/subtrans.h"
#include "access/twophase.h" #include "access/twophase.h"
#include "access/xlog.h" #include "access/xlog.h"
...@@ -88,6 +89,7 @@ CreateSharedMemoryAndSemaphores(bool makePrivate, int port) ...@@ -88,6 +89,7 @@ CreateSharedMemoryAndSemaphores(bool makePrivate, int port)
size = add_size(size, SInvalShmemSize()); size = add_size(size, SInvalShmemSize());
size = add_size(size, FreeSpaceShmemSize()); size = add_size(size, FreeSpaceShmemSize());
size = add_size(size, BgWriterShmemSize()); size = add_size(size, BgWriterShmemSize());
size = add_size(size, BTreeShmemSize());
#ifdef EXEC_BACKEND #ifdef EXEC_BACKEND
size = add_size(size, ShmemBackendArraySize()); size = add_size(size, ShmemBackendArraySize());
#endif #endif
...@@ -182,6 +184,11 @@ CreateSharedMemoryAndSemaphores(bool makePrivate, int port) ...@@ -182,6 +184,11 @@ CreateSharedMemoryAndSemaphores(bool makePrivate, int port)
PMSignalInit(); PMSignalInit();
BgWriterShmemInit(); BgWriterShmemInit();
/*
* Set up other modules that need some shared memory space
*/
BTreeShmemInit();
#ifdef EXEC_BACKEND #ifdef EXEC_BACKEND
/* /*
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/access/nbtree.h,v 1.97 2006/05/07 01:21:30 tgl Exp $ * $PostgreSQL: pgsql/src/include/access/nbtree.h,v 1.98 2006/05/08 00:00:10 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,10 @@ ...@@ -19,6 +19,10 @@
#include "access/sdir.h" #include "access/sdir.h"
#include "access/xlogutils.h" #include "access/xlogutils.h"
/* There's room for a 16-bit vacuum cycle ID in BTPageOpaqueData */
typedef uint16 BTCycleId;
/* /*
* BTPageOpaqueData -- At the end of every page, we store a pointer * BTPageOpaqueData -- At the end of every page, we store a pointer
* to both siblings in the tree. This is used to do forward/backward * to both siblings in the tree. This is used to do forward/backward
...@@ -31,6 +35,16 @@ ...@@ -31,6 +35,16 @@
* and status. If the page is deleted, we replace the level with the * and status. If the page is deleted, we replace the level with the
* next-transaction-ID value indicating when it is safe to reclaim the page. * next-transaction-ID value indicating when it is safe to reclaim the page.
* *
* We also store a "vacuum cycle ID". When a page is split while VACUUM is
* processing the index, a nonzero value associated with the VACUUM run is
* stored into both halves of the split page. (If VACUUM is not running,
* both pages receive zero cycleids.) This allows VACUUM to detect whether
* a page was split since it started, with a small probability of false match
* if the page was last split some exact multiple of 65536 VACUUMs ago.
* Also, during a split, the BTP_SPLIT_END flag is cleared in the left
* (original) page, and set in the right page, but only if the next page
* to its right has a different cycleid.
*
* NOTE: the BTP_LEAF flag bit is redundant since level==0 could be tested * NOTE: the BTP_LEAF flag bit is redundant since level==0 could be tested
* instead. * instead.
*/ */
...@@ -45,6 +59,7 @@ typedef struct BTPageOpaqueData ...@@ -45,6 +59,7 @@ typedef struct BTPageOpaqueData
TransactionId xact; /* next transaction ID, if deleted */ TransactionId xact; /* next transaction ID, if deleted */
} btpo; } btpo;
uint16 btpo_flags; /* flag bits, see below */ uint16 btpo_flags; /* flag bits, see below */
BTCycleId btpo_cycleid; /* vacuum cycle ID of latest split */
} BTPageOpaqueData; } BTPageOpaqueData;
typedef BTPageOpaqueData *BTPageOpaque; typedef BTPageOpaqueData *BTPageOpaque;
...@@ -55,6 +70,7 @@ typedef BTPageOpaqueData *BTPageOpaque; ...@@ -55,6 +70,7 @@ typedef BTPageOpaqueData *BTPageOpaque;
#define BTP_DELETED (1 << 2) /* page has been deleted from tree */ #define BTP_DELETED (1 << 2) /* page has been deleted from tree */
#define BTP_META (1 << 3) /* meta-page */ #define BTP_META (1 << 3) /* meta-page */
#define BTP_HALF_DEAD (1 << 4) /* empty, but still in tree */ #define BTP_HALF_DEAD (1 << 4) /* empty, but still in tree */
#define BTP_SPLIT_END (1 << 5) /* rightmost page of split group */
/* /*
...@@ -492,6 +508,11 @@ extern bool _bt_checkkeys(IndexScanDesc scan, ...@@ -492,6 +508,11 @@ extern bool _bt_checkkeys(IndexScanDesc scan,
Page page, OffsetNumber offnum, Page page, OffsetNumber offnum,
ScanDirection dir, bool *continuescan); ScanDirection dir, bool *continuescan);
extern void _bt_killitems(IndexScanDesc scan, bool haveLock); extern void _bt_killitems(IndexScanDesc scan, bool haveLock);
extern BTCycleId _bt_vacuum_cycleid(Relation rel);
extern BTCycleId _bt_start_vacuum(Relation rel);
extern void _bt_end_vacuum(Relation rel);
extern Size BTreeShmemSize(void);
extern void BTreeShmemInit(void);
/* /*
* prototypes for functions in nbtsort.c * prototypes for functions in nbtsort.c
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2006, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/storage/lwlock.h,v 1.27 2006/03/05 15:58:59 momjian Exp $ * $PostgreSQL: pgsql/src/include/storage/lwlock.h,v 1.28 2006/05/08 00:00:17 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -47,6 +47,7 @@ typedef enum LWLockId ...@@ -47,6 +47,7 @@ typedef enum LWLockId
BgWriterCommLock, BgWriterCommLock,
TwoPhaseStateLock, TwoPhaseStateLock,
TablespaceCreateLock, TablespaceCreateLock,
BtreeVacuumLock,
FirstLockMgrLock, /* must be last except for MaxDynamicLWLock */ FirstLockMgrLock, /* must be last except for MaxDynamicLWLock */
MaxDynamicLWLock = 1000000000 MaxDynamicLWLock = 1000000000
......
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