Commit 9f3665fb authored by Peter Geoghegan's avatar Peter Geoghegan

Don't consider newly inserted tuples in nbtree VACUUM.

Remove the entire idea of "stale stats" within nbtree VACUUM (stop
caring about stats involving the number of inserted tuples).  Also
remove the vacuum_cleanup_index_scale_factor GUC/param on the master
branch (though just disable them on postgres 13).

The vacuum_cleanup_index_scale_factor/stats interface made the nbtree AM
partially responsible for deciding when pg_class.reltuples stats needed
to be updated.  This seems contrary to the spirit of the index AM API,
though -- it is not actually necessary for an index AM's bulk delete and
cleanup callbacks to provide accurate stats when it happens to be
inconvenient.  The core code owns that.  (Index AMs have the authority
to perform or not perform certain kinds of deferred cleanup based on
their own considerations, such as page deletion and recycling, but that
has little to do with pg_class.reltuples/num_index_tuples.)

This issue was fairly harmless until the introduction of the
autovacuum_vacuum_insert_threshold feature by commit b07642db, which had
an undesirable interaction with the vacuum_cleanup_index_scale_factor
mechanism: it made insert-driven autovacuums perform full index scans,
even though there is no real benefit to doing so.  This has been tied to
a regression with an append-only insert benchmark [1].

Also have remaining cases that perform a full scan of an index during a
cleanup-only nbtree VACUUM indicate that the final tuple count is only
an estimate.  This prevents vacuumlazy.c from setting the index's
pg_class.reltuples in those cases (it will now only update pg_class when
vacuumlazy.c had TIDs for nbtree to bulk delete).  This arguably fixes
an oversight in deduplication-related bugfix commit 48e12913.

[1] https://smalldatum.blogspot.com/2021/01/insert-benchmark-postgres-is-still.html

