mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Show index search count in EXPLAIN ANALYZE, take 2.
Expose the count of index searches/index descents in EXPLAIN ANALYZE's output for index scan/index-only scan/bitmap index scan nodes. This information is particularly useful with scans that use ScalarArrayOp quals, where the number of index searches can be unpredictable due to implementation details that interact with physical index characteristics (at least with nbtree SAOP scans, since Postgres 17 commit5bf748b8). The information shown also provides useful context when EXPLAIN ANALYZE runs a plan with an index scan node that successfully applied the skip scan optimization (set to be added to nbtree by an upcoming patch). The instrumentation works by teaching all index AMs to increment a new nsearches counter whenever a new index search begins. The counter is incremented at exactly the same point that index AMs already increment the pg_stat_*_indexes.idx_scan counter (we're counting the same event, but at the scan level rather than the relation level). Parallel queries have workers copy their local counter struct into shared memory when an index scan node ends -- even when it isn't a parallel aware scan node. An earlier version of this patch that only worked with parallel aware scans became commit5ead85fb(though that was quickly reverted by commitd00107cdfollowing "debug_parallel_query=regress" buildfarm failures). Our approach doesn't match the approach used when tracking other index scan related costs (e.g., "Rows Removed by Filter:"). It is comparable to the approach used in similar cases involving costs that are only readily accessible inside an access method, not from the executor proper (e.g., "Heap Blocks:" output for a Bitmap Heap Scan, which was recently enhanced to show per-worker costs by commit5a1e6df3, using essentially the same scheme as the one used here). It is necessary for index AMs to have direct responsibility for maintaining the new counter, since the counter might need to be incremented multiple times per amgettuple call (or per amgetbitmap call). But it is also necessary for the executor proper to manage the shared memory now used to transfer each worker's counter struct to the leader. Author: Peter Geoghegan <pg@bowt.ie> Reviewed-By: Robert Haas <robertmhaas@gmail.com> Reviewed-By: Tomas Vondra <tomas@vondra.me> Reviewed-By: Masahiro Ikeda <ikedamsh@oss.nttdata.com> Reviewed-By: Matthias van de Meent <boekewurm+postgres@gmail.com> Discussion: https://postgr.es/m/CAH2-WzkRqvaqR2CTNqTZP0z6FuL4-3ED6eQB0yx38XBNj1v-4Q@mail.gmail.com Discussion: https://postgr.es/m/CAH2-Wz=PKR6rB7qbx+Vnd7eqeB5VTcrW=iJvAsTsKbdG+kW_UA@mail.gmail.com
This commit is contained in:
		| @@ -125,6 +125,7 @@ static void show_recursive_union_info(RecursiveUnionState *rstate, | ||||
| static void show_memoize_info(MemoizeState *mstate, List *ancestors, | ||||
| 							  ExplainState *es); | ||||
| static void show_hashagg_info(AggState *aggstate, ExplainState *es); | ||||
| static void show_indexsearches_info(PlanState *planstate, ExplainState *es); | ||||
| static void show_tidbitmap_info(BitmapHeapScanState *planstate, | ||||
| 								ExplainState *es); | ||||
| static void show_instrumentation_count(const char *qlabel, int which, | ||||
| @@ -2096,6 +2097,7 @@ ExplainNode(PlanState *planstate, List *ancestors, | ||||
| 			if (plan->qual) | ||||
| 				show_instrumentation_count("Rows Removed by Filter", 1, | ||||
| 										   planstate, es); | ||||
| 			show_indexsearches_info(planstate, es); | ||||
| 			break; | ||||
| 		case T_IndexOnlyScan: | ||||
| 			show_scan_qual(((IndexOnlyScan *) plan)->indexqual, | ||||
| @@ -2112,10 +2114,12 @@ ExplainNode(PlanState *planstate, List *ancestors, | ||||
| 			if (es->analyze) | ||||
| 				ExplainPropertyFloat("Heap Fetches", NULL, | ||||
| 									 planstate->instrument->ntuples2, 0, es); | ||||
| 			show_indexsearches_info(planstate, es); | ||||
| 			break; | ||||
| 		case T_BitmapIndexScan: | ||||
| 			show_scan_qual(((BitmapIndexScan *) plan)->indexqualorig, | ||||
| 						   "Index Cond", planstate, ancestors, es); | ||||
| 			show_indexsearches_info(planstate, es); | ||||
| 			break; | ||||
| 		case T_BitmapHeapScan: | ||||
| 			show_scan_qual(((BitmapHeapScan *) plan)->bitmapqualorig, | ||||
| @@ -3855,6 +3859,65 @@ show_hashagg_info(AggState *aggstate, ExplainState *es) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Show the total number of index searches for a | ||||
|  * IndexScan/IndexOnlyScan/BitmapIndexScan node | ||||
|  */ | ||||
| static void | ||||
| show_indexsearches_info(PlanState *planstate, ExplainState *es) | ||||
| { | ||||
| 	Plan	   *plan = planstate->plan; | ||||
| 	SharedIndexScanInstrumentation *SharedInfo = NULL; | ||||
| 	uint64		nsearches = 0; | ||||
|  | ||||
| 	if (!es->analyze) | ||||
| 		return; | ||||
|  | ||||
| 	/* Initialize counters with stats from the local process first */ | ||||
| 	switch (nodeTag(plan)) | ||||
| 	{ | ||||
| 		case T_IndexScan: | ||||
| 			{ | ||||
| 				IndexScanState *indexstate = ((IndexScanState *) planstate); | ||||
|  | ||||
| 				nsearches = indexstate->iss_Instrument.nsearches; | ||||
| 				SharedInfo = indexstate->iss_SharedInfo; | ||||
| 				break; | ||||
| 			} | ||||
| 		case T_IndexOnlyScan: | ||||
| 			{ | ||||
| 				IndexOnlyScanState *indexstate = ((IndexOnlyScanState *) planstate); | ||||
|  | ||||
| 				nsearches = indexstate->ioss_Instrument.nsearches; | ||||
| 				SharedInfo = indexstate->ioss_SharedInfo; | ||||
| 				break; | ||||
| 			} | ||||
| 		case T_BitmapIndexScan: | ||||
| 			{ | ||||
| 				BitmapIndexScanState *indexstate = ((BitmapIndexScanState *) planstate); | ||||
|  | ||||
| 				nsearches = indexstate->biss_Instrument.nsearches; | ||||
| 				SharedInfo = indexstate->biss_SharedInfo; | ||||
| 				break; | ||||
| 			} | ||||
| 		default: | ||||
| 			break; | ||||
| 	} | ||||
|  | ||||
| 	/* Next get the sum of the counters set within each and every process */ | ||||
| 	if (SharedInfo) | ||||
| 	{ | ||||
| 		for (int i = 0; i < SharedInfo->num_workers; ++i) | ||||
| 		{ | ||||
| 			IndexScanInstrumentation *winstrument = &SharedInfo->winstrument[i]; | ||||
|  | ||||
| 			nsearches += winstrument->nsearches; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	ExplainPropertyUInteger("Index Searches", NULL, nsearches, es); | ||||
| } | ||||
|  | ||||
| /* | ||||
|  * Show exact/lossy pages for a BitmapHeapScan node | ||||
|  */ | ||||
|   | ||||
		Reference in New Issue
	
	Block a user