1
0
mirror of https://github.com/postgres/postgres.git synced 2025-10-24 01:29:19 +03:00

Move I/O before the index_update_stats() buffer lock region.

Commit a07e03fd8f enlarged the work done
here under the pg_class heap buffer lock.  Two preexisting actions are
best done before holding that lock.  Both RelationGetNumberOfBlocks()
and visibilitymap_count() do I/O, and the latter might exclusive-lock a
visibility map buffer.  Moving these reduces contention and risk of
undetected LWLock deadlock.  Back-patch to v12, like that commit.

Discussion: https://postgr.es/m/20241031200139.b4@rfd.leadboat.com
This commit is contained in:
Noah Misch
2024-11-02 09:04:55 -07:00
parent fe8091c9e3
commit 6b01cac0be

View File

@@ -2788,6 +2788,9 @@ index_update_stats(Relation rel,
bool hasindex, bool hasindex,
double reltuples) double reltuples)
{ {
bool update_stats;
BlockNumber relpages;
BlockNumber relallvisible;
Oid relid = RelationGetRelid(rel); Oid relid = RelationGetRelid(rel);
Relation pg_class; Relation pg_class;
ScanKeyData key[1]; ScanKeyData key[1];
@@ -2796,6 +2799,38 @@ index_update_stats(Relation rel,
Form_pg_class rd_rel; Form_pg_class rd_rel;
bool dirty; bool dirty;
/*
* 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. The rd_rel we modify may
* differ from rel->rd_rel due to e.g. commit of concurrent GRANT, but the
* commands that change reltuples take locks conflicting with ours. (Even
* if a command changed reltuples under a weaker lock, this affects only
* statistics for an empty table.)
*/
if (reltuples == 0 && rel->rd_rel->reltuples < 0)
reltuples = -1;
update_stats = reltuples >= 0;
/*
* Finish I/O and visibility map buffer locks before
* systable_inplace_update_begin() locks the pg_class buffer. The rd_rel
* we modify may differ from rel->rd_rel due to e.g. commit of concurrent
* GRANT, but no command changes a relkind from non-index to index. (Even
* if one did, relallvisible doesn't break functionality.)
*/
if (update_stats)
{
relpages = RelationGetNumberOfBlocks(rel);
if (rel->rd_rel->relkind != RELKIND_INDEX)
visibilitymap_count(rel, &relallvisible, NULL);
else /* don't bother for indexes */
relallvisible = 0;
}
/* /*
* We always update the pg_class row using a non-transactional, * We always update the pg_class row using a non-transactional,
* overwrite-in-place update. There are several reasons for this: * overwrite-in-place update. There are several reasons for this:
@@ -2849,16 +2884,8 @@ index_update_stats(Relation rel,
dirty = true; dirty = true;
} }
if (reltuples >= 0) if (update_stats)
{ {
BlockNumber relpages = RelationGetNumberOfBlocks(rel);
BlockNumber relallvisible;
if (rd_rel->relkind != RELKIND_INDEX)
visibilitymap_count(rel, &relallvisible, NULL);
else /* don't bother for indexes */
relallvisible = 0;
if (rd_rel->relpages != (int32) relpages) if (rd_rel->relpages != (int32) relpages)
{ {
rd_rel->relpages = (int32) relpages; rd_rel->relpages = (int32) relpages;