Commit 3d351d91 authored by Tom Lane's avatar Tom Lane

Redefine pg_class.reltuples to be -1 before the first VACUUM or ANALYZE.

Historically, we've considered the state with relpages and reltuples
both zero as indicating that we do not know the table's tuple density.
This is problematic because it's impossible to distinguish "never yet
vacuumed" from "vacuumed and seen to be empty".  In particular, a user
cannot use VACUUM or ANALYZE to override the planner's normal heuristic
that an empty table should not be believed to be empty because it is
probably about to get populated.  That heuristic is a good safety
measure, so I don't care to abandon it, but there should be a way to
override it if the table is indeed intended to stay empty.

Hence, represent the initial state of ignorance by setting reltuples
to -1 (relpages is still set to zero), and apply the minimum-ten-pages
heuristic only when reltuples is still -1.  If the table is empty,
VACUUM or ANALYZE (but not CREATE INDEX) will override that to
reltuples = relpages = 0, and then we'll plan on that basis.

This requires a bunch of fiddly little changes, but we can get rid of
some ugly kluges that were formerly needed to maintain the old definition.

One notable point is that FDWs' GetForeignRelSize methods will see
baserel->tuples = -1 when no ANALYZE has been done on the foreign table.
That seems like a net improvement, since those methods were formerly
also in the dark about what baserel->tuples = 0 really meant.  Still,
it is an API change.

I bumped catversion because code predating this change would get confused
by seeing reltuples = -1.

