Commit 0e69f705 authored by Alvaro Herrera's avatar Alvaro Herrera

Set pg_class.reltuples for partitioned tables

When commit 0827e8af added auto-analyze support for partitioned
tables, it included code to obtain reltuples for the partitioned table
as a number of catalog accesses to read pg_class.reltuples for each
partition.  That's not only very inefficient, but also problematic
because autovacuum doesn't hold any locks on any of those tables -- and
doesn't want to.  Replace that code with a read of pg_class.reltuples
for the partitioned table, and make sure ANALYZE and TRUNCATE properly
maintain that value.

I found no code that would be affected by the change of relpages from
zero to non-zero for partitioned tables, and no other code that should
be maintaining it, but if there is, hopefully it'll be an easy fix.

Per buildfarm.

Author: Álvaro Herrera <alvherre@alvh.no-ip.org>
Reviewed-by: default avatarZhihong Yu <zyu@yugabyte.com>
Discussion: https://postgr.es/m/1823909.1617862590@sss.pgh.pa.us
parent 1798d8f8
...@@ -656,6 +656,18 @@ do_analyze_rel(Relation onerel, VacuumParams *params, ...@@ -656,6 +656,18 @@ do_analyze_rel(Relation onerel, VacuumParams *params,
in_outer_xact); in_outer_xact);
} }
} }
else if (onerel->rd_rel->relkind == RELKIND_PARTITIONED_TABLE)
{
/*
* Partitioned tables don't have storage, so we don't set any fields in
* their pg_class entries except for relpages, which is necessary for
* auto-analyze to work properly.
*/
vac_update_relstats(onerel, -1, totalrows,
0, false, InvalidTransactionId,
InvalidMultiXactId,
in_outer_xact);
}
/* /*
* Now report ANALYZE to the stats collector. For regular tables, we do * Now report ANALYZE to the stats collector. For regular tables, we do
......
...@@ -337,6 +337,7 @@ typedef struct ForeignTruncateInfo ...@@ -337,6 +337,7 @@ typedef struct ForeignTruncateInfo
static void truncate_check_rel(Oid relid, Form_pg_class reltuple); static void truncate_check_rel(Oid relid, Form_pg_class reltuple);
static void truncate_check_perms(Oid relid, Form_pg_class reltuple); static void truncate_check_perms(Oid relid, Form_pg_class reltuple);
static void truncate_check_activity(Relation rel); static void truncate_check_activity(Relation rel);
static void truncate_update_partedrel_stats(List *parted_rels);
static void RangeVarCallbackForTruncate(const RangeVar *relation, static void RangeVarCallbackForTruncate(const RangeVar *relation,
Oid relId, Oid oldRelId, void *arg); Oid relId, Oid oldRelId, void *arg);
static List *MergeAttributes(List *schema, List *supers, char relpersistence, static List *MergeAttributes(List *schema, List *supers, char relpersistence,
...@@ -1755,6 +1756,7 @@ ExecuteTruncateGuts(List *explicit_rels, ...@@ -1755,6 +1756,7 @@ ExecuteTruncateGuts(List *explicit_rels,
{ {
List *rels; List *rels;
List *seq_relids = NIL; List *seq_relids = NIL;
List *parted_rels = NIL;
HTAB *ft_htab = NULL; HTAB *ft_htab = NULL;
EState *estate; EState *estate;
ResultRelInfo *resultRelInfos; ResultRelInfo *resultRelInfos;
...@@ -1908,9 +1910,15 @@ ExecuteTruncateGuts(List *explicit_rels, ...@@ -1908,9 +1910,15 @@ ExecuteTruncateGuts(List *explicit_rels,
Relation rel = (Relation) lfirst(lc1); Relation rel = (Relation) lfirst(lc1);
int extra = lfirst_int(lc2); int extra = lfirst_int(lc2);
/* Skip partitioned tables as there is nothing to do */ /*
* Save OID of partitioned tables for later; nothing else to do for
* them here.
*/
if (rel->rd_rel->relkind == RELKIND_PARTITIONED_TABLE) if (rel->rd_rel->relkind == RELKIND_PARTITIONED_TABLE)
{
parted_rels = lappend_oid(parted_rels, RelationGetRelid(rel));
continue; continue;
}
/* /*
* Build the lists of foreign tables belonging to each foreign server * Build the lists of foreign tables belonging to each foreign server
...@@ -2061,6 +2069,9 @@ ExecuteTruncateGuts(List *explicit_rels, ...@@ -2061,6 +2069,9 @@ ExecuteTruncateGuts(List *explicit_rels,
ResetSequence(seq_relid); ResetSequence(seq_relid);
} }
/* Reset partitioned tables' pg_class.reltuples */
truncate_update_partedrel_stats(parted_rels);
/* /*
* Write a WAL record to allow this set of actions to be logically * Write a WAL record to allow this set of actions to be logically
* decoded. * decoded.
...@@ -2207,6 +2218,40 @@ truncate_check_activity(Relation rel) ...@@ -2207,6 +2218,40 @@ truncate_check_activity(Relation rel)
CheckTableNotInUse(rel, "TRUNCATE"); CheckTableNotInUse(rel, "TRUNCATE");
} }
/*
* Update pg_class.reltuples for all the given partitioned tables to 0.
*/
static void
truncate_update_partedrel_stats(List *parted_rels)
{
Relation pg_class;
ListCell *lc;
pg_class = table_open(RelationRelationId, RowExclusiveLock);
foreach(lc, parted_rels)
{
Oid relid = lfirst_oid(lc);
HeapTuple tuple;
Form_pg_class rd_rel;
tuple = SearchSysCacheCopy1(RELOID, ObjectIdGetDatum(relid));
if (!HeapTupleIsValid(tuple))
elog(ERROR, "could not find tuple for relation %u", relid);
rd_rel = (Form_pg_class) GETSTRUCT(tuple);
if (rd_rel->reltuples != (float4) 0)
{
rd_rel->reltuples = (float4) 0;
heap_inplace_update(pg_class, tuple);
}
heap_freetuple(tuple);
}
table_close(pg_class, RowExclusiveLock);
}
/* /*
* storage_name * storage_name
* returns the name corresponding to a typstorage/attstorage enum value * returns the name corresponding to a typstorage/attstorage enum value
......
...@@ -3208,45 +3208,8 @@ relation_needs_vacanalyze(Oid relid, ...@@ -3208,45 +3208,8 @@ relation_needs_vacanalyze(Oid relid,
* that's not being forced. * that's not being forced.
*/ */
if (PointerIsValid(tabentry) && AutoVacuumingActive()) if (PointerIsValid(tabentry) && AutoVacuumingActive())
{
if (classForm->relkind != RELKIND_PARTITIONED_TABLE)
{ {
reltuples = classForm->reltuples; reltuples = classForm->reltuples;
}
else
{
/*
* If the relation is a partitioned table, we must add up
* children's reltuples.
*/
List *children;
ListCell *lc;
reltuples = 0;
/* Find all members of inheritance set taking AccessShareLock */
children = find_all_inheritors(relid, AccessShareLock, NULL);
foreach(lc, children)
{
Oid childOID = lfirst_oid(lc);
HeapTuple childtuple;
Form_pg_class childclass;
childtuple = SearchSysCache1(RELOID, ObjectIdGetDatum(childOID));
childclass = (Form_pg_class) GETSTRUCT(childtuple);
/* Skip a partitioned table and foreign partitions */
if (RELKIND_HAS_STORAGE(childclass->relkind))
{
/* Sum up the child's reltuples for its parent table */
reltuples += childclass->reltuples;
}
ReleaseSysCache(childtuple);
}
list_free(children);
}
vactuples = tabentry->n_dead_tuples; vactuples = tabentry->n_dead_tuples;
instuples = tabentry->inserts_since_vacuum; instuples = tabentry->inserts_since_vacuum;
anltuples = tabentry->changes_since_analyze; anltuples = tabentry->changes_since_analyze;
......
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