Commit ab0dfc96 authored by Alvaro Herrera's avatar Alvaro Herrera

Report progress of CREATE INDEX operations

This uses the progress reporting infrastructure added by c16dc1ac,
adding support for CREATE INDEX and CREATE INDEX CONCURRENTLY.

There are two pieces to this: one is index-AM-agnostic, and the other is
AM-specific.  The latter is fairly elaborate for btrees, including
reportage for parallel index builds and the separate phases that btree
index creation uses; other index AMs, which are much simpler in their
building procedures, have simplistic reporting only, but that seems
sufficient, at least for non-concurrent builds.

The index-AM-agnostic part is fairly complete, providing insight into
the CONCURRENTLY wait phases as well as block-based progress during the
index validation table scan.  (The index validation index scan requires
patching each AM, which has not been included here.)

Reviewers: Rahila Syed, Pavan Deolasee, Tatsuro Yamada
Discussion: https://postgr.es/m/20181220220022.mg63bhk26zdpvmcj@alvherre.pgsql
parent 4d0e994e
...@@ -566,7 +566,7 @@ bt_check_every_level(Relation rel, Relation heaprel, bool heapkeyspace, ...@@ -566,7 +566,7 @@ bt_check_every_level(Relation rel, Relation heaprel, bool heapkeyspace,
RelationGetRelationName(state->rel), RelationGetRelationName(state->rel),
RelationGetRelationName(state->heaprel)); RelationGetRelationName(state->heaprel));
table_index_build_scan(state->heaprel, state->rel, indexinfo, true, table_index_build_scan(state->heaprel, state->rel, indexinfo, true, false,
bt_tuple_present_callback, (void *) state, scan); bt_tuple_present_callback, (void *) state, scan);
ereport(DEBUG1, ereport(DEBUG1,
......
...@@ -142,7 +142,7 @@ blbuild(Relation heap, Relation index, IndexInfo *indexInfo) ...@@ -142,7 +142,7 @@ blbuild(Relation heap, Relation index, IndexInfo *indexInfo)
initCachedPage(&buildstate); initCachedPage(&buildstate);
/* Do the heap scan */ /* Do the heap scan */
reltuples = table_index_build_scan(heap, index, indexInfo, true, reltuples = table_index_build_scan(heap, index, indexInfo, true, true,
bloomBuildCallback, (void *) &buildstate, bloomBuildCallback, (void *) &buildstate,
NULL); NULL);
......
...@@ -132,6 +132,7 @@ blhandler(PG_FUNCTION_ARGS) ...@@ -132,6 +132,7 @@ blhandler(PG_FUNCTION_ARGS)
amroutine->amcostestimate = blcostestimate; amroutine->amcostestimate = blcostestimate;
amroutine->amoptions = bloptions; amroutine->amoptions = bloptions;
amroutine->amproperty = NULL; amroutine->amproperty = NULL;
amroutine->ambuildphasename = NULL;
amroutine->amvalidate = blvalidate; amroutine->amvalidate = blvalidate;
amroutine->ambeginscan = blbeginscan; amroutine->ambeginscan = blbeginscan;
amroutine->amrescan = blrescan; amroutine->amrescan = blrescan;
......
...@@ -127,6 +127,7 @@ typedef struct IndexAmRoutine ...@@ -127,6 +127,7 @@ typedef struct IndexAmRoutine
amcostestimate_function amcostestimate; amcostestimate_function amcostestimate;
amoptions_function amoptions; amoptions_function amoptions;
amproperty_function amproperty; /* can be NULL */ amproperty_function amproperty; /* can be NULL */
ambuildphasename_function ambuildphasename; /* can be NULL */
amvalidate_function amvalidate; amvalidate_function amvalidate;
ambeginscan_function ambeginscan; ambeginscan_function ambeginscan;
amrescan_function amrescan; amrescan_function amrescan;
...@@ -468,6 +469,18 @@ amproperty (Oid index_oid, int attno, ...@@ -468,6 +469,18 @@ amproperty (Oid index_oid, int attno,
<para> <para>
<programlisting> <programlisting>
char *
ambuildphasename (int64 phasenum);
</programlisting>
Return the textual name of the given build phase number.
The phase numbers are those reported during an index build via the
<function>pgstat_progress_update_param</function> interface.
The phase names are then exposed in the
<structname>pg_stat_progress_create_index</structname> view.
</para>
<para>
<programlisting>
bool bool
amvalidate (Oid opclassoid); amvalidate (Oid opclassoid);
</programlisting> </programlisting>
......
...@@ -336,6 +336,14 @@ postgres 27093 0.0 0.0 30096 2752 ? Ss 11:34 0:00 postgres: ser ...@@ -336,6 +336,14 @@ postgres 27093 0.0 0.0 30096 2752 ? Ss 11:34 0:00 postgres: ser
</entry> </entry>
</row> </row>
<row>
<entry><structname>pg_stat_progress_create_index</structname><indexterm><primary>pg_stat_progress_create_index</primary></indexterm></entry>
<entry>One row for each backend running <command>CREATE INDEX</command>, showing
current progress.
See <xref linkend='create-index-progress-reporting'/>.
</entry>
</row>
<row> <row>
<entry><structname>pg_stat_progress_vacuum</structname><indexterm><primary>pg_stat_progress_vacuum</primary></indexterm></entry> <entry><structname>pg_stat_progress_vacuum</structname><indexterm><primary>pg_stat_progress_vacuum</primary></indexterm></entry>
<entry>One row for each backend (including autovacuum worker processes) running <entry>One row for each backend (including autovacuum worker processes) running
...@@ -3403,10 +3411,224 @@ SELECT pg_stat_get_backend_pid(s.backendid) AS pid, ...@@ -3403,10 +3411,224 @@ SELECT pg_stat_get_backend_pid(s.backendid) AS pid,
<para> <para>
<productname>PostgreSQL</productname> has the ability to report the progress of <productname>PostgreSQL</productname> has the ability to report the progress of
certain commands during command execution. Currently, the only commands certain commands during command execution. Currently, the only commands
which support progress reporting are <command>VACUUM</command> and which support progress reporting are <command>CREATE INDEX</command>,
<command>VACUUM</command> and
<command>CLUSTER</command>. This may be expanded in the future. <command>CLUSTER</command>. This may be expanded in the future.
</para> </para>
<sect2 id="create-index-progress-reporting">
<title>CREATE INDEX Progress Reporting</title>
<para>
Whenever <command>CREATE INDEX</command> is running, the
<structname>pg_stat_progress_create_index</structname> view will contain
one row for each backend that is currently creating indexes. The tables
below describe the information that will be reported and provide information
about how to interpret it.
</para>
<table id="pg-stat-progress-create-index-view" xreflabel="pg_stat_progress_create_index">
<title><structname>pg_stat_progress_create_index</structname> View</title>
<tgroup cols="3">
<thead>
<row>
<entry>Column</entry>
<entry>Type</entry>
<entry>Description</entry>
</row>
</thead>
<tbody>
<row>
<entry><structfield>pid</structfield></entry>
<entry><type>integer</type></entry>
<entry>Process ID of backend.</entry>
</row>
<row>
<entry><structfield>datid</structfield></entry>
<entry><type>oid</type></entry>
<entry>OID of the database to which this backend is connected.</entry>
</row>
<row>
<entry><structfield>datname</structfield></entry>
<entry><type>name</type></entry>
<entry>Name of the database to which this backend is connected.</entry>
</row>
<row>
<entry><structfield>relid</structfield></entry>
<entry><type>oid</type></entry>
<entry>OID of the table on which the index is being created.</entry>
</row>
<row>
<entry><structfield>phase</structfield></entry>
<entry><type>text</type></entry>
<entry>
Current processing phase of index creation. See <xref linkend='create-index-phases'/>.
</entry>
</row>
<row>
<entry><structfield>lockers_total</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
Total number of lockers to wait for, when applicable.
</entry>
</row>
<row>
<entry><structfield>lockers_done</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
Number of lockers already waited for.
</entry>
</row>
<row>
<entry><structfield>current_locked_pid</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
Process ID of the locker currently being waited for.
</entry>
</row>
<row>
<entry><structfield>blocks_total</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
Total number of blocks to be processed in the current phase.
</entry>
</row>
<row>
<entry><structfield>blocks_done</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
Number of blocks already processed in the current phase.
</entry>
</row>
<row>
<entry><structfield>tuples_total</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
Total number of tuples to be processed in the current phase.
</entry>
</row>
<row>
<entry><structfield>tuples_done</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
Number of tuples already processed in the current phase.
</entry>
</row>
<row>
<entry><structfield>partitions_total</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
When creating an index on a partitioned table, this column is set to
the total number of partitions on which the index is to be created.
</entry>
</row>
<row>
<entry><structfield>partitions_done</structfield></entry>
<entry><type>bigint</type></entry>
<entry>
When creating an index on a partitioned table, this column is set to
the number of partitions on which the index has been completed.
</entry>
</row>
</tbody>
</tgroup>
</table>
<table id="create-index-phases">
<title>CREATE INDEX phases</title>
<tgroup cols="2">
<thead>
<row>
<entry>Phase</entry>
<entry>Description</entry>
</row>
</thead>
<tbody>
<row>
<entry><literal>initializing</literal></entry>
<entry>
<command>CREATE INDEX</command> is preparing to create the index. This
phase is expected to be very brief.
</entry>
</row>
<row>
<entry><literal>waiting for old snapshots</literal></entry>
<entry>
<command>CREATE INDEX CONCURRENTLY</command> is waiting for transactions
that can potentially see the table to release their snapshots.
This phase is skipped when not in concurrent mode.
Columns <structname>lockers_total</structname>, <structname>lockers_done</structname>
and <structname>current_locker_pid</structname> contain the progress
information for this phase.
</entry>
</row>
<row>
<entry><literal>building index</literal></entry>
<entry>
The index is being built by the access method-specific code. In this phase,
access methods that support progress reporting fill in their own progress data,
and the subphase is indicated in this column. Typically,
<structname>blocks_total</structname> and <structname>blocks_done</structname>
will contain progress data, as well as potentially
<structname>tuples_total</structname> and <structname>tuples_done</structname>.
</entry>
</row>
<row>
<entry><literal>waiting for writer snapshots</literal></entry>
<entry>
<command>CREATE INDEX CONCURRENTLY</command> is waiting for transactions
that can potentially write into the table to release their snapshots.
This phase is skipped when not in concurrent mode.
Columns <structname>lockers_total</structname>, <structname>lockers_done</structname>
and <structname>current_locker_pid</structname> contain the progress
information for this phase.
</entry>
</row>
<row>
<entry><literal>index validation: scanning index</literal></entry>
<entry>
<command>CREATE INDEX CONCURRENTLY</command> is scanning the index searching
for tuples that need to be validated.
This phase is skipped when not in concurrent mode.
Columns <structname>blocks_total</structname> (set to the total size of the index)
and <structname>blocks_done</structname> contain the progress information for this phase.
</entry>
</row>
<row>
<entry><literal>index validation: sorting tuples</literal></entry>
<entry>
<command>CREATE INDEX CONCURRENTLY</command> is sorting the output of the
index scanning phase.
</entry>
</row>
<row>
<entry><literal>index validation: scanning table</literal></entry>
<entry>
<command>CREATE INDEX CONCURRENTLY</command> is scanning the table
to validate the index tuples collected in the previous two phases.
This phase is skipped when not in concurrent mode.
Columns <structname>blocks_total</structname> (set to the total size of the table)
and <structname>blocks_done</structname> contain the progress information for this phase.
</entry>
</row>
<row>
<entry><literal>waiting for reader snapshots</literal></entry>
<entry>
<command>CREATE INDEX CONCURRENTLY</command> is waiting for transactions
that can potentially see the table to release their snapshots. This
phase is skipped when not in concurrent mode.
Columns <structname>lockers_total</structname>, <structname>lockers_done</structname>
and <structname>current_locker_pid</structname> contain the progress
information for this phase.
</entry>
</row>
</tbody>
</tgroup>
</table>
</sect2>
<sect2 id="vacuum-progress-reporting"> <sect2 id="vacuum-progress-reporting">
<title>VACUUM Progress Reporting</title> <title>VACUUM Progress Reporting</title>
......
...@@ -112,6 +112,7 @@ brinhandler(PG_FUNCTION_ARGS) ...@@ -112,6 +112,7 @@ brinhandler(PG_FUNCTION_ARGS)
amroutine->amcostestimate = brincostestimate; amroutine->amcostestimate = brincostestimate;
amroutine->amoptions = brinoptions; amroutine->amoptions = brinoptions;
amroutine->amproperty = NULL; amroutine->amproperty = NULL;
amroutine->ambuildphasename = NULL;
amroutine->amvalidate = brinvalidate; amroutine->amvalidate = brinvalidate;
amroutine->ambeginscan = brinbeginscan; amroutine->ambeginscan = brinbeginscan;
amroutine->amrescan = brinrescan; amroutine->amrescan = brinrescan;
...@@ -719,7 +720,7 @@ brinbuild(Relation heap, Relation index, IndexInfo *indexInfo) ...@@ -719,7 +720,7 @@ brinbuild(Relation heap, Relation index, IndexInfo *indexInfo)
* Now scan the relation. No syncscan allowed here because we want the * Now scan the relation. No syncscan allowed here because we want the
* heap blocks in physical order. * heap blocks in physical order.
*/ */
reltuples = table_index_build_scan(heap, index, indexInfo, false, reltuples = table_index_build_scan(heap, index, indexInfo, false, true,
brinbuildCallback, (void *) state, NULL); brinbuildCallback, (void *) state, NULL);
/* process the final batch */ /* process the final batch */
...@@ -1236,7 +1237,7 @@ summarize_range(IndexInfo *indexInfo, BrinBuildState *state, Relation heapRel, ...@@ -1236,7 +1237,7 @@ summarize_range(IndexInfo *indexInfo, BrinBuildState *state, Relation heapRel,
* cases. * cases.
*/ */
state->bs_currRangeStart = heapBlk; state->bs_currRangeStart = heapBlk;
table_index_build_range_scan(heapRel, state->bs_irel, indexInfo, false, true, table_index_build_range_scan(heapRel, state->bs_irel, indexInfo, false, true, false,
heapBlk, scanNumBlks, heapBlk, scanNumBlks,
brinbuildCallback, (void *) state, NULL); brinbuildCallback, (void *) state, NULL);
......
...@@ -395,7 +395,7 @@ ginbuild(Relation heap, Relation index, IndexInfo *indexInfo) ...@@ -395,7 +395,7 @@ ginbuild(Relation heap, Relation index, IndexInfo *indexInfo)
* Do the heap scan. We disallow sync scan here because dataPlaceToPage * Do the heap scan. We disallow sync scan here because dataPlaceToPage
* prefers to receive tuples in TID order. * prefers to receive tuples in TID order.
*/ */
reltuples = table_index_build_scan(heap, index, indexInfo, false, reltuples = table_index_build_scan(heap, index, indexInfo, false, true,
ginBuildCallback, (void *) &buildstate, ginBuildCallback, (void *) &buildstate,
NULL); NULL);
......
...@@ -64,6 +64,7 @@ ginhandler(PG_FUNCTION_ARGS) ...@@ -64,6 +64,7 @@ ginhandler(PG_FUNCTION_ARGS)
amroutine->amcostestimate = gincostestimate; amroutine->amcostestimate = gincostestimate;
amroutine->amoptions = ginoptions; amroutine->amoptions = ginoptions;
amroutine->amproperty = NULL; amroutine->amproperty = NULL;
amroutine->ambuildphasename = NULL;
amroutine->amvalidate = ginvalidate; amroutine->amvalidate = ginvalidate;
amroutine->ambeginscan = ginbeginscan; amroutine->ambeginscan = ginbeginscan;
amroutine->amrescan = ginrescan; amroutine->amrescan = ginrescan;
......
...@@ -86,6 +86,7 @@ gisthandler(PG_FUNCTION_ARGS) ...@@ -86,6 +86,7 @@ gisthandler(PG_FUNCTION_ARGS)
amroutine->amcostestimate = gistcostestimate; amroutine->amcostestimate = gistcostestimate;
amroutine->amoptions = gistoptions; amroutine->amoptions = gistoptions;
amroutine->amproperty = gistproperty; amroutine->amproperty = gistproperty;
amroutine->ambuildphasename = NULL;
amroutine->amvalidate = gistvalidate; amroutine->amvalidate = gistvalidate;
amroutine->ambeginscan = gistbeginscan; amroutine->ambeginscan = gistbeginscan;
amroutine->amrescan = gistrescan; amroutine->amrescan = gistrescan;
......
...@@ -205,7 +205,7 @@ gistbuild(Relation heap, Relation index, IndexInfo *indexInfo) ...@@ -205,7 +205,7 @@ gistbuild(Relation heap, Relation index, IndexInfo *indexInfo)
/* /*
* Do the heap scan. * Do the heap scan.
*/ */
reltuples = table_index_build_scan(heap, index, indexInfo, true, reltuples = table_index_build_scan(heap, index, indexInfo, true, true,
gistBuildCallback, gistBuildCallback,
(void *) &buildstate, NULL); (void *) &buildstate, NULL);
......
...@@ -23,9 +23,11 @@ ...@@ -23,9 +23,11 @@
#include "access/relscan.h" #include "access/relscan.h"
#include "access/tableam.h" #include "access/tableam.h"
#include "catalog/index.h" #include "catalog/index.h"
#include "commands/progress.h"
#include "commands/vacuum.h" #include "commands/vacuum.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "optimizer/plancat.h" #include "optimizer/plancat.h"
#include "pgstat.h"
#include "utils/builtins.h" #include "utils/builtins.h"
#include "utils/index_selfuncs.h" #include "utils/index_selfuncs.h"
#include "utils/rel.h" #include "utils/rel.h"
...@@ -83,6 +85,7 @@ hashhandler(PG_FUNCTION_ARGS) ...@@ -83,6 +85,7 @@ hashhandler(PG_FUNCTION_ARGS)
amroutine->amcostestimate = hashcostestimate; amroutine->amcostestimate = hashcostestimate;
amroutine->amoptions = hashoptions; amroutine->amoptions = hashoptions;
amroutine->amproperty = NULL; amroutine->amproperty = NULL;
amroutine->ambuildphasename = NULL;
amroutine->amvalidate = hashvalidate; amroutine->amvalidate = hashvalidate;
amroutine->ambeginscan = hashbeginscan; amroutine->ambeginscan = hashbeginscan;
amroutine->amrescan = hashrescan; amroutine->amrescan = hashrescan;
...@@ -160,9 +163,11 @@ hashbuild(Relation heap, Relation index, IndexInfo *indexInfo) ...@@ -160,9 +163,11 @@ hashbuild(Relation heap, Relation index, IndexInfo *indexInfo)
buildstate.heapRel = heap; buildstate.heapRel = heap;
/* do the heap scan */ /* do the heap scan */
reltuples = table_index_build_scan(heap, index, indexInfo, true, reltuples = table_index_build_scan(heap, index, indexInfo, true, true,
hashbuildCallback, hashbuildCallback,
(void *) &buildstate, NULL); (void *) &buildstate, NULL);
pgstat_progress_update_param(PROGRESS_CREATEIDX_TUPLES_TOTAL,
buildstate.indtuples);
if (buildstate.spool) if (buildstate.spool)
{ {
......
...@@ -26,7 +26,9 @@ ...@@ -26,7 +26,9 @@
#include "postgres.h" #include "postgres.h"
#include "access/hash.h" #include "access/hash.h"
#include "commands/progress.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "pgstat.h"
#include "utils/tuplesort.h" #include "utils/tuplesort.h"
...@@ -116,6 +118,7 @@ void ...@@ -116,6 +118,7 @@ void
_h_indexbuild(HSpool *hspool, Relation heapRel) _h_indexbuild(HSpool *hspool, Relation heapRel)
{ {
IndexTuple itup; IndexTuple itup;
long tups_done = 0;
#ifdef USE_ASSERT_CHECKING #ifdef USE_ASSERT_CHECKING
uint32 hashkey = 0; uint32 hashkey = 0;
#endif #endif
...@@ -141,5 +144,8 @@ _h_indexbuild(HSpool *hspool, Relation heapRel) ...@@ -141,5 +144,8 @@ _h_indexbuild(HSpool *hspool, Relation heapRel)
#endif #endif
_hash_doinsert(hspool->index, itup, heapRel); _hash_doinsert(hspool->index, itup, heapRel);
pgstat_progress_update_param(PROGRESS_CREATEIDX_TUPLES_DONE,
++tups_done);
} }
} }
...@@ -57,6 +57,8 @@ static bool SampleHeapTupleVisible(TableScanDesc scan, Buffer buffer, ...@@ -57,6 +57,8 @@ static bool SampleHeapTupleVisible(TableScanDesc scan, Buffer buffer,
HeapTuple tuple, HeapTuple tuple,
OffsetNumber tupoffset); OffsetNumber tupoffset);
static BlockNumber heapam_scan_get_blocks_done(HeapScanDesc hscan);
static const TableAmRoutine heapam_methods; static const TableAmRoutine heapam_methods;
...@@ -1120,6 +1122,7 @@ heapam_index_build_range_scan(Relation heapRelation, ...@@ -1120,6 +1122,7 @@ heapam_index_build_range_scan(Relation heapRelation,
IndexInfo *indexInfo, IndexInfo *indexInfo,
bool allow_sync, bool allow_sync,
bool anyvisible, bool anyvisible,
bool progress,
BlockNumber start_blockno, BlockNumber start_blockno,
BlockNumber numblocks, BlockNumber numblocks,
IndexBuildCallback callback, IndexBuildCallback callback,
...@@ -1140,6 +1143,7 @@ heapam_index_build_range_scan(Relation heapRelation, ...@@ -1140,6 +1143,7 @@ heapam_index_build_range_scan(Relation heapRelation,
Snapshot snapshot; Snapshot snapshot;
bool need_unregister_snapshot = false; bool need_unregister_snapshot = false;
TransactionId OldestXmin; TransactionId OldestXmin;
BlockNumber previous_blkno = InvalidBlockNumber;
BlockNumber root_blkno = InvalidBlockNumber; BlockNumber root_blkno = InvalidBlockNumber;
OffsetNumber root_offsets[MaxHeapTuplesPerPage]; OffsetNumber root_offsets[MaxHeapTuplesPerPage];
...@@ -1227,6 +1231,25 @@ heapam_index_build_range_scan(Relation heapRelation, ...@@ -1227,6 +1231,25 @@ heapam_index_build_range_scan(Relation heapRelation,
hscan = (HeapScanDesc) scan; hscan = (HeapScanDesc) scan;
/* Publish number of blocks to scan */
if (progress)
{
BlockNumber nblocks;
if (hscan->rs_base.rs_parallel != NULL)
{
ParallelBlockTableScanDesc pbscan;
pbscan = (ParallelBlockTableScanDesc) hscan->rs_base.rs_parallel;
nblocks = pbscan->phs_nblocks;
}
else
nblocks = hscan->rs_nblocks;
pgstat_progress_update_param(PROGRESS_SCAN_BLOCKS_TOTAL,
nblocks);
}
/* /*
* Must call GetOldestXmin() with SnapshotAny. Should never call * Must call GetOldestXmin() with SnapshotAny. Should never call
* GetOldestXmin() with MVCC snapshot. (It's especially worth checking * GetOldestXmin() with MVCC snapshot. (It's especially worth checking
...@@ -1259,6 +1282,19 @@ heapam_index_build_range_scan(Relation heapRelation, ...@@ -1259,6 +1282,19 @@ heapam_index_build_range_scan(Relation heapRelation,
CHECK_FOR_INTERRUPTS(); CHECK_FOR_INTERRUPTS();
/* Report scan progress, if asked to. */
if (progress)
{
BlockNumber blocks_done = heapam_scan_get_blocks_done(hscan);
if (blocks_done != previous_blkno)
{
pgstat_progress_update_param(PROGRESS_SCAN_BLOCKS_DONE,
blocks_done);
previous_blkno = blocks_done;
}
}
/* /*
* When dealing with a HOT-chain of updated tuples, we want to index * When dealing with a HOT-chain of updated tuples, we want to index
* the values of the live tuple (if any), but index it under the TID * the values of the live tuple (if any), but index it under the TID
...@@ -1600,6 +1636,25 @@ heapam_index_build_range_scan(Relation heapRelation, ...@@ -1600,6 +1636,25 @@ heapam_index_build_range_scan(Relation heapRelation,
} }
} }
/* Report scan progress one last time. */
if (progress)
{
BlockNumber blks_done;
if (hscan->rs_base.rs_parallel != NULL)
{
ParallelBlockTableScanDesc pbscan;
pbscan = (ParallelBlockTableScanDesc) hscan->rs_base.rs_parallel;
blks_done = pbscan->phs_nblocks;
}
else
blks_done = hscan->rs_nblocks;
pgstat_progress_update_param(PROGRESS_SCAN_BLOCKS_DONE,
blks_done);
}
table_endscan(scan); table_endscan(scan);
/* we can now forget our snapshot, if set and registered by us */ /* we can now forget our snapshot, if set and registered by us */
...@@ -1636,6 +1691,7 @@ heapam_index_validate_scan(Relation heapRelation, ...@@ -1636,6 +1691,7 @@ heapam_index_validate_scan(Relation heapRelation,
BlockNumber root_blkno = InvalidBlockNumber; BlockNumber root_blkno = InvalidBlockNumber;
OffsetNumber root_offsets[MaxHeapTuplesPerPage]; OffsetNumber root_offsets[MaxHeapTuplesPerPage];
bool in_index[MaxHeapTuplesPerPage]; bool in_index[MaxHeapTuplesPerPage];
BlockNumber previous_blkno = InvalidBlockNumber;
/* state variables for the merge */ /* state variables for the merge */
ItemPointer indexcursor = NULL; ItemPointer indexcursor = NULL;
...@@ -1676,6 +1732,9 @@ heapam_index_validate_scan(Relation heapRelation, ...@@ -1676,6 +1732,9 @@ heapam_index_validate_scan(Relation heapRelation,
false); /* syncscan not OK */ false); /* syncscan not OK */
hscan = (HeapScanDesc) scan; hscan = (HeapScanDesc) scan;
pgstat_progress_update_param(PROGRESS_SCAN_BLOCKS_TOTAL,
hscan->rs_nblocks);
/* /*
* Scan all tuples matching the snapshot. * Scan all tuples matching the snapshot.
*/ */
...@@ -1689,6 +1748,14 @@ heapam_index_validate_scan(Relation heapRelation, ...@@ -1689,6 +1748,14 @@ heapam_index_validate_scan(Relation heapRelation,
state->htups += 1; state->htups += 1;
if ((previous_blkno == InvalidBlockNumber) ||
(hscan->rs_cblock != previous_blkno))
{
pgstat_progress_update_param(PROGRESS_SCAN_BLOCKS_DONE,
hscan->rs_cblock);
previous_blkno = hscan->rs_cblock;
}
/* /*
* As commented in table_index_build_scan, we should index heap-only * As commented in table_index_build_scan, we should index heap-only
* tuples under the TIDs of their root tuples; so when we advance onto * tuples under the TIDs of their root tuples; so when we advance onto
...@@ -1849,6 +1916,46 @@ heapam_index_validate_scan(Relation heapRelation, ...@@ -1849,6 +1916,46 @@ heapam_index_validate_scan(Relation heapRelation,
indexInfo->ii_PredicateState = NULL; indexInfo->ii_PredicateState = NULL;
} }
/*
* Return the number of blocks that have been read by this scan since
* starting. This is meant for progress reporting rather than be fully
* accurate: in a parallel scan, workers can be concurrently reading blocks
* further ahead than what we report.
*/
static BlockNumber
heapam_scan_get_blocks_done(HeapScanDesc hscan)
{
ParallelBlockTableScanDesc bpscan = NULL;
BlockNumber startblock;
BlockNumber blocks_done;
if (hscan->rs_base.rs_parallel != NULL)
{
bpscan = (ParallelBlockTableScanDesc) hscan->rs_base.rs_parallel;
startblock = bpscan->phs_startblock;
}
else
startblock = hscan->rs_startblock;
/*
* Might have wrapped around the end of the relation, if startblock was
* not zero.
*/
if (hscan->rs_cblock > startblock)
blocks_done = hscan->rs_cblock - startblock;
else
{
BlockNumber nblocks;
nblocks = bpscan != NULL ? bpscan->phs_nblocks : hscan->rs_nblocks;
blocks_done = nblocks - startblock +
hscan->rs_cblock;
}
return blocks_done;
}
/* ------------------------------------------------------------------------ /* ------------------------------------------------------------------------
* Planner related callbacks for the heap AM * Planner related callbacks for the heap AM
......
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
#include "access/nbtxlog.h" #include "access/nbtxlog.h"
#include "access/relscan.h" #include "access/relscan.h"
#include "access/xlog.h" #include "access/xlog.h"
#include "commands/progress.h"
#include "commands/vacuum.h" #include "commands/vacuum.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "nodes/execnodes.h" #include "nodes/execnodes.h"
...@@ -133,6 +134,7 @@ bthandler(PG_FUNCTION_ARGS) ...@@ -133,6 +134,7 @@ bthandler(PG_FUNCTION_ARGS)
amroutine->amcostestimate = btcostestimate; amroutine->amcostestimate = btcostestimate;
amroutine->amoptions = btoptions; amroutine->amoptions = btoptions;
amroutine->amproperty = btproperty; amroutine->amproperty = btproperty;
amroutine->ambuildphasename = btbuildphasename;
amroutine->amvalidate = btvalidate; amroutine->amvalidate = btvalidate;
amroutine->ambeginscan = btbeginscan; amroutine->ambeginscan = btbeginscan;
amroutine->amrescan = btrescan; amroutine->amrescan = btrescan;
...@@ -1021,6 +1023,10 @@ btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats, ...@@ -1021,6 +1023,10 @@ btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats,
if (needLock) if (needLock)
UnlockRelationForExtension(rel, ExclusiveLock); UnlockRelationForExtension(rel, ExclusiveLock);
if (info->report_progress)
pgstat_progress_update_param(PROGRESS_SCAN_BLOCKS_TOTAL,
num_pages);
/* Quit if we've scanned the whole relation */ /* Quit if we've scanned the whole relation */
if (blkno >= num_pages) if (blkno >= num_pages)
break; break;
...@@ -1028,6 +1034,9 @@ btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats, ...@@ -1028,6 +1034,9 @@ btvacuumscan(IndexVacuumInfo *info, IndexBulkDeleteResult *stats,
for (; blkno < num_pages; blkno++) for (; blkno < num_pages; blkno++)
{ {
btvacuumpage(&vstate, blkno, blkno); btvacuumpage(&vstate, blkno, blkno);
if (info->report_progress)
pgstat_progress_update_param(PROGRESS_SCAN_BLOCKS_DONE,
blkno);
} }
} }
......
...@@ -66,6 +66,7 @@ ...@@ -66,6 +66,7 @@
#include "access/xlog.h" #include "access/xlog.h"
#include "access/xloginsert.h" #include "access/xloginsert.h"
#include "catalog/index.h" #include "catalog/index.h"
#include "commands/progress.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "pgstat.h" #include "pgstat.h"
#include "storage/smgr.h" #include "storage/smgr.h"
...@@ -298,7 +299,8 @@ static double _bt_parallel_heapscan(BTBuildState *buildstate, ...@@ -298,7 +299,8 @@ static double _bt_parallel_heapscan(BTBuildState *buildstate,
static void _bt_leader_participate_as_worker(BTBuildState *buildstate); static void _bt_leader_participate_as_worker(BTBuildState *buildstate);
static void _bt_parallel_scan_and_sort(BTSpool *btspool, BTSpool *btspool2, static void _bt_parallel_scan_and_sort(BTSpool *btspool, BTSpool *btspool2,
BTShared *btshared, Sharedsort *sharedsort, BTShared *btshared, Sharedsort *sharedsort,
Sharedsort *sharedsort2, int sortmem); Sharedsort *sharedsort2, int sortmem,
bool progress);
/* /*
...@@ -394,6 +396,10 @@ _bt_spools_heapscan(Relation heap, Relation index, BTBuildState *buildstate, ...@@ -394,6 +396,10 @@ _bt_spools_heapscan(Relation heap, Relation index, BTBuildState *buildstate,
/* Save as primary spool */ /* Save as primary spool */
buildstate->spool = btspool; buildstate->spool = btspool;
/* Report table scan phase started */
pgstat_progress_update_param(PROGRESS_CREATEIDX_SUBPHASE,
PROGRESS_BTREE_PHASE_INDEXBUILD_TABLESCAN);
/* Attempt to launch parallel worker scan when required */ /* Attempt to launch parallel worker scan when required */
if (indexInfo->ii_ParallelWorkers > 0) if (indexInfo->ii_ParallelWorkers > 0)
_bt_begin_parallel(buildstate, indexInfo->ii_Concurrent, _bt_begin_parallel(buildstate, indexInfo->ii_Concurrent,
...@@ -480,13 +486,31 @@ _bt_spools_heapscan(Relation heap, Relation index, BTBuildState *buildstate, ...@@ -480,13 +486,31 @@ _bt_spools_heapscan(Relation heap, Relation index, BTBuildState *buildstate,
/* Fill spool using either serial or parallel heap scan */ /* Fill spool using either serial or parallel heap scan */
if (!buildstate->btleader) if (!buildstate->btleader)
reltuples = table_index_build_scan(heap, index, indexInfo, true, reltuples = table_index_build_scan(heap, index, indexInfo, true, true,
_bt_build_callback, (void *) buildstate, _bt_build_callback, (void *) buildstate,
NULL); NULL);
else else
reltuples = _bt_parallel_heapscan(buildstate, reltuples = _bt_parallel_heapscan(buildstate,
&indexInfo->ii_BrokenHotChain); &indexInfo->ii_BrokenHotChain);
/*
* Set the progress target for the next phase. Reset the block number
* values set by table_index_build_scan
*/
{
const int index[] = {
PROGRESS_CREATEIDX_TUPLES_TOTAL,
PROGRESS_SCAN_BLOCKS_TOTAL,
PROGRESS_SCAN_BLOCKS_DONE
};
const int64 val[] = {
buildstate->indtuples,
0, 0
};
pgstat_progress_update_multi_param(3, index, val);
}
/* okay, all heap tuples are spooled */ /* okay, all heap tuples are spooled */
if (buildstate->spool2 && !buildstate->havedead) if (buildstate->spool2 && !buildstate->havedead)
{ {
...@@ -535,9 +559,15 @@ _bt_leafbuild(BTSpool *btspool, BTSpool *btspool2) ...@@ -535,9 +559,15 @@ _bt_leafbuild(BTSpool *btspool, BTSpool *btspool2)
} }
#endif /* BTREE_BUILD_STATS */ #endif /* BTREE_BUILD_STATS */
pgstat_progress_update_param(PROGRESS_CREATEIDX_SUBPHASE,
PROGRESS_BTREE_PHASE_PERFORMSORT_1);
tuplesort_performsort(btspool->sortstate); tuplesort_performsort(btspool->sortstate);
if (btspool2) if (btspool2)
{
pgstat_progress_update_param(PROGRESS_CREATEIDX_SUBPHASE,
PROGRESS_BTREE_PHASE_PERFORMSORT_2);
tuplesort_performsort(btspool2->sortstate); tuplesort_performsort(btspool2->sortstate);
}
wstate.heap = btspool->heap; wstate.heap = btspool->heap;
wstate.index = btspool->index; wstate.index = btspool->index;
...@@ -554,6 +584,8 @@ _bt_leafbuild(BTSpool *btspool, BTSpool *btspool2) ...@@ -554,6 +584,8 @@ _bt_leafbuild(BTSpool *btspool, BTSpool *btspool2)
wstate.btws_pages_written = 0; wstate.btws_pages_written = 0;
wstate.btws_zeropage = NULL; /* until needed */ wstate.btws_zeropage = NULL; /* until needed */
pgstat_progress_update_param(PROGRESS_CREATEIDX_SUBPHASE,
PROGRESS_BTREE_PHASE_LEAF_LOAD);
_bt_load(&wstate, btspool, btspool2); _bt_load(&wstate, btspool, btspool2);
} }
...@@ -1098,6 +1130,7 @@ _bt_load(BTWriteState *wstate, BTSpool *btspool, BTSpool *btspool2) ...@@ -1098,6 +1130,7 @@ _bt_load(BTWriteState *wstate, BTSpool *btspool, BTSpool *btspool2)
int i, int i,
keysz = IndexRelationGetNumberOfKeyAttributes(wstate->index); keysz = IndexRelationGetNumberOfKeyAttributes(wstate->index);
SortSupport sortKeys; SortSupport sortKeys;
long tuples_done = 0;
if (merge) if (merge)
{ {
...@@ -1202,6 +1235,10 @@ _bt_load(BTWriteState *wstate, BTSpool *btspool, BTSpool *btspool2) ...@@ -1202,6 +1235,10 @@ _bt_load(BTWriteState *wstate, BTSpool *btspool, BTSpool *btspool2)
_bt_buildadd(wstate, state, itup2); _bt_buildadd(wstate, state, itup2);
itup2 = tuplesort_getindextuple(btspool2->sortstate, true); itup2 = tuplesort_getindextuple(btspool2->sortstate, true);
} }
/* Report progress */
pgstat_progress_update_param(PROGRESS_CREATEIDX_TUPLES_DONE,
++tuples_done);
} }
pfree(sortKeys); pfree(sortKeys);
} }
...@@ -1216,6 +1253,10 @@ _bt_load(BTWriteState *wstate, BTSpool *btspool, BTSpool *btspool2) ...@@ -1216,6 +1253,10 @@ _bt_load(BTWriteState *wstate, BTSpool *btspool, BTSpool *btspool2)
state = _bt_pagestate(wstate, 0); state = _bt_pagestate(wstate, 0);
_bt_buildadd(wstate, state, itup); _bt_buildadd(wstate, state, itup);
/* Report progress */
pgstat_progress_update_param(PROGRESS_CREATEIDX_TUPLES_DONE,
++tuples_done);
} }
} }
...@@ -1528,7 +1569,7 @@ _bt_leader_participate_as_worker(BTBuildState *buildstate) ...@@ -1528,7 +1569,7 @@ _bt_leader_participate_as_worker(BTBuildState *buildstate)
/* Perform work common to all participants */ /* Perform work common to all participants */
_bt_parallel_scan_and_sort(leaderworker, leaderworker2, btleader->btshared, _bt_parallel_scan_and_sort(leaderworker, leaderworker2, btleader->btshared,
btleader->sharedsort, btleader->sharedsort2, btleader->sharedsort, btleader->sharedsort2,
sortmem); sortmem, true);
#ifdef BTREE_BUILD_STATS #ifdef BTREE_BUILD_STATS
if (log_btree_build_stats) if (log_btree_build_stats)
...@@ -1619,7 +1660,7 @@ _bt_parallel_build_main(dsm_segment *seg, shm_toc *toc) ...@@ -1619,7 +1660,7 @@ _bt_parallel_build_main(dsm_segment *seg, shm_toc *toc)
/* Perform sorting of spool, and possibly a spool2 */ /* Perform sorting of spool, and possibly a spool2 */
sortmem = maintenance_work_mem / btshared->scantuplesortstates; sortmem = maintenance_work_mem / btshared->scantuplesortstates;
_bt_parallel_scan_and_sort(btspool, btspool2, btshared, sharedsort, _bt_parallel_scan_and_sort(btspool, btspool2, btshared, sharedsort,
sharedsort2, sortmem); sharedsort2, sortmem, false);
#ifdef BTREE_BUILD_STATS #ifdef BTREE_BUILD_STATS
if (log_btree_build_stats) if (log_btree_build_stats)
...@@ -1648,7 +1689,7 @@ _bt_parallel_build_main(dsm_segment *seg, shm_toc *toc) ...@@ -1648,7 +1689,7 @@ _bt_parallel_build_main(dsm_segment *seg, shm_toc *toc)
static void static void
_bt_parallel_scan_and_sort(BTSpool *btspool, BTSpool *btspool2, _bt_parallel_scan_and_sort(BTSpool *btspool, BTSpool *btspool2,
BTShared *btshared, Sharedsort *sharedsort, BTShared *btshared, Sharedsort *sharedsort,
Sharedsort *sharedsort2, int sortmem) Sharedsort *sharedsort2, int sortmem, bool progress)
{ {
SortCoordinate coordinate; SortCoordinate coordinate;
BTBuildState buildstate; BTBuildState buildstate;
...@@ -1705,9 +1746,10 @@ _bt_parallel_scan_and_sort(BTSpool *btspool, BTSpool *btspool2, ...@@ -1705,9 +1746,10 @@ _bt_parallel_scan_and_sort(BTSpool *btspool, BTSpool *btspool2,
/* Join parallel scan */ /* Join parallel scan */
indexInfo = BuildIndexInfo(btspool->index); indexInfo = BuildIndexInfo(btspool->index);
indexInfo->ii_Concurrent = btshared->isconcurrent; indexInfo->ii_Concurrent = btshared->isconcurrent;
scan = table_beginscan_parallel(btspool->heap, ParallelTableScanFromBTShared(btshared)); scan = table_beginscan_parallel(btspool->heap,
ParallelTableScanFromBTShared(btshared));
reltuples = table_index_build_scan(btspool->heap, btspool->index, indexInfo, reltuples = table_index_build_scan(btspool->heap, btspool->index, indexInfo,
true, _bt_build_callback, true, progress, _bt_build_callback,
(void *) &buildstate, scan); (void *) &buildstate, scan);
/* /*
......
...@@ -20,6 +20,7 @@ ...@@ -20,6 +20,7 @@
#include "access/nbtree.h" #include "access/nbtree.h"
#include "access/reloptions.h" #include "access/reloptions.h"
#include "access/relscan.h" #include "access/relscan.h"
#include "commands/progress.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "utils/array.h" #include "utils/array.h"
#include "utils/datum.h" #include "utils/datum.h"
...@@ -2051,6 +2052,29 @@ btproperty(Oid index_oid, int attno, ...@@ -2051,6 +2052,29 @@ btproperty(Oid index_oid, int attno,
} }
} }
/*
* btbuildphasename() -- Return name of index build phase.
*/
char *
btbuildphasename(int64 phasenum)
{
switch (phasenum)
{
case PROGRESS_CREATEIDX_SUBPHASE_INITIALIZE:
return "initializing";
case PROGRESS_BTREE_PHASE_INDEXBUILD_TABLESCAN:
return "scanning table";
case PROGRESS_BTREE_PHASE_PERFORMSORT_1:
return "sorting live tuples";
case PROGRESS_BTREE_PHASE_PERFORMSORT_2:
return "sorting dead tuples";
case PROGRESS_BTREE_PHASE_LEAF_LOAD:
return "loading tuples in tree";
default:
return NULL;
}
}
/* /*
* _bt_truncate() -- create tuple without unneeded suffix attributes. * _bt_truncate() -- create tuple without unneeded suffix attributes.
* *
......
...@@ -143,7 +143,7 @@ spgbuild(Relation heap, Relation index, IndexInfo *indexInfo) ...@@ -143,7 +143,7 @@ spgbuild(Relation heap, Relation index, IndexInfo *indexInfo)
"SP-GiST build temporary context", "SP-GiST build temporary context",
ALLOCSET_DEFAULT_SIZES); ALLOCSET_DEFAULT_SIZES);
reltuples = table_index_build_scan(heap, index, indexInfo, true, reltuples = table_index_build_scan(heap, index, indexInfo, true, true,
spgistBuildCallback, (void *) &buildstate, spgistBuildCallback, (void *) &buildstate,
NULL); NULL);
......
...@@ -67,6 +67,7 @@ spghandler(PG_FUNCTION_ARGS) ...@@ -67,6 +67,7 @@ spghandler(PG_FUNCTION_ARGS)
amroutine->amcostestimate = spgcostestimate; amroutine->amcostestimate = spgcostestimate;
amroutine->amoptions = spgoptions; amroutine->amoptions = spgoptions;
amroutine->amproperty = spgproperty; amroutine->amproperty = spgproperty;
amroutine->ambuildphasename = NULL;
amroutine->amvalidate = spgvalidate; amroutine->amvalidate = spgvalidate;
amroutine->ambeginscan = spgbeginscan; amroutine->ambeginscan = spgbeginscan;
amroutine->amrescan = spgrescan; amroutine->amrescan = spgrescan;
......
...@@ -51,9 +51,9 @@ ...@@ -51,9 +51,9 @@
#include "catalog/pg_trigger.h" #include "catalog/pg_trigger.h"
#include "catalog/pg_type.h" #include "catalog/pg_type.h"
#include "catalog/storage.h" #include "catalog/storage.h"
#include "commands/tablecmds.h"
#include "commands/event_trigger.h" #include "commands/event_trigger.h"
#include "commands/progress.h" #include "commands/progress.h"
#include "commands/tablecmds.h"
#include "commands/trigger.h" #include "commands/trigger.h"
#include "executor/executor.h" #include "executor/executor.h"
#include "miscadmin.h" #include "miscadmin.h"
...@@ -2047,7 +2047,7 @@ index_drop(Oid indexId, bool concurrent, bool concurrent_lock_mode) ...@@ -2047,7 +2047,7 @@ index_drop(Oid indexId, bool concurrent, bool concurrent_lock_mode)
* to acquire an exclusive lock on our table. The lock code will * to acquire an exclusive lock on our table. The lock code will
* detect deadlock and error out properly. * detect deadlock and error out properly.
*/ */
WaitForLockers(heaplocktag, AccessExclusiveLock); WaitForLockers(heaplocktag, AccessExclusiveLock, true);
/* Finish invalidation of index and mark it as dead */ /* Finish invalidation of index and mark it as dead */
index_concurrently_set_dead(heapId, indexId); index_concurrently_set_dead(heapId, indexId);
...@@ -2063,7 +2063,7 @@ index_drop(Oid indexId, bool concurrent, bool concurrent_lock_mode) ...@@ -2063,7 +2063,7 @@ index_drop(Oid indexId, bool concurrent, bool concurrent_lock_mode)
* Wait till every transaction that saw the old index state has * Wait till every transaction that saw the old index state has
* finished. * finished.
*/ */
WaitForLockers(heaplocktag, AccessExclusiveLock); WaitForLockers(heaplocktag, AccessExclusiveLock, true);
/* /*
* Re-open relations to allow us to complete our actions. * Re-open relations to allow us to complete our actions.
...@@ -2712,6 +2712,25 @@ index_build(Relation heapRelation, ...@@ -2712,6 +2712,25 @@ index_build(Relation heapRelation,
save_sec_context | SECURITY_RESTRICTED_OPERATION); save_sec_context | SECURITY_RESTRICTED_OPERATION);
save_nestlevel = NewGUCNestLevel(); save_nestlevel = NewGUCNestLevel();
/* Set up initial progress report status */
{
const int index[] = {
PROGRESS_CREATEIDX_PHASE,
PROGRESS_CREATEIDX_SUBPHASE,
PROGRESS_CREATEIDX_TUPLES_DONE,
PROGRESS_CREATEIDX_TUPLES_TOTAL,
PROGRESS_SCAN_BLOCKS_DONE,
PROGRESS_SCAN_BLOCKS_TOTAL
};
const int64 val[] = {
PROGRESS_CREATEIDX_PHASE_BUILD,
PROGRESS_CREATEIDX_SUBPHASE_INITIALIZE,
0, 0, 0, 0
};
pgstat_progress_update_multi_param(6, index, val);
}
/* /*
* Call the access method's build procedure * Call the access method's build procedure
*/ */
...@@ -3000,6 +3019,21 @@ validate_index(Oid heapId, Oid indexId, Snapshot snapshot) ...@@ -3000,6 +3019,21 @@ validate_index(Oid heapId, Oid indexId, Snapshot snapshot)
int save_sec_context; int save_sec_context;
int save_nestlevel; int save_nestlevel;
{
const int index[] = {
PROGRESS_CREATEIDX_PHASE,
PROGRESS_CREATEIDX_TUPLES_DONE,
PROGRESS_CREATEIDX_TUPLES_TOTAL,
PROGRESS_SCAN_BLOCKS_DONE,
PROGRESS_SCAN_BLOCKS_TOTAL
};
const int64 val[] = {
PROGRESS_CREATEIDX_PHASE_VALIDATE_IDXSCAN,
0, 0, 0, 0
};
pgstat_progress_update_multi_param(5, index, val);
}
/* Open and lock the parent heap relation */ /* Open and lock the parent heap relation */
heapRelation = table_open(heapId, ShareUpdateExclusiveLock); heapRelation = table_open(heapId, ShareUpdateExclusiveLock);
/* And the target index relation */ /* And the target index relation */
...@@ -3030,6 +3064,7 @@ validate_index(Oid heapId, Oid indexId, Snapshot snapshot) ...@@ -3030,6 +3064,7 @@ validate_index(Oid heapId, Oid indexId, Snapshot snapshot)
*/ */
ivinfo.index = indexRelation; ivinfo.index = indexRelation;
ivinfo.analyze_only = false; ivinfo.analyze_only = false;
ivinfo.report_progress = true;
ivinfo.estimated_count = true; ivinfo.estimated_count = true;
ivinfo.message_level = DEBUG2; ivinfo.message_level = DEBUG2;
ivinfo.num_heap_tuples = heapRelation->rd_rel->reltuples; ivinfo.num_heap_tuples = heapRelation->rd_rel->reltuples;
...@@ -3047,15 +3082,31 @@ validate_index(Oid heapId, Oid indexId, Snapshot snapshot) ...@@ -3047,15 +3082,31 @@ validate_index(Oid heapId, Oid indexId, Snapshot snapshot)
NULL, false); NULL, false);
state.htups = state.itups = state.tups_inserted = 0; state.htups = state.itups = state.tups_inserted = 0;
/* ambulkdelete updates progress metrics */
(void) index_bulk_delete(&ivinfo, NULL, (void) index_bulk_delete(&ivinfo, NULL,
validate_index_callback, (void *) &state); validate_index_callback, (void *) &state);
/* Execute the sort */ /* Execute the sort */
{
const int index[] = {
PROGRESS_CREATEIDX_PHASE,
PROGRESS_SCAN_BLOCKS_DONE,
PROGRESS_SCAN_BLOCKS_TOTAL
};
const int64 val[] = {
PROGRESS_CREATEIDX_PHASE_VALIDATE_SORT,
0, 0
};
pgstat_progress_update_multi_param(3, index, val);
}
tuplesort_performsort(state.tuplesort); tuplesort_performsort(state.tuplesort);
/* /*
* Now scan the heap and "merge" it with the index * Now scan the heap and "merge" it with the index
*/ */
pgstat_progress_update_param(PROGRESS_CREATEIDX_PHASE,
PROGRESS_CREATEIDX_PHASE_VALIDATE_TABLESCAN);
table_index_validate_scan(heapRelation, table_index_validate_scan(heapRelation,
indexRelation, indexRelation,
indexInfo, indexInfo,
......
...@@ -934,6 +934,33 @@ CREATE VIEW pg_stat_progress_cluster AS ...@@ -934,6 +934,33 @@ CREATE VIEW pg_stat_progress_cluster AS
FROM pg_stat_get_progress_info('CLUSTER') AS S FROM pg_stat_get_progress_info('CLUSTER') AS S
LEFT JOIN pg_database D ON S.datid = D.oid; LEFT JOIN pg_database D ON S.datid = D.oid;
CREATE VIEW pg_stat_progress_create_index AS
SELECT
S.pid AS pid, S.datid AS datid, D.datname AS datname,
S.relid AS relid,
CASE S.param10 WHEN 0 THEN 'initializing'
WHEN 1 THEN 'waiting for old snapshots'
WHEN 2 THEN 'building index' ||
COALESCE((': ' || pg_indexam_progress_phasename(S.param9::oid, S.param11)),
'')
WHEN 3 THEN 'waiting for writer snapshots'
WHEN 4 THEN 'index validation: scanning index'
WHEN 5 THEN 'index validation: sorting tuples'
WHEN 6 THEN 'index validation: scanning table'
WHEN 7 THEN 'waiting for reader snapshots'
END as phase,
S.param4 AS lockers_total,
S.param5 AS lockers_done,
S.param6 AS current_locker_pid,
S.param16 AS blocks_total,
S.param17 AS blocks_done,
S.param12 AS tuples_total,
S.param13 AS tuples_done,
S.param14 AS partitions_total,
S.param15 AS partitions_done
FROM pg_stat_get_progress_info('CREATE INDEX') AS S
LEFT JOIN pg_database D ON S.datid = D.oid;
CREATE VIEW pg_user_mappings AS CREATE VIEW pg_user_mappings AS
SELECT SELECT
U.oid AS umid, U.oid AS umid,
......
...@@ -36,6 +36,7 @@ ...@@ -36,6 +36,7 @@
#include "commands/dbcommands.h" #include "commands/dbcommands.h"
#include "commands/defrem.h" #include "commands/defrem.h"
#include "commands/event_trigger.h" #include "commands/event_trigger.h"
#include "commands/progress.h"
#include "commands/tablecmds.h" #include "commands/tablecmds.h"
#include "commands/tablespace.h" #include "commands/tablespace.h"
#include "mb/pg_wchar.h" #include "mb/pg_wchar.h"
...@@ -47,10 +48,12 @@ ...@@ -47,10 +48,12 @@
#include "parser/parse_func.h" #include "parser/parse_func.h"
#include "parser/parse_oper.h" #include "parser/parse_oper.h"
#include "partitioning/partdesc.h" #include "partitioning/partdesc.h"
#include "pgstat.h"
#include "rewrite/rewriteManip.h" #include "rewrite/rewriteManip.h"
#include "storage/lmgr.h" #include "storage/lmgr.h"
#include "storage/proc.h" #include "storage/proc.h"
#include "storage/procarray.h" #include "storage/procarray.h"
#include "storage/sinvaladt.h"
#include "utils/acl.h" #include "utils/acl.h"
#include "utils/builtins.h" #include "utils/builtins.h"
#include "utils/fmgroids.h" #include "utils/fmgroids.h"
...@@ -334,7 +337,7 @@ CheckIndexCompatible(Oid oldId, ...@@ -334,7 +337,7 @@ CheckIndexCompatible(Oid oldId,
* doesn't show up in the output, we know we can forget about it. * doesn't show up in the output, we know we can forget about it.
*/ */
static void static void
WaitForOlderSnapshots(TransactionId limitXmin) WaitForOlderSnapshots(TransactionId limitXmin, bool progress)
{ {
int n_old_snapshots; int n_old_snapshots;
int i; int i;
...@@ -343,6 +346,8 @@ WaitForOlderSnapshots(TransactionId limitXmin) ...@@ -343,6 +346,8 @@ WaitForOlderSnapshots(TransactionId limitXmin)
old_snapshots = GetCurrentVirtualXIDs(limitXmin, true, false, old_snapshots = GetCurrentVirtualXIDs(limitXmin, true, false,
PROC_IS_AUTOVACUUM | PROC_IN_VACUUM, PROC_IS_AUTOVACUUM | PROC_IN_VACUUM,
&n_old_snapshots); &n_old_snapshots);
if (progress)
pgstat_progress_update_param(PROGRESS_WAITFOR_TOTAL, n_old_snapshots);
for (i = 0; i < n_old_snapshots; i++) for (i = 0; i < n_old_snapshots; i++)
{ {
...@@ -378,7 +383,19 @@ WaitForOlderSnapshots(TransactionId limitXmin) ...@@ -378,7 +383,19 @@ WaitForOlderSnapshots(TransactionId limitXmin)
} }
if (VirtualTransactionIdIsValid(old_snapshots[i])) if (VirtualTransactionIdIsValid(old_snapshots[i]))
{
if (progress)
{
PGPROC *holder = BackendIdGetProc(old_snapshots[i].backendId);
pgstat_progress_update_param(PROGRESS_WAITFOR_CURRENT_PID,
holder->pid);
}
VirtualXactLock(old_snapshots[i], true); VirtualXactLock(old_snapshots[i], true);
}
if (progress)
pgstat_progress_update_param(PROGRESS_WAITFOR_DONE, i + 1);
} }
} }
...@@ -452,6 +469,15 @@ DefineIndex(Oid relationId, ...@@ -452,6 +469,15 @@ DefineIndex(Oid relationId,
Snapshot snapshot; Snapshot snapshot;
int i; int i;
/*
* Start progress report. If we're building a partition, this was already
* done.
*/
if (!OidIsValid(parentIndexId))
pgstat_progress_start_command(PROGRESS_COMMAND_CREATE_INDEX,
relationId);
/* /*
* count key attributes in index * count key attributes in index
*/ */
...@@ -668,6 +694,9 @@ DefineIndex(Oid relationId, ...@@ -668,6 +694,9 @@ DefineIndex(Oid relationId,
accessMethodId = accessMethodForm->oid; accessMethodId = accessMethodForm->oid;
amRoutine = GetIndexAmRoutine(accessMethodForm->amhandler); amRoutine = GetIndexAmRoutine(accessMethodForm->amhandler);
pgstat_progress_update_param(PROGRESS_CREATEIDX_ACCESS_METHOD_OID,
accessMethodId);
if (stmt->unique && !amRoutine->amcanunique) if (stmt->unique && !amRoutine->amcanunique)
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED), (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
...@@ -948,6 +977,11 @@ DefineIndex(Oid relationId, ...@@ -948,6 +977,11 @@ DefineIndex(Oid relationId,
if (!OidIsValid(indexRelationId)) if (!OidIsValid(indexRelationId))
{ {
table_close(rel, NoLock); table_close(rel, NoLock);
/* If this is the top-level index, we're done */
if (!OidIsValid(parentIndexId))
pgstat_progress_end_command();
return address; return address;
} }
...@@ -973,6 +1007,9 @@ DefineIndex(Oid relationId, ...@@ -973,6 +1007,9 @@ DefineIndex(Oid relationId,
TupleDesc parentDesc; TupleDesc parentDesc;
Oid *opfamOids; Oid *opfamOids;
pgstat_progress_update_param(PROGRESS_CREATEIDX_PARTITIONS_TOTAL,
nparts);
memcpy(part_oids, partdesc->oids, sizeof(Oid) * nparts); memcpy(part_oids, partdesc->oids, sizeof(Oid) * nparts);
parentDesc = CreateTupleDescCopy(RelationGetDescr(rel)); parentDesc = CreateTupleDescCopy(RelationGetDescr(rel));
...@@ -1122,6 +1159,8 @@ DefineIndex(Oid relationId, ...@@ -1122,6 +1159,8 @@ DefineIndex(Oid relationId,
skip_build, quiet); skip_build, quiet);
} }
pgstat_progress_update_param(PROGRESS_CREATEIDX_PARTITIONS_DONE,
i + 1);
pfree(attmap); pfree(attmap);
} }
...@@ -1156,6 +1195,8 @@ DefineIndex(Oid relationId, ...@@ -1156,6 +1195,8 @@ DefineIndex(Oid relationId,
* Indexes on partitioned tables are not themselves built, so we're * Indexes on partitioned tables are not themselves built, so we're
* done here. * done here.
*/ */
if (!OidIsValid(parentIndexId))
pgstat_progress_end_command();
return address; return address;
} }
...@@ -1163,6 +1204,11 @@ DefineIndex(Oid relationId, ...@@ -1163,6 +1204,11 @@ DefineIndex(Oid relationId,
{ {
/* Close the heap and we're done, in the non-concurrent case */ /* Close the heap and we're done, in the non-concurrent case */
table_close(rel, NoLock); table_close(rel, NoLock);
/* If this is the top-level index, we're done. */
if (!OidIsValid(parentIndexId))
pgstat_progress_end_command();
return address; return address;
} }
...@@ -1214,7 +1260,9 @@ DefineIndex(Oid relationId, ...@@ -1214,7 +1260,9 @@ DefineIndex(Oid relationId,
* exclusive lock on our table. The lock code will detect deadlock and * exclusive lock on our table. The lock code will detect deadlock and
* error out properly. * error out properly.
*/ */
WaitForLockers(heaplocktag, ShareLock); pgstat_progress_update_param(PROGRESS_CREATEIDX_PHASE,
PROGRESS_CREATEIDX_PHASE_WAIT_1);
WaitForLockers(heaplocktag, ShareLock, true);
/* /*
* At this moment we are sure that there are no transactions with the * At this moment we are sure that there are no transactions with the
...@@ -1255,7 +1303,9 @@ DefineIndex(Oid relationId, ...@@ -1255,7 +1303,9 @@ DefineIndex(Oid relationId,
* We once again wait until no transaction can have the table open with * We once again wait until no transaction can have the table open with
* the index marked as read-only for updates. * the index marked as read-only for updates.
*/ */
WaitForLockers(heaplocktag, ShareLock); pgstat_progress_update_param(PROGRESS_CREATEIDX_PHASE,
PROGRESS_CREATEIDX_PHASE_WAIT_2);
WaitForLockers(heaplocktag, ShareLock, true);
/* /*
* Now take the "reference snapshot" that will be used by validate_index() * Now take the "reference snapshot" that will be used by validate_index()
...@@ -1312,7 +1362,9 @@ DefineIndex(Oid relationId, ...@@ -1312,7 +1362,9 @@ DefineIndex(Oid relationId,
* before the reference snap was taken, we have to wait out any * before the reference snap was taken, we have to wait out any
* transactions that might have older snapshots. * transactions that might have older snapshots.
*/ */
WaitForOlderSnapshots(limitXmin); pgstat_progress_update_param(PROGRESS_CREATEIDX_PHASE,
PROGRESS_CREATEIDX_PHASE_WAIT_3);
WaitForOlderSnapshots(limitXmin, true);
/* /*
* Index can now be marked valid -- update its pg_index entry * Index can now be marked valid -- update its pg_index entry
...@@ -1334,6 +1386,8 @@ DefineIndex(Oid relationId, ...@@ -1334,6 +1386,8 @@ DefineIndex(Oid relationId,
*/ */
UnlockRelationIdForSession(&heaprelid, ShareUpdateExclusiveLock); UnlockRelationIdForSession(&heaprelid, ShareUpdateExclusiveLock);
pgstat_progress_end_command();
return address; return address;
} }
...@@ -2913,7 +2967,7 @@ ReindexRelationConcurrently(Oid relationOid, int options) ...@@ -2913,7 +2967,7 @@ ReindexRelationConcurrently(Oid relationOid, int options)
* DefineIndex() for more details. * DefineIndex() for more details.
*/ */
WaitForLockersMultiple(lockTags, ShareLock); WaitForLockersMultiple(lockTags, ShareLock, false);
CommitTransactionCommand(); CommitTransactionCommand();
forboth(lc, indexIds, lc2, newIndexIds) forboth(lc, indexIds, lc2, newIndexIds)
...@@ -2955,7 +3009,7 @@ ReindexRelationConcurrently(Oid relationOid, int options) ...@@ -2955,7 +3009,7 @@ ReindexRelationConcurrently(Oid relationOid, int options)
* for more details. * for more details.
*/ */
WaitForLockersMultiple(lockTags, ShareLock); WaitForLockersMultiple(lockTags, ShareLock, false);
CommitTransactionCommand(); CommitTransactionCommand();
foreach(lc, newIndexIds) foreach(lc, newIndexIds)
...@@ -3003,7 +3057,7 @@ ReindexRelationConcurrently(Oid relationOid, int options) ...@@ -3003,7 +3057,7 @@ ReindexRelationConcurrently(Oid relationOid, int options)
* before the reference snap was taken, we have to wait out any * before the reference snap was taken, we have to wait out any
* transactions that might have older snapshots. * transactions that might have older snapshots.
*/ */
WaitForOlderSnapshots(limitXmin); WaitForOlderSnapshots(limitXmin, false);
CommitTransactionCommand(); CommitTransactionCommand();
} }
...@@ -3074,7 +3128,7 @@ ReindexRelationConcurrently(Oid relationOid, int options) ...@@ -3074,7 +3128,7 @@ ReindexRelationConcurrently(Oid relationOid, int options)
* index_drop() for more details. * index_drop() for more details.
*/ */
WaitForLockersMultiple(lockTags, AccessExclusiveLock); WaitForLockersMultiple(lockTags, AccessExclusiveLock, false);
foreach(lc, indexIds) foreach(lc, indexIds)
{ {
...@@ -3096,7 +3150,7 @@ ReindexRelationConcurrently(Oid relationOid, int options) ...@@ -3096,7 +3150,7 @@ ReindexRelationConcurrently(Oid relationOid, int options)
* Drop the old indexes. * Drop the old indexes.
*/ */
WaitForLockersMultiple(lockTags, AccessExclusiveLock); WaitForLockersMultiple(lockTags, AccessExclusiveLock, false);
PushActiveSnapshot(GetTransactionSnapshot()); PushActiveSnapshot(GetTransactionSnapshot());
......
...@@ -401,7 +401,7 @@ ResolveRecoveryConflictWithLock(LOCKTAG locktag) ...@@ -401,7 +401,7 @@ ResolveRecoveryConflictWithLock(LOCKTAG locktag)
*/ */
VirtualTransactionId *backends; VirtualTransactionId *backends;
backends = GetLockConflicts(&locktag, AccessExclusiveLock); backends = GetLockConflicts(&locktag, AccessExclusiveLock, NULL);
ResolveRecoveryConflictWithVirtualXIDs(backends, ResolveRecoveryConflictWithVirtualXIDs(backends,
PROCSIG_RECOVERY_CONFLICT_LOCK); PROCSIG_RECOVERY_CONFLICT_LOCK);
} }
......
...@@ -19,9 +19,12 @@ ...@@ -19,9 +19,12 @@
#include "access/transam.h" #include "access/transam.h"
#include "access/xact.h" #include "access/xact.h"
#include "catalog/catalog.h" #include "catalog/catalog.h"
#include "commands/progress.h"
#include "miscadmin.h" #include "miscadmin.h"
#include "pgstat.h"
#include "storage/lmgr.h" #include "storage/lmgr.h"
#include "storage/procarray.h" #include "storage/procarray.h"
#include "storage/sinvaladt.h"
#include "utils/inval.h" #include "utils/inval.h"
...@@ -857,10 +860,12 @@ XactLockTableWaitErrorCb(void *arg) ...@@ -857,10 +860,12 @@ XactLockTableWaitErrorCb(void *arg)
* after we obtained our initial list of lockers, we will not wait for them. * after we obtained our initial list of lockers, we will not wait for them.
*/ */
void void
WaitForLockersMultiple(List *locktags, LOCKMODE lockmode) WaitForLockersMultiple(List *locktags, LOCKMODE lockmode, bool progress)
{ {
List *holders = NIL; List *holders = NIL;
ListCell *lc; ListCell *lc;
int total = 0;
int done = 0;
/* Done if no locks to wait for */ /* Done if no locks to wait for */
if (list_length(locktags) == 0) if (list_length(locktags) == 0)
...@@ -870,10 +875,17 @@ WaitForLockersMultiple(List *locktags, LOCKMODE lockmode) ...@@ -870,10 +875,17 @@ WaitForLockersMultiple(List *locktags, LOCKMODE lockmode)
foreach(lc, locktags) foreach(lc, locktags)
{ {
LOCKTAG *locktag = lfirst(lc); LOCKTAG *locktag = lfirst(lc);
int count;
holders = lappend(holders, GetLockConflicts(locktag, lockmode)); holders = lappend(holders,
GetLockConflicts(locktag, lockmode,
progress ? &count : NULL));
total += count;
} }
if (progress)
pgstat_progress_update_param(PROGRESS_WAITFOR_TOTAL, total);
/* /*
* Note: GetLockConflicts() never reports our own xid, hence we need not * Note: GetLockConflicts() never reports our own xid, hence we need not
* check for that. Also, prepared xacts are not reported, which is fine * check for that. Also, prepared xacts are not reported, which is fine
...@@ -887,10 +899,36 @@ WaitForLockersMultiple(List *locktags, LOCKMODE lockmode) ...@@ -887,10 +899,36 @@ WaitForLockersMultiple(List *locktags, LOCKMODE lockmode)
while (VirtualTransactionIdIsValid(*lockholders)) while (VirtualTransactionIdIsValid(*lockholders))
{ {
/*
* If requested, publish who we're going to wait for. This is not
* 100% accurate if they're already gone, but we don't care.
*/
if (progress)
{
PGPROC *holder = BackendIdGetProc(lockholders->backendId);
pgstat_progress_update_param(PROGRESS_WAITFOR_CURRENT_PID,
holder->pid);
}
VirtualXactLock(*lockholders, true); VirtualXactLock(*lockholders, true);
lockholders++; lockholders++;
if (progress)
pgstat_progress_update_param(PROGRESS_WAITFOR_DONE, ++done);
} }
} }
if (progress)
{
const int index[] = {
PROGRESS_WAITFOR_TOTAL,
PROGRESS_WAITFOR_DONE,
PROGRESS_WAITFOR_CURRENT_PID
};
const int64 values[] = {
0, 0, 0
};
pgstat_progress_update_multi_param(3, index, values);
}
list_free_deep(holders); list_free_deep(holders);
} }
...@@ -901,12 +939,12 @@ WaitForLockersMultiple(List *locktags, LOCKMODE lockmode) ...@@ -901,12 +939,12 @@ WaitForLockersMultiple(List *locktags, LOCKMODE lockmode)
* Same as WaitForLockersMultiple, for a single lock tag. * Same as WaitForLockersMultiple, for a single lock tag.
*/ */
void void
WaitForLockers(LOCKTAG heaplocktag, LOCKMODE lockmode) WaitForLockers(LOCKTAG heaplocktag, LOCKMODE lockmode, bool progress)
{ {
List *l; List *l;
l = list_make1(&heaplocktag); l = list_make1(&heaplocktag);
WaitForLockersMultiple(l, lockmode); WaitForLockersMultiple(l, lockmode, progress);
list_free(l); list_free(l);
} }
......
...@@ -2807,6 +2807,7 @@ FastPathGetRelationLockEntry(LOCALLOCK *locallock) ...@@ -2807,6 +2807,7 @@ FastPathGetRelationLockEntry(LOCALLOCK *locallock)
* xacts merely awaiting such a lock are NOT reported. * xacts merely awaiting such a lock are NOT reported.
* *
* The result array is palloc'd and is terminated with an invalid VXID. * The result array is palloc'd and is terminated with an invalid VXID.
* *countp, if not null, is updated to the number of items set.
* *
* Of course, the result could be out of date by the time it's returned, * Of course, the result could be out of date by the time it's returned,
* so use of this function has to be thought about carefully. * so use of this function has to be thought about carefully.
...@@ -2817,7 +2818,7 @@ FastPathGetRelationLockEntry(LOCALLOCK *locallock) ...@@ -2817,7 +2818,7 @@ FastPathGetRelationLockEntry(LOCALLOCK *locallock)
* uses of the result. * uses of the result.
*/ */
VirtualTransactionId * VirtualTransactionId *
GetLockConflicts(const LOCKTAG *locktag, LOCKMODE lockmode) GetLockConflicts(const LOCKTAG *locktag, LOCKMODE lockmode, int *countp)
{ {
static VirtualTransactionId *vxids; static VirtualTransactionId *vxids;
LOCKMETHODID lockmethodid = locktag->locktag_lockmethodid; LOCKMETHODID lockmethodid = locktag->locktag_lockmethodid;
...@@ -2964,6 +2965,8 @@ GetLockConflicts(const LOCKTAG *locktag, LOCKMODE lockmode) ...@@ -2964,6 +2965,8 @@ GetLockConflicts(const LOCKTAG *locktag, LOCKMODE lockmode)
LWLockRelease(partitionLock); LWLockRelease(partitionLock);
vxids[count].backendId = InvalidBackendId; vxids[count].backendId = InvalidBackendId;
vxids[count].localTransactionId = InvalidLocalTransactionId; vxids[count].localTransactionId = InvalidLocalTransactionId;
if (countp)
*countp = count;
return vxids; return vxids;
} }
...@@ -3019,6 +3022,8 @@ GetLockConflicts(const LOCKTAG *locktag, LOCKMODE lockmode) ...@@ -3019,6 +3022,8 @@ GetLockConflicts(const LOCKTAG *locktag, LOCKMODE lockmode)
vxids[count].backendId = InvalidBackendId; vxids[count].backendId = InvalidBackendId;
vxids[count].localTransactionId = InvalidLocalTransactionId; vxids[count].localTransactionId = InvalidLocalTransactionId;
if (countp)
*countp = count;
return vxids; return vxids;
} }
......
...@@ -445,3 +445,26 @@ pg_index_column_has_property(PG_FUNCTION_ARGS) ...@@ -445,3 +445,26 @@ pg_index_column_has_property(PG_FUNCTION_ARGS)
return indexam_property(fcinfo, propname, InvalidOid, relid, attno); return indexam_property(fcinfo, propname, InvalidOid, relid, attno);
} }
/*
* Return the name of the given phase, as used for progress reporting by the
* given AM.
*/
Datum
pg_indexam_progress_phasename(PG_FUNCTION_ARGS)
{
Oid amoid = PG_GETARG_OID(0);
int32 phasenum = PG_GETARG_INT32(1);
IndexAmRoutine *routine;
char *name;
routine = GetIndexAmRoutineByAmId(amoid, true);
if (routine == NULL || !routine->ambuildphasename)
PG_RETURN_NULL();
name = routine->ambuildphasename(phasenum);
if (!name)
PG_RETURN_NULL();
PG_RETURN_TEXT_P(CStringGetTextDatum(name));
}
...@@ -470,6 +470,8 @@ pg_stat_get_progress_info(PG_FUNCTION_ARGS) ...@@ -470,6 +470,8 @@ pg_stat_get_progress_info(PG_FUNCTION_ARGS)
cmdtype = PROGRESS_COMMAND_VACUUM; cmdtype = PROGRESS_COMMAND_VACUUM;
else if (pg_strcasecmp(cmd, "CLUSTER") == 0) else if (pg_strcasecmp(cmd, "CLUSTER") == 0)
cmdtype = PROGRESS_COMMAND_CLUSTER; cmdtype = PROGRESS_COMMAND_CLUSTER;
else if (pg_strcasecmp(cmd, "CREATE INDEX") == 0)
cmdtype = PROGRESS_COMMAND_CREATE_INDEX;
else else
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
......
...@@ -108,6 +108,9 @@ typedef bool (*amproperty_function) (Oid index_oid, int attno, ...@@ -108,6 +108,9 @@ typedef bool (*amproperty_function) (Oid index_oid, int attno,
IndexAMProperty prop, const char *propname, IndexAMProperty prop, const char *propname,
bool *res, bool *isnull); bool *res, bool *isnull);
/* name of phase as used in progress reporting */
typedef char *(*ambuildphasename_function) (int64 phasenum);
/* validate definition of an opclass for this AM */ /* validate definition of an opclass for this AM */
typedef bool (*amvalidate_function) (Oid opclassoid); typedef bool (*amvalidate_function) (Oid opclassoid);
...@@ -213,6 +216,7 @@ typedef struct IndexAmRoutine ...@@ -213,6 +216,7 @@ typedef struct IndexAmRoutine
amcostestimate_function amcostestimate; amcostestimate_function amcostestimate;
amoptions_function amoptions; amoptions_function amoptions;
amproperty_function amproperty; /* can be NULL */ amproperty_function amproperty; /* can be NULL */
ambuildphasename_function ambuildphasename; /* can be NULL */
amvalidate_function amvalidate; amvalidate_function amvalidate;
ambeginscan_function ambeginscan; ambeginscan_function ambeginscan;
amrescan_function amrescan; amrescan_function amrescan;
......
...@@ -45,6 +45,7 @@ typedef struct IndexVacuumInfo ...@@ -45,6 +45,7 @@ typedef struct IndexVacuumInfo
{ {
Relation index; /* the index being vacuumed */ Relation index; /* the index being vacuumed */
bool analyze_only; /* ANALYZE (without any actual vacuum) */ bool analyze_only; /* ANALYZE (without any actual vacuum) */
bool report_progress; /* emit progress.h status reports */
bool estimated_count; /* num_heap_tuples is an estimate */ bool estimated_count; /* num_heap_tuples is an estimate */
int message_level; /* ereport level for progress messages */ int message_level; /* ereport level for progress messages */
double num_heap_tuples; /* tuples remaining in heap */ double num_heap_tuples; /* tuples remaining in heap */
......
...@@ -671,6 +671,16 @@ typedef BTScanOpaqueData *BTScanOpaque; ...@@ -671,6 +671,16 @@ typedef BTScanOpaqueData *BTScanOpaque;
#define SK_BT_DESC (INDOPTION_DESC << SK_BT_INDOPTION_SHIFT) #define SK_BT_DESC (INDOPTION_DESC << SK_BT_INDOPTION_SHIFT)
#define SK_BT_NULLS_FIRST (INDOPTION_NULLS_FIRST << SK_BT_INDOPTION_SHIFT) #define SK_BT_NULLS_FIRST (INDOPTION_NULLS_FIRST << SK_BT_INDOPTION_SHIFT)
/*
* Constant definition for progress reporting. Phase numbers must match
* btbuildphasename.
*/
/* PROGRESS_CREATEIDX_SUBPHASE_INITIALIZE is 1 (see progress.h) */
#define PROGRESS_BTREE_PHASE_INDEXBUILD_TABLESCAN 2
#define PROGRESS_BTREE_PHASE_PERFORMSORT_1 3
#define PROGRESS_BTREE_PHASE_PERFORMSORT_2 4
#define PROGRESS_BTREE_PHASE_LEAF_LOAD 5
/* /*
* external entry points for btree, in nbtree.c * external entry points for btree, in nbtree.c
*/ */
...@@ -784,6 +794,7 @@ extern bytea *btoptions(Datum reloptions, bool validate); ...@@ -784,6 +794,7 @@ extern bytea *btoptions(Datum reloptions, bool validate);
extern bool btproperty(Oid index_oid, int attno, extern bool btproperty(Oid index_oid, int attno,
IndexAMProperty prop, const char *propname, IndexAMProperty prop, const char *propname,
bool *res, bool *isnull); bool *res, bool *isnull);
extern char *btbuildphasename(int64 phasenum);
extern IndexTuple _bt_truncate(Relation rel, IndexTuple lastleft, extern IndexTuple _bt_truncate(Relation rel, IndexTuple lastleft,
IndexTuple firstright, BTScanInsert itup_key); IndexTuple firstright, BTScanInsert itup_key);
extern int _bt_keep_natts_fast(Relation rel, IndexTuple lastleft, extern int _bt_keep_natts_fast(Relation rel, IndexTuple lastleft,
......
...@@ -507,6 +507,7 @@ typedef struct TableAmRoutine ...@@ -507,6 +507,7 @@ typedef struct TableAmRoutine
struct IndexInfo *index_nfo, struct IndexInfo *index_nfo,
bool allow_sync, bool allow_sync,
bool anyvisible, bool anyvisible,
bool progress,
BlockNumber start_blockno, BlockNumber start_blockno,
BlockNumber end_blockno, BlockNumber end_blockno,
IndexBuildCallback callback, IndexBuildCallback callback,
...@@ -1369,6 +1370,8 @@ table_scan_analyze_next_tuple(TableScanDesc scan, TransactionId OldestXmin, ...@@ -1369,6 +1370,8 @@ table_scan_analyze_next_tuple(TableScanDesc scan, TransactionId OldestXmin,
* so here because the AM might reject some of the tuples for its own reasons, * so here because the AM might reject some of the tuples for its own reasons,
* such as being unable to store NULLs. * such as being unable to store NULLs.
* *
* If 'progress', the PROGRESS_SCAN_BLOCKS_TOTAL counter is updated when
* starting the scan, and PROGRESS_SCAN_BLOCKS_DONE is updated as we go along.
* *
* A side effect is to set indexInfo->ii_BrokenHotChain to true if we detect * A side effect is to set indexInfo->ii_BrokenHotChain to true if we detect
* any potentially broken HOT chains. Currently, we set this if there are any * any potentially broken HOT chains. Currently, we set this if there are any
...@@ -1382,6 +1385,7 @@ table_index_build_scan(Relation heap_rel, ...@@ -1382,6 +1385,7 @@ table_index_build_scan(Relation heap_rel,
Relation index_rel, Relation index_rel,
struct IndexInfo *index_nfo, struct IndexInfo *index_nfo,
bool allow_sync, bool allow_sync,
bool progress,
IndexBuildCallback callback, IndexBuildCallback callback,
void *callback_state, void *callback_state,
TableScanDesc scan) TableScanDesc scan)
...@@ -1391,6 +1395,7 @@ table_index_build_scan(Relation heap_rel, ...@@ -1391,6 +1395,7 @@ table_index_build_scan(Relation heap_rel,
index_nfo, index_nfo,
allow_sync, allow_sync,
false, false,
progress,
0, 0,
InvalidBlockNumber, InvalidBlockNumber,
callback, callback,
...@@ -1414,6 +1419,7 @@ table_index_build_range_scan(Relation heap_rel, ...@@ -1414,6 +1419,7 @@ table_index_build_range_scan(Relation heap_rel,
struct IndexInfo *index_nfo, struct IndexInfo *index_nfo,
bool allow_sync, bool allow_sync,
bool anyvisible, bool anyvisible,
bool progress,
BlockNumber start_blockno, BlockNumber start_blockno,
BlockNumber numblocks, BlockNumber numblocks,
IndexBuildCallback callback, IndexBuildCallback callback,
...@@ -1425,6 +1431,7 @@ table_index_build_range_scan(Relation heap_rel, ...@@ -1425,6 +1431,7 @@ table_index_build_range_scan(Relation heap_rel,
index_nfo, index_nfo,
allow_sync, allow_sync,
anyvisible, anyvisible,
progress,
start_blockno, start_blockno,
numblocks, numblocks,
callback, callback,
......
...@@ -53,6 +53,6 @@ ...@@ -53,6 +53,6 @@
*/ */
/* yyyymmddN */ /* yyyymmddN */
#define CATALOG_VERSION_NO 201904011 #define CATALOG_VERSION_NO 201904021
#endif #endif
...@@ -924,6 +924,10 @@ ...@@ -924,6 +924,10 @@
proname => 'pg_index_column_has_property', provolatile => 's', proname => 'pg_index_column_has_property', provolatile => 's',
prorettype => 'bool', proargtypes => 'regclass int4 text', prorettype => 'bool', proargtypes => 'regclass int4 text',
prosrc => 'pg_index_column_has_property' }, prosrc => 'pg_index_column_has_property' },
{ oid => '676', descr => 'return name of given index build phase',
proname => 'pg_indexam_progress_phasename', provolatile => 'i',
prorettype => 'text', proargtypes => 'oid int8',
prosrc => 'pg_indexam_progress_phasename' },
{ oid => '339', { oid => '339',
proname => 'poly_same', prorettype => 'bool', proname => 'poly_same', prorettype => 'bool',
...@@ -5122,9 +5126,9 @@ ...@@ -5122,9 +5126,9 @@
proname => 'pg_stat_get_progress_info', prorows => '100', proretset => 't', proname => 'pg_stat_get_progress_info', prorows => '100', proretset => 't',
provolatile => 's', proparallel => 'r', prorettype => 'record', provolatile => 's', proparallel => 'r', prorettype => 'record',
proargtypes => 'text', proargtypes => 'text',
proallargtypes => '{text,int4,oid,oid,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8}', proallargtypes => '{text,int4,oid,oid,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8,int8}',
proargmodes => '{i,o,o,o,o,o,o,o,o,o,o,o,o,o}', proargmodes => '{i,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o,o}',
proargnames => '{cmdtype,pid,datid,relid,param1,param2,param3,param4,param5,param6,param7,param8,param9,param10}', proargnames => '{cmdtype,pid,datid,relid,param1,param2,param3,param4,param5,param6,param7,param8,param9,param10,param11,param12,param13,param14,param15,param16,param17,param18,param19,param20}',
prosrc => 'pg_stat_get_progress_info' }, prosrc => 'pg_stat_get_progress_info' },
{ oid => '3099', { oid => '3099',
descr => 'statistics: information about currently active replication', descr => 'statistics: information about currently active replication',
......
...@@ -44,7 +44,7 @@ ...@@ -44,7 +44,7 @@
#define PROGRESS_CLUSTER_HEAP_BLKS_SCANNED 6 #define PROGRESS_CLUSTER_HEAP_BLKS_SCANNED 6
#define PROGRESS_CLUSTER_INDEX_REBUILD_COUNT 7 #define PROGRESS_CLUSTER_INDEX_REBUILD_COUNT 7
/* Phases of cluster (as dvertised via PROGRESS_CLUSTER_PHASE) */ /* Phases of cluster (as advertised via PROGRESS_CLUSTER_PHASE) */
#define PROGRESS_CLUSTER_PHASE_SEQ_SCAN_HEAP 1 #define PROGRESS_CLUSTER_PHASE_SEQ_SCAN_HEAP 1
#define PROGRESS_CLUSTER_PHASE_INDEX_SCAN_HEAP 2 #define PROGRESS_CLUSTER_PHASE_INDEX_SCAN_HEAP 2
#define PROGRESS_CLUSTER_PHASE_SORT_TUPLES 3 #define PROGRESS_CLUSTER_PHASE_SORT_TUPLES 3
...@@ -57,4 +57,39 @@ ...@@ -57,4 +57,39 @@
#define PROGRESS_CLUSTER_COMMAND_CLUSTER 1 #define PROGRESS_CLUSTER_COMMAND_CLUSTER 1
#define PROGRESS_CLUSTER_COMMAND_VACUUM_FULL 2 #define PROGRESS_CLUSTER_COMMAND_VACUUM_FULL 2
/* Progress parameters for CREATE INDEX */
/* 3, 4 and 5 reserved for "waitfor" metrics */
#define PROGRESS_CREATEIDX_ACCESS_METHOD_OID 8
#define PROGRESS_CREATEIDX_PHASE 9 /* AM-agnostic phase # */
#define PROGRESS_CREATEIDX_SUBPHASE 10 /* phase # filled by AM */
#define PROGRESS_CREATEIDX_TUPLES_TOTAL 11
#define PROGRESS_CREATEIDX_TUPLES_DONE 12
#define PROGRESS_CREATEIDX_PARTITIONS_TOTAL 13
#define PROGRESS_CREATEIDX_PARTITIONS_DONE 14
/* 15 and 16 reserved for "block number" metrics */
/* Phases of CREATE INDEX (as advertised via PROGRESS_CREATEIDX_PHASE) */
#define PROGRESS_CREATEIDX_PHASE_WAIT_1 1
#define PROGRESS_CREATEIDX_PHASE_BUILD 2
#define PROGRESS_CREATEIDX_PHASE_WAIT_2 3
#define PROGRESS_CREATEIDX_PHASE_VALIDATE_IDXSCAN 4
#define PROGRESS_CREATEIDX_PHASE_VALIDATE_SORT 5
#define PROGRESS_CREATEIDX_PHASE_VALIDATE_TABLESCAN 6
#define PROGRESS_CREATEIDX_PHASE_WAIT_3 7
/*
* Subphases of CREATE INDEX, for index_build.
*/
#define PROGRESS_CREATEIDX_SUBPHASE_INITIALIZE 1
/* Additional phases are defined by each AM */
/* Lock holder wait counts */
#define PROGRESS_WAITFOR_TOTAL 3
#define PROGRESS_WAITFOR_DONE 4
#define PROGRESS_WAITFOR_CURRENT_PID 5
/* Block numbers in a generic relation scan */
#define PROGRESS_SCAN_BLOCKS_TOTAL 15
#define PROGRESS_SCAN_BLOCKS_DONE 16
#endif #endif
...@@ -951,10 +951,11 @@ typedef enum ProgressCommandType ...@@ -951,10 +951,11 @@ typedef enum ProgressCommandType
{ {
PROGRESS_COMMAND_INVALID, PROGRESS_COMMAND_INVALID,
PROGRESS_COMMAND_VACUUM, PROGRESS_COMMAND_VACUUM,
PROGRESS_COMMAND_CLUSTER PROGRESS_COMMAND_CLUSTER,
PROGRESS_COMMAND_CREATE_INDEX
} ProgressCommandType; } ProgressCommandType;
#define PGSTAT_NUM_PROGRESS_PARAM 10 #define PGSTAT_NUM_PROGRESS_PARAM 20
/* ---------- /* ----------
* Shared-memory data structures * Shared-memory data structures
......
...@@ -78,8 +78,8 @@ extern void XactLockTableWait(TransactionId xid, Relation rel, ...@@ -78,8 +78,8 @@ extern void XactLockTableWait(TransactionId xid, Relation rel,
extern bool ConditionalXactLockTableWait(TransactionId xid); extern bool ConditionalXactLockTableWait(TransactionId xid);
/* Lock VXIDs, specified by conflicting locktags */ /* Lock VXIDs, specified by conflicting locktags */
extern void WaitForLockers(LOCKTAG heaplocktag, LOCKMODE lockmode); extern void WaitForLockers(LOCKTAG heaplocktag, LOCKMODE lockmode, bool progress);
extern void WaitForLockersMultiple(List *locktags, LOCKMODE lockmode); extern void WaitForLockersMultiple(List *locktags, LOCKMODE lockmode, bool progress);
/* Lock an XID for tuple insertion (used to wait for an insertion to finish) */ /* Lock an XID for tuple insertion (used to wait for an insertion to finish) */
extern uint32 SpeculativeInsertionLockAcquire(TransactionId xid); extern uint32 SpeculativeInsertionLockAcquire(TransactionId xid);
......
...@@ -544,7 +544,7 @@ extern bool LockHeldByMe(const LOCKTAG *locktag, LOCKMODE lockmode); ...@@ -544,7 +544,7 @@ extern bool LockHeldByMe(const LOCKTAG *locktag, LOCKMODE lockmode);
extern bool LockHasWaiters(const LOCKTAG *locktag, extern bool LockHasWaiters(const LOCKTAG *locktag,
LOCKMODE lockmode, bool sessionLock); LOCKMODE lockmode, bool sessionLock);
extern VirtualTransactionId *GetLockConflicts(const LOCKTAG *locktag, extern VirtualTransactionId *GetLockConflicts(const LOCKTAG *locktag,
LOCKMODE lockmode); LOCKMODE lockmode, int *countp);
extern void AtPrepare_Locks(void); extern void AtPrepare_Locks(void);
extern void PostPrepare_Locks(TransactionId xid); extern void PostPrepare_Locks(TransactionId xid);
extern int LockCheckConflicts(LockMethod lockMethodTable, extern int LockCheckConflicts(LockMethod lockMethodTable,
......
...@@ -1856,7 +1856,33 @@ pg_stat_progress_cluster| SELECT s.pid, ...@@ -1856,7 +1856,33 @@ pg_stat_progress_cluster| SELECT s.pid,
s.param6 AS heap_blks_total, s.param6 AS heap_blks_total,
s.param7 AS heap_blks_scanned, s.param7 AS heap_blks_scanned,
s.param8 AS index_rebuild_count s.param8 AS index_rebuild_count
FROM (pg_stat_get_progress_info('CLUSTER'::text) s(pid, datid, relid, param1, param2, param3, param4, param5, param6, param7, param8, param9, param10) FROM (pg_stat_get_progress_info('CLUSTER'::text) s(pid, datid, relid, param1, param2, param3, param4, param5, param6, param7, param8, param9, param10, param11, param12, param13, param14, param15, param16, param17, param18, param19, param20)
LEFT JOIN pg_database d ON ((s.datid = d.oid)));
pg_stat_progress_create_index| SELECT s.pid,
s.datid,
d.datname,
s.relid,
CASE s.param10
WHEN 0 THEN 'initializing'::text
WHEN 1 THEN 'waiting for old snapshots'::text
WHEN 2 THEN ('building index'::text || COALESCE((': '::text || pg_indexam_progress_phasename((s.param9)::oid, s.param11)), ''::text))
WHEN 3 THEN 'waiting for writer snapshots'::text
WHEN 4 THEN 'index validation: scan index'::text
WHEN 5 THEN 'index validation: sort index scan results'::text
WHEN 6 THEN 'index validation: scan heap'::text
WHEN 7 THEN 'waiting for reader snapshots'::text
ELSE NULL::text
END AS phase,
s.param4 AS lockers_total,
s.param5 AS lockers_done,
s.param6 AS current_locker_pid,
s.param16 AS blocks_total,
s.param17 AS blocks_done,
s.param12 AS tuples_total,
s.param13 AS tuples_done,
s.param14 AS partitions_total,
s.param15 AS partitions_done
FROM (pg_stat_get_progress_info('CREATE INDEX'::text) s(pid, datid, relid, param1, param2, param3, param4, param5, param6, param7, param8, param9, param10, param11, param12, param13, param14, param15, param16, param17, param18, param19, param20)
LEFT JOIN pg_database d ON ((s.datid = d.oid))); LEFT JOIN pg_database d ON ((s.datid = d.oid)));
pg_stat_progress_vacuum| SELECT s.pid, pg_stat_progress_vacuum| SELECT s.pid,
s.datid, s.datid,
...@@ -1878,7 +1904,7 @@ pg_stat_progress_vacuum| SELECT s.pid, ...@@ -1878,7 +1904,7 @@ pg_stat_progress_vacuum| SELECT s.pid,
s.param5 AS index_vacuum_count, s.param5 AS index_vacuum_count,
s.param6 AS max_dead_tuples, s.param6 AS max_dead_tuples,
s.param7 AS num_dead_tuples s.param7 AS num_dead_tuples
FROM (pg_stat_get_progress_info('VACUUM'::text) s(pid, datid, relid, param1, param2, param3, param4, param5, param6, param7, param8, param9, param10) FROM (pg_stat_get_progress_info('VACUUM'::text) s(pid, datid, relid, param1, param2, param3, param4, param5, param6, param7, param8, param9, param10, param11, param12, param13, param14, param15, param16, param17, param18, param19, param20)
LEFT JOIN pg_database d ON ((s.datid = d.oid))); LEFT JOIN pg_database d ON ((s.datid = d.oid)));
pg_stat_replication| SELECT s.pid, pg_stat_replication| SELECT s.pid,
s.usesysid, s.usesysid,
......
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