Discussion: https://postgr.es/m/F02298E0-6EF4-49A1-BCB6-C484794D9ACC@thebuild.com
parent 9511fb37
...@@ -996,7 +996,7 @@ estimate_size(PlannerInfo *root, RelOptInfo *baserel, ...@@ -996,7 +996,7 @@ estimate_size(PlannerInfo *root, RelOptInfo *baserel,
/* /*
* Estimate the number of tuples in the file. * Estimate the number of tuples in the file.
*/ */
if (baserel->pages > 0) if (baserel->tuples >= 0 && baserel->pages > 0)
{ {
/* /*
* We have # of pages and # of tuples from pg_class (that is, from a * We have # of pages and # of tuples from pg_class (that is, from a
......
...@@ -195,6 +195,9 @@ statapprox_heap(Relation rel, output_type *stat) ...@@ -195,6 +195,9 @@ statapprox_heap(Relation rel, output_type *stat)
stat->tuple_count = vac_estimate_reltuples(rel, nblocks, scanned, stat->tuple_count = vac_estimate_reltuples(rel, nblocks, scanned,
stat->tuple_count); stat->tuple_count);
/* It's not clear if we could get -1 here, but be safe. */
stat->tuple_count = Max(stat->tuple_count, 0);
/* /*
* Calculate percentages if the relation has one or more pages. * Calculate percentages if the relation has one or more pages.
*/ */
......
...@@ -692,15 +692,14 @@ postgresGetForeignRelSize(PlannerInfo *root, ...@@ -692,15 +692,14 @@ postgresGetForeignRelSize(PlannerInfo *root,
else else
{ {
/* /*
* If the foreign table has never been ANALYZEd, it will have relpages * If the foreign table has never been ANALYZEd, it will have
* and reltuples equal to zero, which most likely has nothing to do * reltuples < 0, meaning "unknown". We can't do much if we're not
* with reality. We can't do a whole lot about that if we're not
* allowed to consult the remote server, but we can use a hack similar * allowed to consult the remote server, but we can use a hack similar
* to plancat.c's treatment of empty relations: use a minimum size * to plancat.c's treatment of empty relations: use a minimum size
* estimate of 10 pages, and divide by the column-datatype-based width * estimate of 10 pages, and divide by the column-datatype-based width
* estimate to get the corresponding number of tuples. * estimate to get the corresponding number of tuples.
*/ */
if (baserel->pages == 0 && baserel->tuples == 0) if (baserel->tuples < 0)
{ {
baserel->pages = 10; baserel->pages = 10;
baserel->tuples = baserel->tuples =
......
...@@ -1977,6 +1977,10 @@ SCRAM-SHA-256$<replaceable>&lt;iteration count&gt;</replaceable>:<replaceable>&l ...@@ -1977,6 +1977,10 @@ SCRAM-SHA-256$<replaceable>&lt;iteration count&gt;</replaceable>:<replaceable>&l
the planner. It is updated by <command>VACUUM</command>, the planner. It is updated by <command>VACUUM</command>,
<command>ANALYZE</command>, and a few DDL commands such as <command>ANALYZE</command>, and a few DDL commands such as
<command>CREATE INDEX</command>. <command>CREATE INDEX</command>.
If the table has never yet been vacuumed or
analyzed, <structfield>reltuples</structfield>
contains <literal>-1</literal> indicating that the row count is
unknown.
</para></entry> </para></entry>
</row> </row>
......
...@@ -130,7 +130,8 @@ GetForeignRelSize(PlannerInfo *root, ...@@ -130,7 +130,8 @@ GetForeignRelSize(PlannerInfo *root,
(The initial value is (The initial value is
from <structname>pg_class</structname>.<structfield>reltuples</structfield> from <structname>pg_class</structname>.<structfield>reltuples</structfield>
which represents the total row count seen by the which represents the total row count seen by the
last <command>ANALYZE</command>.) last <command>ANALYZE</command>; it will be <literal>-1</literal> if
no <command>ANALYZE</command> has been done on this foreign table.)
</para> </para>
<para> <para>
......
...@@ -727,7 +727,7 @@ ginvacuumcleanup(IndexVacuumInfo *info, IndexBulkDeleteResult *stats) ...@@ -727,7 +727,7 @@ ginvacuumcleanup(IndexVacuumInfo *info, IndexBulkDeleteResult *stats)
* entries. This is bogus if the index is partial, but it's real hard to * entries. This is bogus if the index is partial, but it's real hard to
* tell how many distinct heap entries are referenced by a GIN index. * tell how many distinct heap entries are referenced by a GIN index.
*/ */
stats->num_index_tuples = info->num_heap_tuples; stats->num_index_tuples = Max(info->num_heap_tuples, 0);
stats->estimated_count = info->estimated_count; stats->estimated_count = info->estimated_count;
/* /*
......
...@@ -208,7 +208,8 @@ typedef struct LVShared ...@@ -208,7 +208,8 @@ typedef struct LVShared
* live tuples in the index vacuum case or the new live tuples in the * live tuples in the index vacuum case or the new live tuples in the
* index cleanup case. * index cleanup case.
* *
* estimated_count is true if reltuples is an estimated value. * estimated_count is true if reltuples is an estimated value. (Note that
* reltuples could be -1 in this case, indicating we have no idea.)
*/ */
double reltuples; double reltuples;
bool estimated_count; bool estimated_count;
...@@ -567,31 +568,19 @@ heap_vacuum_rel(Relation onerel, VacuumParams *params, ...@@ -567,31 +568,19 @@ heap_vacuum_rel(Relation onerel, VacuumParams *params,
/* /*
* Update statistics in pg_class. * Update statistics in pg_class.
* *
* A corner case here is that if we scanned no pages at all because every * In principle new_live_tuples could be -1 indicating that we (still)
* page is all-visible, we should not update relpages/reltuples, because * don't know the tuple count. In practice that probably can't happen,
* we have no new information to contribute. In particular this keeps us * since we'd surely have scanned some pages if the table is new and
* from replacing relpages=reltuples=0 (which means "unknown tuple * nonempty.
* density") with nonzero relpages and reltuples=0 (which means "zero
* tuple density") unless there's some actual evidence for the latter.
* *
* It's important that we use tupcount_pages and not scanned_pages for the * For safety, clamp relallvisible to be not more than what we're setting
* check described above; scanned_pages counts pages where we could not * relpages to.
* get cleanup lock, and which were processed only for frozenxid purposes.
*
* We do update relallvisible even in the corner case, since if the table
* is all-visible we'd definitely like to know that. But clamp the value
* to be not more than what we're setting relpages to.
* *
* Also, don't change relfrozenxid/relminmxid if we skipped any pages, * Also, don't change relfrozenxid/relminmxid if we skipped any pages,
* since then we don't know for certain that all tuples have a newer xmin. * since then we don't know for certain that all tuples have a newer xmin.
*/ */
new_rel_pages = vacrelstats->rel_pages; new_rel_pages = vacrelstats->rel_pages;
new_live_tuples = vacrelstats->new_live_tuples; new_live_tuples = vacrelstats->new_live_tuples;
if (vacrelstats->tupcount_pages == 0 && new_rel_pages > 0)
{
new_rel_pages = vacrelstats->old_rel_pages;
new_live_tuples = vacrelstats->old_live_tuples;
}
visibilitymap_count(onerel, &new_rel_allvisible, NULL); visibilitymap_count(onerel, &new_rel_allvisible, NULL);
if (new_rel_allvisible > new_rel_pages) if (new_rel_allvisible > new_rel_pages)
...@@ -612,7 +601,7 @@ heap_vacuum_rel(Relation onerel, VacuumParams *params, ...@@ -612,7 +601,7 @@ heap_vacuum_rel(Relation onerel, VacuumParams *params,
/* report results to the stats collector, too */ /* report results to the stats collector, too */
pgstat_report_vacuum(RelationGetRelid(onerel), pgstat_report_vacuum(RelationGetRelid(onerel),
onerel->rd_rel->relisshared, onerel->rd_rel->relisshared,
new_live_tuples, Max(new_live_tuples, 0),
vacrelstats->new_dead_tuples); vacrelstats->new_dead_tuples);
pgstat_progress_end_command(); pgstat_progress_end_command();
...@@ -1695,9 +1684,12 @@ lazy_scan_heap(Relation onerel, VacuumParams *params, LVRelStats *vacrelstats, ...@@ -1695,9 +1684,12 @@ lazy_scan_heap(Relation onerel, VacuumParams *params, LVRelStats *vacrelstats,
vacrelstats->tupcount_pages, vacrelstats->tupcount_pages,
live_tuples); live_tuples);
/* also compute total number of surviving heap entries */ /*
* Also compute the total number of surviving heap entries. In the
* (unlikely) scenario that new_live_tuples is -1, take it as zero.
*/
vacrelstats->new_rel_tuples = vacrelstats->new_rel_tuples =
vacrelstats->new_live_tuples + vacrelstats->new_dead_tuples; Max(vacrelstats->new_live_tuples, 0) + vacrelstats->new_dead_tuples;
/* /*
* Release any remaining pin on visibility map page. * Release any remaining pin on visibility map page.
...@@ -2434,7 +2426,7 @@ lazy_cleanup_all_indexes(Relation *Irel, IndexBulkDeleteResult **stats, ...@@ -2434,7 +2426,7 @@ lazy_cleanup_all_indexes(Relation *Irel, IndexBulkDeleteResult **stats,
* dead_tuples, and update running statistics. * dead_tuples, and update running statistics.
* *
* reltuples is the number of heap tuples to be passed to the * reltuples is the number of heap tuples to be passed to the
* bulkdelete callback. * bulkdelete callback. It's always assumed to be estimated.
*/ */
static void static void
lazy_vacuum_index(Relation indrel, IndexBulkDeleteResult **stats, lazy_vacuum_index(Relation indrel, IndexBulkDeleteResult **stats,
......
...@@ -853,6 +853,7 @@ _bt_vacuum_needs_cleanup(IndexVacuumInfo *info) ...@@ -853,6 +853,7 @@ _bt_vacuum_needs_cleanup(IndexVacuumInfo *info)
prev_num_heap_tuples = metad->btm_last_cleanup_num_heap_tuples; prev_num_heap_tuples = metad->btm_last_cleanup_num_heap_tuples;
if (cleanup_scale_factor <= 0 || if (cleanup_scale_factor <= 0 ||
info->num_heap_tuples < 0 ||
prev_num_heap_tuples <= 0 || prev_num_heap_tuples <= 0 ||
(info->num_heap_tuples - prev_num_heap_tuples) / (info->num_heap_tuples - prev_num_heap_tuples) /
prev_num_heap_tuples >= cleanup_scale_factor) prev_num_heap_tuples >= cleanup_scale_factor)
......
...@@ -701,18 +701,14 @@ table_block_relation_estimate_size(Relation rel, int32 *attr_widths, ...@@ -701,18 +701,14 @@ table_block_relation_estimate_size(Relation rel, int32 *attr_widths,
* doesn't happen instantaneously, and it won't happen at all for cases * doesn't happen instantaneously, and it won't happen at all for cases
* such as temporary tables.) * such as temporary tables.)
* *
* We approximate "never vacuumed" by "has relpages = 0", which means this * We test "never vacuumed" by seeing whether reltuples < 0.
* will also fire on genuinely empty relations. Not great, but
* fortunately that's a seldom-seen case in the real world, and it
* shouldn't degrade the quality of the plan too much anyway to err in
* this direction.
* *
* If the table has inheritance children, we don't apply this heuristic. * If the table has inheritance children, we don't apply this heuristic.
* Totally empty parent tables are quite common, so we should be willing * Totally empty parent tables are quite common, so we should be willing
* to believe that they are empty. * to believe that they are empty.
*/ */
if (curpages < 10 && if (curpages < 10 &&
relpages == 0 && reltuples < 0 &&
!rel->rd_rel->relhassubclass) !rel->rd_rel->relhassubclass)
curpages = 10; curpages = 10;
...@@ -727,17 +723,17 @@ table_block_relation_estimate_size(Relation rel, int32 *attr_widths, ...@@ -727,17 +723,17 @@ table_block_relation_estimate_size(Relation rel, int32 *attr_widths,
} }
/* estimate number of tuples from previous tuple density */ /* estimate number of tuples from previous tuple density */
if (relpages > 0) if (reltuples >= 0 && relpages > 0)
density = reltuples / (double) relpages; density = reltuples / (double) relpages;
else else
{ {
/* /*
* When we have no data because the relation was truncated, estimate * When we have no data because the relation was never yet vacuumed,
* tuple width from attribute datatypes. We assume here that the * estimate tuple width from attribute datatypes. We assume here that
* pages are completely full, which is OK for tables (since they've * the pages are completely full, which is OK for tables but is
* presumably not been VACUUMed yet) but is probably an overestimate * probably an overestimate for indexes. Fortunately
* for indexes. Fortunately get_relation_info() can clamp the * get_relation_info() can clamp the overestimate to the parent
* overestimate to the parent table's size. * table's size.
* *
* Note: this code intentionally disregards alignment considerations, * Note: this code intentionally disregards alignment considerations,
* because (a) that would be gilding the lily considering how crude * because (a) that would be gilding the lily considering how crude
......
...@@ -1015,7 +1015,7 @@ AddNewRelationTuple(Relation pg_class_desc, ...@@ -1015,7 +1015,7 @@ AddNewRelationTuple(Relation pg_class_desc,
case RELKIND_TOASTVALUE: case RELKIND_TOASTVALUE:
/* The relation is real, but as yet empty */ /* The relation is real, but as yet empty */
new_rel_reltup->relpages = 0; new_rel_reltup->relpages = 0;
new_rel_reltup->reltuples = 0; new_rel_reltup->reltuples = -1;
new_rel_reltup->relallvisible = 0; new_rel_reltup->relallvisible = 0;
break; break;
case RELKIND_SEQUENCE: case RELKIND_SEQUENCE:
...@@ -1027,7 +1027,7 @@ AddNewRelationTuple(Relation pg_class_desc, ...@@ -1027,7 +1027,7 @@ AddNewRelationTuple(Relation pg_class_desc,
default: default:
/* Views, etc, have no disk storage */ /* Views, etc, have no disk storage */
new_rel_reltup->relpages = 0; new_rel_reltup->relpages = 0;
new_rel_reltup->reltuples = 0; new_rel_reltup->reltuples = -1;
new_rel_reltup->relallvisible = 0; new_rel_reltup->relallvisible = 0;
break; break;
} }
......
...@@ -2722,6 +2722,15 @@ index_update_stats(Relation rel, ...@@ -2722,6 +2722,15 @@ index_update_stats(Relation rel,
/* Should this be a more comprehensive test? */ /* Should this be a more comprehensive test? */
Assert(rd_rel->relkind != RELKIND_PARTITIONED_INDEX); Assert(rd_rel->relkind != RELKIND_PARTITIONED_INDEX);
/*
* As a special hack, if we are dealing with an empty table and the
* existing reltuples is -1, we leave that alone. This ensures that
* creating an index as part of CREATE TABLE doesn't cause the table to
* prematurely look like it's been vacuumed.
*/
if (reltuples == 0 && rd_rel->reltuples < 0)
reltuples = -1;
/* Apply required updates, if any, to copied tuple */ /* Apply required updates, if any, to copied tuple */
dirty = false; dirty = false;
......
...@@ -1128,8 +1128,8 @@ vacuum_set_xid_limits(Relation rel, ...@@ -1128,8 +1128,8 @@ vacuum_set_xid_limits(Relation rel,
* live tuples seen; but if we did not, we should not blindly extrapolate * live tuples seen; but if we did not, we should not blindly extrapolate
* from that number, since VACUUM may have scanned a quite nonrandom * from that number, since VACUUM may have scanned a quite nonrandom
* subset of the table. When we have only partial information, we take * subset of the table. When we have only partial information, we take
* the old value of pg_class.reltuples as a measurement of the * the old value of pg_class.reltuples/pg_class.relpages as a measurement
* tuple density in the unscanned pages. * of the tuple density in the unscanned pages.
* *
* Note: scanned_tuples should count only *live* tuples, since * Note: scanned_tuples should count only *live* tuples, since
* pg_class.reltuples is defined that way. * pg_class.reltuples is defined that way.
...@@ -1152,18 +1152,16 @@ vac_estimate_reltuples(Relation relation, ...@@ -1152,18 +1152,16 @@ vac_estimate_reltuples(Relation relation,
/* /*
* If scanned_pages is zero but total_pages isn't, keep the existing value * If scanned_pages is zero but total_pages isn't, keep the existing value
* of reltuples. (Note: callers should avoid updating the pg_class * of reltuples. (Note: we might be returning -1 in this case.)
* statistics in this situation, since no new information has been
* provided.)
*/ */
if (scanned_pages == 0) if (scanned_pages == 0)
return old_rel_tuples; return old_rel_tuples;
/* /*
* If old value of relpages is zero, old density is indeterminate; we * If old density is unknown, we can't do much except scale up
* can't do much except scale up scanned_tuples to match total_pages. * scanned_tuples to match total_pages.
*/ */
if (old_rel_pages == 0) if (old_rel_tuples < 0 || old_rel_pages == 0)
return floor((scanned_tuples / scanned_pages) * total_pages + 0.5); return floor((scanned_tuples / scanned_pages) * total_pages + 0.5);
/* /*
......
...@@ -912,7 +912,11 @@ set_foreign_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte) ...@@ -912,7 +912,11 @@ set_foreign_size(PlannerInfo *root, RelOptInfo *rel, RangeTblEntry *rte)
/* ... but do not let it set the rows estimate to zero */ /* ... but do not let it set the rows estimate to zero */
rel->rows = clamp_row_est(rel->rows); rel->rows = clamp_row_est(rel->rows);
/* also, make sure rel->tuples is not insane relative to rel->rows */ /*
* Also, make sure rel->tuples is not insane relative to rel->rows.
* Notably, this ensures sanity if pg_class.reltuples contains -1 and the
* FDW doesn't do anything to replace that.
*/
rel->tuples = Max(rel->tuples, rel->rows); rel->tuples = Max(rel->tuples, rel->rows);
} }
......
...@@ -974,11 +974,6 @@ estimate_rel_size(Relation rel, int32 *attr_widths, ...@@ -974,11 +974,6 @@ estimate_rel_size(Relation rel, int32 *attr_widths,
/* it has storage, ok to call the smgr */ /* it has storage, ok to call the smgr */
curpages = RelationGetNumberOfBlocks(rel); curpages = RelationGetNumberOfBlocks(rel);
/* coerce values in pg_class to more desirable types */
relpages = (BlockNumber) rel->rd_rel->relpages;
reltuples = (double) rel->rd_rel->reltuples;
relallvisible = (BlockNumber) rel->rd_rel->relallvisible;
/* report estimated # pages */ /* report estimated # pages */
*pages = curpages; *pages = curpages;
/* quick exit if rel is clearly empty */ /* quick exit if rel is clearly empty */
...@@ -988,6 +983,7 @@ estimate_rel_size(Relation rel, int32 *attr_widths, ...@@ -988,6 +983,7 @@ estimate_rel_size(Relation rel, int32 *attr_widths,
*allvisfrac = 0; *allvisfrac = 0;
break; break;
} }
/* coerce values in pg_class to more desirable types */ /* coerce values in pg_class to more desirable types */
relpages = (BlockNumber) rel->rd_rel->relpages; relpages = (BlockNumber) rel->rd_rel->relpages;
reltuples = (double) rel->rd_rel->reltuples; reltuples = (double) rel->rd_rel->reltuples;
...@@ -1006,12 +1002,12 @@ estimate_rel_size(Relation rel, int32 *attr_widths, ...@@ -1006,12 +1002,12 @@ estimate_rel_size(Relation rel, int32 *attr_widths,
} }
/* estimate number of tuples from previous tuple density */ /* estimate number of tuples from previous tuple density */
if (relpages > 0) if (reltuples >= 0 && relpages > 0)
density = reltuples / (double) relpages; density = reltuples / (double) relpages;
else else
{ {
/* /*
* When we have no data because the relation was truncated, * If we have no data because the relation was never vacuumed,
* estimate tuple width from attribute datatypes. We assume * estimate tuple width from attribute datatypes. We assume
* here that the pages are completely full, which is OK for * here that the pages are completely full, which is OK for
* tables (since they've presumably not been VACUUMed yet) but * tables (since they've presumably not been VACUUMed yet) but
...@@ -1059,6 +1055,7 @@ estimate_rel_size(Relation rel, int32 *attr_widths, ...@@ -1059,6 +1055,7 @@ estimate_rel_size(Relation rel, int32 *attr_widths,
break; break;
case RELKIND_FOREIGN_TABLE: case RELKIND_FOREIGN_TABLE:
/* Just use whatever's in pg_class */ /* Just use whatever's in pg_class */
/* Note that FDW must cope if reltuples is -1! */
*pages = rel->rd_rel->relpages; *pages = rel->rd_rel->relpages;
*tuples = rel->rd_rel->reltuples; *tuples = rel->rd_rel->reltuples;
*allvisfrac = 0; *allvisfrac = 0;
......
...@@ -3080,6 +3080,10 @@ relation_needs_vacanalyze(Oid relid, ...@@ -3080,6 +3080,10 @@ relation_needs_vacanalyze(Oid relid,
instuples = tabentry->inserts_since_vacuum; instuples = tabentry->inserts_since_vacuum;
anltuples = tabentry->changes_since_analyze; anltuples = tabentry->changes_since_analyze;
/* If the table hasn't yet been vacuumed, take reltuples as zero */
if (reltuples < 0)
reltuples = 0;
vacthresh = (float4) vac_base_thresh + vac_scale_factor * reltuples; vacthresh = (float4) vac_base_thresh + vac_scale_factor * reltuples;
vacinsthresh = (float4) vac_ins_base_thresh + vac_ins_scale_factor * reltuples; vacinsthresh = (float4) vac_ins_base_thresh + vac_ins_scale_factor * reltuples;
anlthresh = (float4) anl_base_thresh + anl_scale_factor * reltuples; anlthresh = (float4) anl_base_thresh + anl_scale_factor * reltuples;
......
...@@ -621,7 +621,7 @@ DefineQueryRewrite(const char *rulename, ...@@ -621,7 +621,7 @@ DefineQueryRewrite(const char *rulename,
classForm->relam = InvalidOid; classForm->relam = InvalidOid;
classForm->reltablespace = InvalidOid; classForm->reltablespace = InvalidOid;
classForm->relpages = 0; classForm->relpages = 0;
classForm->reltuples = 0; classForm->reltuples = -1;
classForm->relallvisible = 0; classForm->relallvisible = 0;
classForm->reltoastrelid = InvalidOid; classForm->reltoastrelid = InvalidOid;
classForm->relhasindex = false; classForm->relhasindex = false;
......
...@@ -1870,7 +1870,7 @@ formrdesc(const char *relationName, Oid relationReltype, ...@@ -1870,7 +1870,7 @@ formrdesc(const char *relationName, Oid relationReltype,
relation->rd_rel->relreplident = REPLICA_IDENTITY_NOTHING; relation->rd_rel->relreplident = REPLICA_IDENTITY_NOTHING;
relation->rd_rel->relpages = 0; relation->rd_rel->relpages = 0;
relation->rd_rel->reltuples = 0; relation->rd_rel->reltuples = -1;
relation->rd_rel->relallvisible = 0; relation->rd_rel->relallvisible = 0;
relation->rd_rel->relkind = RELKIND_RELATION; relation->rd_rel->relkind = RELKIND_RELATION;
relation->rd_rel->relnatts = (int16) natts; relation->rd_rel->relnatts = (int16) natts;
...@@ -3692,7 +3692,7 @@ RelationSetNewRelfilenode(Relation relation, char persistence) ...@@ -3692,7 +3692,7 @@ RelationSetNewRelfilenode(Relation relation, char persistence)
if (relation->rd_rel->relkind != RELKIND_SEQUENCE) if (relation->rd_rel->relkind != RELKIND_SEQUENCE)
{ {
classform->relpages = 0; /* it's empty until further notice */ classform->relpages = 0; /* it's empty until further notice */
classform->reltuples = 0; classform->reltuples = -1;
classform->relallvisible = 0; classform->relallvisible = 0;
} }
classform->relfrozenxid = freezeXid; classform->relfrozenxid = freezeXid;
......
...@@ -38,8 +38,8 @@ typedef struct IndexBuildResult ...@@ -38,8 +38,8 @@ typedef struct IndexBuildResult
* *
* num_heap_tuples is accurate only when estimated_count is false; * num_heap_tuples is accurate only when estimated_count is false;
* otherwise it's just an estimate (currently, the estimate is the * otherwise it's just an estimate (currently, the estimate is the
* prior value of the relation's pg_class.reltuples field). It will * prior value of the relation's pg_class.reltuples field, so it could
* always just be an estimate during ambulkdelete. * even be -1). It will always just be an estimate during ambulkdelete.
*/ */
typedef struct IndexVacuumInfo typedef struct IndexVacuumInfo
{ {
......
...@@ -53,6 +53,6 @@ ...@@ -53,6 +53,6 @@
*/ */
/* yyyymmddN */ /* yyyymmddN */
#define CATALOG_VERSION_NO 202008261 #define CATALOG_VERSION_NO 202008301
#endif #endif
...@@ -62,8 +62,8 @@ CATALOG(pg_class,1259,RelationRelationId) BKI_BOOTSTRAP BKI_ROWTYPE_OID(83,Relat ...@@ -62,8 +62,8 @@ CATALOG(pg_class,1259,RelationRelationId) BKI_BOOTSTRAP BKI_ROWTYPE_OID(83,Relat
/* # of blocks (not always up-to-date) */ /* # of blocks (not always up-to-date) */
int32 relpages BKI_DEFAULT(0); int32 relpages BKI_DEFAULT(0);
/* # of tuples (not always up-to-date) */ /* # of tuples (not always up-to-date; -1 means "unknown") */
float4 reltuples BKI_DEFAULT(0); float4 reltuples BKI_DEFAULT(-1);
/* # of all-visible blocks (not always up-to-date) */ /* # of all-visible blocks (not always up-to-date) */
int32 relallvisible BKI_DEFAULT(0); int32 relallvisible BKI_DEFAULT(0);
......
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