Author: Peter Geoghegan <pg@bowt.ie>
Reviewed-By: default avatarMasahiko Sawada <sawada.mshk@gmail.com>
Discussion: https://postgr.es/m/CAD21AoA4WHthN5uU6+WScZ7+J_RcEjmcuH94qcoUPuB42ShXzg@mail.gmail.com
Backpatch: 13-, where autovacuum_vacuum_insert_threshold was added.
parent 845ac7f8
...@@ -8544,46 +8544,6 @@ COPY postgres_log FROM '/full/path/to/logfile.csv' WITH csv; ...@@ -8544,46 +8544,6 @@ COPY postgres_log FROM '/full/path/to/logfile.csv' WITH csv;
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry id="guc-vacuum-cleanup-index-scale-factor" xreflabel="vacuum_cleanup_index_scale_factor">
<term><varname>vacuum_cleanup_index_scale_factor</varname> (<type>floating point</type>)
<indexterm>
<primary><varname>vacuum_cleanup_index_scale_factor</varname></primary>
<secondary>configuration parameter</secondary>
</indexterm>
</term>
<listitem>
<para>
Specifies the fraction of the total number of heap tuples counted in
the previous statistics collection that can be inserted without
incurring an index scan at the <command>VACUUM</command> cleanup stage.
This setting currently applies to B-tree indexes only.
</para>
<para>
If no tuples were deleted from the heap, B-tree indexes are still
scanned at the <command>VACUUM</command> cleanup stage when the
index's statistics are stale. Index statistics are considered
stale if the number of newly inserted tuples exceeds the
<varname>vacuum_cleanup_index_scale_factor</varname>
fraction of the total number of heap tuples detected by the previous
statistics collection. The total number of heap tuples is stored in
the index meta-page. Note that the meta-page does not include this data
until <command>VACUUM</command> finds no dead tuples, so B-tree index
scan at the cleanup stage can only be skipped if the second and
subsequent <command>VACUUM</command> cycles detect no dead tuples.
</para>
<para>
The value can range from <literal>0</literal> to
<literal>10000000000</literal>.
When <varname>vacuum_cleanup_index_scale_factor</varname> is set to
<literal>0</literal>, index scans are never skipped during
<command>VACUUM</command> cleanup. The default value is <literal>0.1</literal>.
</para>
</listitem>
</varlistentry>
<varlistentry id="guc-bytea-output" xreflabel="bytea_output"> <varlistentry id="guc-bytea-output" xreflabel="bytea_output">
<term><varname>bytea_output</varname> (<type>enum</type>) <term><varname>bytea_output</varname> (<type>enum</type>)
<indexterm> <indexterm>
......
...@@ -456,20 +456,6 @@ CREATE [ UNIQUE ] INDEX [ CONCURRENTLY ] [ [ IF NOT EXISTS ] <replaceable class= ...@@ -456,20 +456,6 @@ CREATE [ UNIQUE ] INDEX [ CONCURRENTLY ] [ [ IF NOT EXISTS ] <replaceable class=
</note> </note>
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry id="index-reloption-vacuum-cleanup-index-scale-factor" xreflabel="vacuum_cleanup_index_scale_factor">
<term><literal>vacuum_cleanup_index_scale_factor</literal> (<type>floating point</type>)
<indexterm>
<primary><varname>vacuum_cleanup_index_scale_factor</varname></primary>
<secondary>storage parameter</secondary>
</indexterm>
</term>
<listitem>
<para>
Per-index value for <xref linkend="guc-vacuum-cleanup-index-scale-factor"/>.
</para>
</listitem>
</varlistentry>
</variablelist> </variablelist>
<para> <para>
......
...@@ -461,15 +461,6 @@ static relopt_real realRelOpts[] = ...@@ -461,15 +461,6 @@ static relopt_real realRelOpts[] =
}, },
0, -1.0, DBL_MAX 0, -1.0, DBL_MAX
}, },
{
{
"vacuum_cleanup_index_scale_factor",
"Number of tuple inserts prior to index cleanup as a fraction of reltuples.",
RELOPT_KIND_BTREE,
ShareUpdateExclusiveLock
},
-1, 0.0, 1e10
},
/* list terminator */ /* list terminator */
{{NULL}} {{NULL}}
}; };
......
...@@ -1332,8 +1332,6 @@ _bt_insertonpg(Relation rel, ...@@ -1332,8 +1332,6 @@ _bt_insertonpg(Relation rel,
xlmeta.fastroot = metad->btm_fastroot; xlmeta.fastroot = metad->btm_fastroot;
xlmeta.fastlevel = metad->btm_fastlevel; xlmeta.fastlevel = metad->btm_fastlevel;
xlmeta.last_cleanup_num_delpages = metad->btm_last_cleanup_num_delpages; xlmeta.last_cleanup_num_delpages = metad->btm_last_cleanup_num_delpages;
xlmeta.last_cleanup_num_heap_tuples =
metad->btm_last_cleanup_num_heap_tuples;
xlmeta.allequalimage = metad->btm_allequalimage; xlmeta.allequalimage = metad->btm_allequalimage;
XLogRegisterBuffer(2, metabuf, XLogRegisterBuffer(2, metabuf,
...@@ -2549,7 +2547,6 @@ _bt_newroot(Relation rel, Buffer lbuf, Buffer rbuf) ...@@ -2549,7 +2547,6 @@ _bt_newroot(Relation rel, Buffer lbuf, Buffer rbuf)
md.fastroot = rootblknum; md.fastroot = rootblknum;
md.fastlevel = metad->btm_level; md.fastlevel = metad->btm_level;
md.last_cleanup_num_delpages = metad->btm_last_cleanup_num_delpages; md.last_cleanup_num_delpages = metad->btm_last_cleanup_num_delpages;
md.last_cleanup_num_heap_tuples = metad->btm_last_cleanup_num_heap_tuples;
md.allequalimage = metad->btm_allequalimage; md.allequalimage = metad->btm_allequalimage;
XLogRegisterBufData(2, (char *) &md, sizeof(xl_btree_metadata)); XLogRegisterBufData(2, (char *) &md, sizeof(xl_btree_metadata));
......
...@@ -175,26 +175,15 @@ _bt_getmeta(Relation rel, Buffer metabuf) ...@@ -175,26 +175,15 @@ _bt_getmeta(Relation rel, Buffer metabuf)
* _bt_vacuum_needs_cleanup() to decide whether or not a btvacuumscan() * _bt_vacuum_needs_cleanup() to decide whether or not a btvacuumscan()
* call should go ahead for an entire VACUUM operation. * call should go ahead for an entire VACUUM operation.
* *
* See btvacuumcleanup() and _bt_vacuum_needs_cleanup() for details of * See btvacuumcleanup() and _bt_vacuum_needs_cleanup() for the
* the two fields that we maintain here. * definition of num_delpages.
*
* The information that we maintain for btvacuumcleanup() describes the
* state of the index (as well as the table it indexes) just _after_ the
* ongoing VACUUM operation. The next _bt_vacuum_needs_cleanup() call
* will consider the information we saved for it during the next VACUUM
* operation (assuming that there will be no btbulkdelete() call during
* the next VACUUM operation -- if there is then the question of skipping
* btvacuumscan() doesn't even arise).
*/ */
void void
_bt_set_cleanup_info(Relation rel, BlockNumber num_delpages, _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages)
float8 num_heap_tuples)
{ {
Buffer metabuf; Buffer metabuf;
Page metapg; Page metapg;
BTMetaPageData *metad; BTMetaPageData *metad;
bool rewrite = false;
XLogRecPtr recptr;
/* /*
* On-disk compatibility note: The btm_last_cleanup_num_delpages metapage * On-disk compatibility note: The btm_last_cleanup_num_delpages metapage
...@@ -209,21 +198,20 @@ _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages, ...@@ -209,21 +198,20 @@ _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages,
* in reality there are only one or two. The worst that can happen is * in reality there are only one or two. The worst that can happen is
* that there will be a call to btvacuumscan a little earlier, which will * that there will be a call to btvacuumscan a little earlier, which will
* set btm_last_cleanup_num_delpages to a sane value when we're called. * set btm_last_cleanup_num_delpages to a sane value when we're called.
*
* Note also that the metapage's btm_last_cleanup_num_heap_tuples field is
* no longer used as of PostgreSQL 14. We set it to -1.0 on rewrite, just
* to be consistent.
*/ */
metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ); metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ);
metapg = BufferGetPage(metabuf); metapg = BufferGetPage(metabuf);
metad = BTPageGetMeta(metapg); metad = BTPageGetMeta(metapg);
/* Always dynamically upgrade index/metapage when BTREE_MIN_VERSION */ /* Don't miss chance to upgrade index/metapage when BTREE_MIN_VERSION */
if (metad->btm_version < BTREE_NOVAC_VERSION) if (metad->btm_version >= BTREE_NOVAC_VERSION &&
rewrite = true; metad->btm_last_cleanup_num_delpages == num_delpages)
else if (metad->btm_last_cleanup_num_delpages != num_delpages)
rewrite = true;
else if (metad->btm_last_cleanup_num_heap_tuples != num_heap_tuples)
rewrite = true;
if (!rewrite)
{ {
/* Usually means index continues to have num_delpages of 0 */
_bt_relbuf(rel, metabuf); _bt_relbuf(rel, metabuf);
return; return;
} }
...@@ -240,13 +228,14 @@ _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages, ...@@ -240,13 +228,14 @@ _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages,
/* update cleanup-related information */ /* update cleanup-related information */
metad->btm_last_cleanup_num_delpages = num_delpages; metad->btm_last_cleanup_num_delpages = num_delpages;
metad->btm_last_cleanup_num_heap_tuples = num_heap_tuples; metad->btm_last_cleanup_num_heap_tuples = -1.0;
MarkBufferDirty(metabuf); MarkBufferDirty(metabuf);
/* write wal record if needed */ /* write wal record if needed */
if (RelationNeedsWAL(rel)) if (RelationNeedsWAL(rel))
{ {
xl_btree_metadata md; xl_btree_metadata md;
XLogRecPtr recptr;
XLogBeginInsert(); XLogBeginInsert();
XLogRegisterBuffer(0, metabuf, REGBUF_WILL_INIT | REGBUF_STANDARD); XLogRegisterBuffer(0, metabuf, REGBUF_WILL_INIT | REGBUF_STANDARD);
...@@ -258,7 +247,6 @@ _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages, ...@@ -258,7 +247,6 @@ _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages,
md.fastroot = metad->btm_fastroot; md.fastroot = metad->btm_fastroot;
md.fastlevel = metad->btm_fastlevel; md.fastlevel = metad->btm_fastlevel;
md.last_cleanup_num_delpages = num_delpages; md.last_cleanup_num_delpages = num_delpages;
md.last_cleanup_num_heap_tuples = num_heap_tuples;
md.allequalimage = metad->btm_allequalimage; md.allequalimage = metad->btm_allequalimage;
XLogRegisterBufData(0, (char *) &md, sizeof(xl_btree_metadata)); XLogRegisterBufData(0, (char *) &md, sizeof(xl_btree_metadata));
...@@ -443,7 +431,6 @@ _bt_getroot(Relation rel, int access) ...@@ -443,7 +431,6 @@ _bt_getroot(Relation rel, int access)
md.fastroot = rootblkno; md.fastroot = rootblkno;
md.fastlevel = 0; md.fastlevel = 0;
md.last_cleanup_num_delpages = 0; md.last_cleanup_num_delpages = 0;
md.last_cleanup_num_heap_tuples = -1.0;
md.allequalimage = metad->btm_allequalimage; md.allequalimage = metad->btm_allequalimage;
XLogRegisterBufData(2, (char *) &md, sizeof(xl_btree_metadata)); XLogRegisterBufData(2, (char *) &md, sizeof(xl_btree_metadata));
...@@ -2632,7 +2619,6 @@ _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, BlockNumber scanblkno, ...@@ -2632,7 +2619,6 @@ _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, BlockNumber scanblkno,
xlmeta.fastroot = metad->btm_fastroot; xlmeta.fastroot = metad->btm_fastroot;
xlmeta.fastlevel = metad->btm_fastlevel; xlmeta.fastlevel = metad->btm_fastlevel;
xlmeta.last_cleanup_num_delpages = metad->btm_last_cleanup_num_delpages; xlmeta.last_cleanup_num_delpages = metad->btm_last_cleanup_num_delpages;
xlmeta.last_cleanup_num_heap_tuples = metad->btm_last_cleanup_num_heap_tuples;
xlmeta.allequalimage = metad->btm_allequalimage; xlmeta.allequalimage = metad->btm_allequalimage;
XLogRegisterBufData(4, (char *) &xlmeta, sizeof(xl_btree_metadata)); XLogRegisterBufData(4, (char *) &xlmeta, sizeof(xl_btree_metadata));
......
...@@ -789,11 +789,8 @@ _bt_vacuum_needs_cleanup(IndexVacuumInfo *info) ...@@ -789,11 +789,8 @@ _bt_vacuum_needs_cleanup(IndexVacuumInfo *info)
Buffer metabuf; Buffer metabuf;
Page metapg; Page metapg;
BTMetaPageData *metad; BTMetaPageData *metad;
BTOptions *relopts;
float8 cleanup_scale_factor;
uint32 btm_version; uint32 btm_version;
BlockNumber prev_num_delpages; BlockNumber prev_num_delpages;
float8 prev_num_heap_tuples;
/* /*
* Copy details from metapage to local variables quickly. * Copy details from metapage to local variables quickly.
...@@ -816,32 +813,8 @@ _bt_vacuum_needs_cleanup(IndexVacuumInfo *info) ...@@ -816,32 +813,8 @@ _bt_vacuum_needs_cleanup(IndexVacuumInfo *info)
} }
prev_num_delpages = metad->btm_last_cleanup_num_delpages; prev_num_delpages = metad->btm_last_cleanup_num_delpages;
prev_num_heap_tuples = metad->btm_last_cleanup_num_heap_tuples;
_bt_relbuf(info->index, metabuf); _bt_relbuf(info->index, metabuf);
/*
* If the underlying table has received a sufficiently high number of
* insertions since the last VACUUM operation that called btvacuumscan(),
* then have the current VACUUM operation call btvacuumscan() now. This
* happens when the statistics are deemed stale.
*
* XXX: We should have a more principled way of determining what
* "staleness" means. The vacuum_cleanup_index_scale_factor GUC (and the
* index-level storage param) seem hard to tune in a principled way.
*/
relopts = (BTOptions *) info->index->rd_options;
cleanup_scale_factor = (relopts &&
relopts->vacuum_cleanup_index_scale_factor >= 0)
? relopts->vacuum_cleanup_index_scale_factor
: vacuum_cleanup_index_scale_factor;
if (cleanup_scale_factor <= 0 ||
info->num_heap_tuples < 0 ||
prev_num_heap_tuples <= 0 ||
(info->num_heap_tuples - prev_num_heap_tuples) /
prev_num_heap_tuples >= cleanup_scale_factor)
return true;
/* /*
* Trigger cleanup in rare cases where prev_num_delpages exceeds 5% of the * Trigger cleanup in rare cases where prev_num_delpages exceeds 5% of the
* total size of the index. We can reasonably expect (though are not * total size of the index. We can reasonably expect (though are not
...@@ -925,48 +898,45 @@ btvacuumcleanup(IndexVacuumInfo *info, IndexBulkDeleteResult *stats) ...@@ -925,48 +898,45 @@ btvacuumcleanup(IndexVacuumInfo *info, IndexBulkDeleteResult *stats)
/* /*
* Since we aren't going to actually delete any leaf items, there's no * Since we aren't going to actually delete any leaf items, there's no
* need to go through all the vacuum-cycle-ID pushups here * need to go through all the vacuum-cycle-ID pushups here.
*
* Posting list tuples are a source of inaccuracy for cleanup-only
* scans. btvacuumscan() will assume that the number of index tuples
* from each page can be used as num_index_tuples, even though
* num_index_tuples is supposed to represent the number of TIDs in the
* index. This naive approach can underestimate the number of tuples
* in the index significantly.
*
* We handle the problem by making num_index_tuples an estimate in
* cleanup-only case.
*/ */
stats = (IndexBulkDeleteResult *) palloc0(sizeof(IndexBulkDeleteResult)); stats = (IndexBulkDeleteResult *) palloc0(sizeof(IndexBulkDeleteResult));
btvacuumscan(info, stats, NULL, NULL, 0); btvacuumscan(info, stats, NULL, NULL, 0);
stats->estimated_count = true;
} }
/* /*
* By here, we know for sure that this VACUUM operation won't be skipping * By here, we know for sure that this VACUUM operation won't be skipping
* its btvacuumscan() call. Maintain the count of the current number of * its btvacuumscan() call. Maintain num_delpages value in metapage.
* heap tuples in the metapage. Also maintain the num_delpages value.
* This information will be used by _bt_vacuum_needs_cleanup() during * This information will be used by _bt_vacuum_needs_cleanup() during
* future VACUUM operations that don't need to call btbulkdelete(). * future VACUUM operations that don't need to call btbulkdelete().
* *
* num_delpages is the number of deleted pages now in the index that were * num_delpages is the number of deleted pages now in the index that were
* not safe to place in the FSM to be recycled just yet. We expect that * not safe to place in the FSM to be recycled just yet. We expect that
* it will almost certainly be possible to place all of these pages in the * it will almost certainly be possible to place all of these pages in the
* FSM during the next VACUUM operation. That factor alone might cause * FSM during the next VACUUM operation. _bt_vacuum_needs_cleanup() will
* _bt_vacuum_needs_cleanup() to force the next VACUUM to proceed with a * force the next VACUUM to consider this before allowing btvacuumscan()
* btvacuumscan() call. * to be skipped entirely.
*
* Note: We must delay the _bt_set_cleanup_info() call until this late
* stage of VACUUM (the btvacuumcleanup() phase), to keep num_heap_tuples
* accurate. The btbulkdelete()-time num_heap_tuples value is generally
* just pg_class.reltuples for the heap relation _before_ VACUUM began.
* In general cleanup info should describe the state of the index/table
* _after_ VACUUM finishes.
*/ */
Assert(stats->pages_deleted >= stats->pages_free); Assert(stats->pages_deleted >= stats->pages_free);
num_delpages = stats->pages_deleted - stats->pages_free; num_delpages = stats->pages_deleted - stats->pages_free;
_bt_set_cleanup_info(info->index, num_delpages, info->num_heap_tuples); _bt_set_cleanup_info(info->index, num_delpages);
/* /*
* It's quite possible for us to be fooled by concurrent page splits into * It's quite possible for us to be fooled by concurrent page splits into
* double-counting some index tuples, so disbelieve any total that exceeds * double-counting some index tuples, so disbelieve any total that exceeds
* the underlying heap's count ... if we know that accurately. Otherwise * the underlying heap's count ... if we know that accurately. Otherwise
* this might just make matters worse. * this might just make matters worse.
*
* Posting list tuples are another source of inaccuracy. Cleanup-only
* btvacuumscan calls assume that the number of index tuples can be used
* as num_index_tuples, even though num_index_tuples is supposed to
* represent the number of TIDs in the index. This naive approach can
* underestimate the number of tuples in the index.
*/ */
if (!info->estimated_count) if (!info->estimated_count)
{ {
...@@ -1016,7 +986,6 @@ btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats, ...@@ -1016,7 +986,6 @@ btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats,
* pages in the index at the end of the VACUUM command.) * pages in the index at the end of the VACUUM command.)
*/ */
stats->num_pages = 0; stats->num_pages = 0;
stats->estimated_count = false;
stats->num_index_tuples = 0; stats->num_index_tuples = 0;
stats->pages_deleted = 0; stats->pages_deleted = 0;
stats->pages_free = 0; stats->pages_free = 0;
...@@ -1421,7 +1390,10 @@ backtrack: ...@@ -1421,7 +1390,10 @@ backtrack:
* We don't count the number of live TIDs during cleanup-only calls to * We don't count the number of live TIDs during cleanup-only calls to
* btvacuumscan (i.e. when callback is not set). We count the number * btvacuumscan (i.e. when callback is not set). We count the number
* of index tuples directly instead. This avoids the expense of * of index tuples directly instead. This avoids the expense of
* directly examining all of the tuples on each page. * directly examining all of the tuples on each page. VACUUM will
* treat num_index_tuples as an estimate in cleanup-only case, so it
* doesn't matter that this underestimates num_index_tuples
* significantly in some cases.
*/ */
if (minoff > maxoff) if (minoff > maxoff)
attempt_pagedel = (blkno == scanblkno); attempt_pagedel = (blkno == scanblkno);
......
...@@ -2105,8 +2105,6 @@ btoptions(Datum reloptions, bool validate) ...@@ -2105,8 +2105,6 @@ btoptions(Datum reloptions, bool validate)
{ {
static const relopt_parse_elt tab[] = { static const relopt_parse_elt tab[] = {
{"fillfactor", RELOPT_TYPE_INT, offsetof(BTOptions, fillfactor)}, {"fillfactor", RELOPT_TYPE_INT, offsetof(BTOptions, fillfactor)},
{"vacuum_cleanup_index_scale_factor", RELOPT_TYPE_REAL,
offsetof(BTOptions, vacuum_cleanup_index_scale_factor)},
{"deduplicate_items", RELOPT_TYPE_BOOL, {"deduplicate_items", RELOPT_TYPE_BOOL,
offsetof(BTOptions, deduplicate_items)} offsetof(BTOptions, deduplicate_items)}
......
...@@ -113,7 +113,7 @@ _bt_restore_meta(XLogReaderState *record, uint8 block_id) ...@@ -113,7 +113,7 @@ _bt_restore_meta(XLogReaderState *record, uint8 block_id)
/* Cannot log BTREE_MIN_VERSION index metapage without upgrade */ /* Cannot log BTREE_MIN_VERSION index metapage without upgrade */
Assert(md->btm_version >= BTREE_NOVAC_VERSION); Assert(md->btm_version >= BTREE_NOVAC_VERSION);
md->btm_last_cleanup_num_delpages = xlrec->last_cleanup_num_delpages; md->btm_last_cleanup_num_delpages = xlrec->last_cleanup_num_delpages;
md->btm_last_cleanup_num_heap_tuples = xlrec->last_cleanup_num_heap_tuples; md->btm_last_cleanup_num_heap_tuples = -1.0;
md->btm_allequalimage = xlrec->allequalimage; md->btm_allequalimage = xlrec->allequalimage;
pageop = (BTPageOpaque) PageGetSpecialPointer(metapg); pageop = (BTPageOpaque) PageGetSpecialPointer(metapg);
......
...@@ -113,9 +113,8 @@ btree_desc(StringInfo buf, XLogReaderState *record) ...@@ -113,9 +113,8 @@ btree_desc(StringInfo buf, XLogReaderState *record)
xlrec = (xl_btree_metadata *) XLogRecGetBlockData(record, 0, xlrec = (xl_btree_metadata *) XLogRecGetBlockData(record, 0,
NULL); NULL);
appendStringInfo(buf, "last_cleanup_num_delpages %u; last_cleanup_num_heap_tuples: %f", appendStringInfo(buf, "last_cleanup_num_delpages %u",
xlrec->last_cleanup_num_delpages, xlrec->last_cleanup_num_delpages);
xlrec->last_cleanup_num_heap_tuples);
break; break;
} }
} }
......
...@@ -148,5 +148,3 @@ int64 VacuumPageDirty = 0; ...@@ -148,5 +148,3 @@ int64 VacuumPageDirty = 0;
int VacuumCostBalance = 0; /* working state for vacuum */ int VacuumCostBalance = 0; /* working state for vacuum */
bool VacuumCostActive = false; bool VacuumCostActive = false;
double vacuum_cleanup_index_scale_factor;
...@@ -3703,16 +3703,6 @@ static struct config_real ConfigureNamesReal[] = ...@@ -3703,16 +3703,6 @@ static struct config_real ConfigureNamesReal[] =
NULL, NULL, NULL NULL, NULL, NULL
}, },
{
{"vacuum_cleanup_index_scale_factor", PGC_USERSET, CLIENT_CONN_STATEMENT,
gettext_noop("Number of tuple inserts prior to index cleanup as a fraction of reltuples."),
NULL
},
&vacuum_cleanup_index_scale_factor,
0.1, 0.0, 1e10,
NULL, NULL, NULL
},
{ {
{"log_statement_sample_rate", PGC_SUSET, LOGGING_WHEN, {"log_statement_sample_rate", PGC_SUSET, LOGGING_WHEN,
gettext_noop("Fraction of statements exceeding log_min_duration_sample to be logged."), gettext_noop("Fraction of statements exceeding log_min_duration_sample to be logged."),
......
...@@ -672,9 +672,6 @@ ...@@ -672,9 +672,6 @@
#vacuum_freeze_table_age = 150000000 #vacuum_freeze_table_age = 150000000
#vacuum_multixact_freeze_min_age = 5000000 #vacuum_multixact_freeze_min_age = 5000000
#vacuum_multixact_freeze_table_age = 150000000 #vacuum_multixact_freeze_table_age = 150000000
#vacuum_cleanup_index_scale_factor = 0.1 # fraction of total number of tuples
# before index cleanup, 0 always performs
# index cleanup
#bytea_output = 'hex' # hex, escape #bytea_output = 'hex' # hex, escape
#xmlbinary = 'base64' #xmlbinary = 'base64'
#xmloption = 'content' #xmloption = 'content'
......
...@@ -1789,14 +1789,14 @@ psql_completion(const char *text, int start, int end) ...@@ -1789,14 +1789,14 @@ psql_completion(const char *text, int start, int end)
/* ALTER INDEX <foo> SET|RESET ( */ /* ALTER INDEX <foo> SET|RESET ( */
else if (Matches("ALTER", "INDEX", MatchAny, "RESET", "(")) else if (Matches("ALTER", "INDEX", MatchAny, "RESET", "("))
COMPLETE_WITH("fillfactor", COMPLETE_WITH("fillfactor",
"vacuum_cleanup_index_scale_factor", "deduplicate_items", /* BTREE */ "deduplicate_items", /* BTREE */
"fastupdate", "gin_pending_list_limit", /* GIN */ "fastupdate", "gin_pending_list_limit", /* GIN */
"buffering", /* GiST */ "buffering", /* GiST */
"pages_per_range", "autosummarize" /* BRIN */ "pages_per_range", "autosummarize" /* BRIN */
); );
else if (Matches("ALTER", "INDEX", MatchAny, "SET", "(")) else if (Matches("ALTER", "INDEX", MatchAny, "SET", "("))
COMPLETE_WITH("fillfactor =", COMPLETE_WITH("fillfactor =",
"vacuum_cleanup_index_scale_factor =", "deduplicate_items =", /* BTREE */ "deduplicate_items =", /* BTREE */
"fastupdate =", "gin_pending_list_limit =", /* GIN */ "fastupdate =", "gin_pending_list_limit =", /* GIN */
"buffering =", /* GiST */ "buffering =", /* GiST */
"pages_per_range =", "autosummarize =" /* BRIN */ "pages_per_range =", "autosummarize =" /* BRIN */
......
...@@ -110,7 +110,7 @@ typedef struct BTMetaPageData ...@@ -110,7 +110,7 @@ typedef struct BTMetaPageData
/* number of deleted, non-recyclable pages during last cleanup */ /* number of deleted, non-recyclable pages during last cleanup */
uint32 btm_last_cleanup_num_delpages; uint32 btm_last_cleanup_num_delpages;
/* number of heap tuples during last cleanup */ /* number of heap tuples during last cleanup (deprecated) */
float8 btm_last_cleanup_num_heap_tuples; float8 btm_last_cleanup_num_heap_tuples;
bool btm_allequalimage; /* are all columns "equalimage"? */ bool btm_allequalimage; /* are all columns "equalimage"? */
...@@ -1067,8 +1067,6 @@ typedef struct BTOptions ...@@ -1067,8 +1067,6 @@ typedef struct BTOptions
{ {
int32 varlena_header_; /* varlena header (do not touch directly!) */ int32 varlena_header_; /* varlena header (do not touch directly!) */
int fillfactor; /* page fill factor in percent (0..100) */ int fillfactor; /* page fill factor in percent (0..100) */
/* fraction of newly inserted tuples needed to trigger index cleanup */
float8 vacuum_cleanup_index_scale_factor;
bool deduplicate_items; /* Try to deduplicate items? */ bool deduplicate_items; /* Try to deduplicate items? */
} BTOptions; } BTOptions;
...@@ -1171,8 +1169,7 @@ extern OffsetNumber _bt_findsplitloc(Relation rel, Page origpage, ...@@ -1171,8 +1169,7 @@ extern OffsetNumber _bt_findsplitloc(Relation rel, Page origpage,
*/ */
extern void _bt_initmetapage(Page page, BlockNumber rootbknum, uint32 level, extern void _bt_initmetapage(Page page, BlockNumber rootbknum, uint32 level,
bool allequalimage); bool allequalimage);
extern void _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages, extern void _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages);
float8 num_heap_tuples);
extern void _bt_upgrademetapage(Page page); extern void _bt_upgrademetapage(Page page);
extern Buffer _bt_getroot(Relation rel, int access); extern Buffer _bt_getroot(Relation rel, int access);
extern Buffer _bt_gettrueroot(Relation rel); extern Buffer _bt_gettrueroot(Relation rel);
......
...@@ -54,7 +54,6 @@ typedef struct xl_btree_metadata ...@@ -54,7 +54,6 @@ typedef struct xl_btree_metadata
BlockNumber fastroot; BlockNumber fastroot;
uint32 fastlevel; uint32 fastlevel;
uint32 last_cleanup_num_delpages; uint32 last_cleanup_num_delpages;
float8 last_cleanup_num_heap_tuples;
bool allequalimage; bool allequalimage;
} xl_btree_metadata; } xl_btree_metadata;
......
...@@ -31,7 +31,7 @@ ...@@ -31,7 +31,7 @@
/* /*
* Each page of XLOG file has a header like this: * Each page of XLOG file has a header like this:
*/ */
#define XLOG_PAGE_MAGIC 0xD10A /* can be used as WAL version indicator */ #define XLOG_PAGE_MAGIC 0xD10B /* can be used as WAL version indicator */
typedef struct XLogPageHeaderData typedef struct XLogPageHeaderData
{ {
......
...@@ -261,8 +261,6 @@ extern int64 VacuumPageDirty; ...@@ -261,8 +261,6 @@ extern int64 VacuumPageDirty;
extern int VacuumCostBalance; extern int VacuumCostBalance;
extern bool VacuumCostActive; extern bool VacuumCostActive;
extern double vacuum_cleanup_index_scale_factor;
/* in tcop/postgres.c */ /* in tcop/postgres.c */
......
...@@ -308,35 +308,6 @@ alter table btree_tall_tbl alter COLUMN t set storage plain; ...@@ -308,35 +308,6 @@ alter table btree_tall_tbl alter COLUMN t set storage plain;
create index btree_tall_idx on btree_tall_tbl (t, id) with (fillfactor = 10); create index btree_tall_idx on btree_tall_tbl (t, id) with (fillfactor = 10);
insert into btree_tall_tbl select g, repeat('x', 250) insert into btree_tall_tbl select g, repeat('x', 250)
from generate_series(1, 130) g; from generate_series(1, 130) g;
--
-- Test vacuum_cleanup_index_scale_factor
--
-- Simple create
create table btree_test(a int);
create index btree_idx1 on btree_test(a) with (vacuum_cleanup_index_scale_factor = 40.0);
select reloptions from pg_class WHERE oid = 'btree_idx1'::regclass;
reloptions
------------------------------------------
{vacuum_cleanup_index_scale_factor=40.0}
(1 row)
-- Fail while setting improper values
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = -10.0);
ERROR: value -10.0 out of bounds for option "vacuum_cleanup_index_scale_factor"
DETAIL: Valid values are between "0.000000" and "10000000000.000000".
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = 100.0);
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = 'string');
ERROR: invalid value for floating point option "vacuum_cleanup_index_scale_factor": string
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = true);
ERROR: invalid value for floating point option "vacuum_cleanup_index_scale_factor": true
-- Simple ALTER INDEX
alter index btree_idx1 set (vacuum_cleanup_index_scale_factor = 70.0);
select reloptions from pg_class WHERE oid = 'btree_idx1'::regclass;
reloptions
------------------------------------------
{vacuum_cleanup_index_scale_factor=70.0}
(1 row)
-- --
-- Test for multilevel page deletion -- Test for multilevel page deletion
-- --
......
...@@ -150,25 +150,6 @@ create index btree_tall_idx on btree_tall_tbl (t, id) with (fillfactor = 10); ...@@ -150,25 +150,6 @@ create index btree_tall_idx on btree_tall_tbl (t, id) with (fillfactor = 10);
insert into btree_tall_tbl select g, repeat('x', 250) insert into btree_tall_tbl select g, repeat('x', 250)
from generate_series(1, 130) g; from generate_series(1, 130) g;
--
-- Test vacuum_cleanup_index_scale_factor
--
-- Simple create
create table btree_test(a int);
create index btree_idx1 on btree_test(a) with (vacuum_cleanup_index_scale_factor = 40.0);
select reloptions from pg_class WHERE oid = 'btree_idx1'::regclass;
-- Fail while setting improper values
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = -10.0);
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = 100.0);
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = 'string');
create index btree_idx_err on btree_test(a) with (vacuum_cleanup_index_scale_factor = true);
-- Simple ALTER INDEX
alter index btree_idx1 set (vacuum_cleanup_index_scale_factor = 70.0);
select reloptions from pg_class WHERE oid = 'btree_idx1'::regclass;
-- --
-- Test for multilevel page deletion -- Test for multilevel page deletion
-- --
......
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